/******************************************************************************
* Product: Adempiere ERP & CRM Smart Business Solution *
* Copyright (C) 1999-2006 ComPiere, Inc. All Rights Reserved. *
* This program is free software; you can redistribute it and/or modify it *
* under the terms version 2 of the GNU General Public License as published *
* by the Free Software Foundation. This program is distributed in the hope *
* that it will be useful, but WITHOUT ANY WARRANTY; without even the implied *
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. *
* See the GNU General Public License for more details. *
* You should have received a copy of the GNU General Public License along *
* with this program; if not, write to the Free Software Foundation, Inc., *
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. *
* For the text or an alternative of this public license, you may reach us *
* ComPiere, Inc., 2620 Augustine Dr. #245, Santa Clara, CA 95054, USA *
* or via info@compiere.org or http://www.compiere.org/license.html *
*****************************************************************************/
package org.compiere.model;
import java.math.BigDecimal;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
import java.util.logging.Level;
import org.adempiere.exceptions.DBException;
import org.compiere.util.CLogger;
import org.compiere.util.DB;
import org.compiere.util.Env;
import org.compiere.util.Util;
/**
* Invoice Tax Model
*
* @author Jorg Janke
* @version $Id: MInvoiceTax.java,v 1.5 2006/10/06 00:42:24 jjanke Exp $
*
* @author Teo Sarca, www.arhipac.ro
*
FR [ 2214883 ] Remove SQL code and Replace for Query
*/
public class MInvoiceTax extends X_C_InvoiceTax
{
/**
* generated serial id
*/
private static final long serialVersionUID = -5560880305482497098L;
/**
* Get Tax Line for Invoice Line
* @param line invoice line
* @param precision currency precision
* @param oldTax if true old tax is returned
* @param trxName transaction name
* @return existing or new tax
*/
public static MInvoiceTax get (MInvoiceLine line, int precision,
boolean oldTax, String trxName)
{
MInvoiceTax retValue = null;
if (line == null || line.getC_Invoice_ID() == 0)
return null;
int C_Tax_ID = line.getC_Tax_ID();
boolean isOldTax = oldTax && line.is_ValueChanged(MInvoiceLine.COLUMNNAME_C_Tax_ID);
if (isOldTax)
{
Object old = line.get_ValueOld(MInvoiceLine.COLUMNNAME_C_Tax_ID);
if (old == null)
return null;
C_Tax_ID = ((Integer)old).intValue();
}
if (C_Tax_ID == 0)
{
if (!line.isDescription())
s_log.warning("C_Tax_ID=0");
return null;
}
retValue = new Query(line.getCtx(), Table_Name, "C_Invoice_ID=? AND C_Tax_ID=?", trxName)
.setParameters(line.getC_Invoice_ID(), C_Tax_ID)
.firstOnly();
if (retValue != null)
{
retValue.set_TrxName(trxName);
retValue.setPrecision(precision);
if (s_log.isLoggable(Level.FINE)) s_log.fine("(old=" + oldTax + ") " + retValue);
return retValue;
}
// If the old tax was required and there is no MInvoiceTax for that
// return null, and not create another MInvoiceTax - teo_sarca [ 1583825 ]
else {
if (isOldTax)
return null;
}
// Create New
retValue = new MInvoiceTax(line.getCtx(), 0, trxName);
retValue.set_TrxName(trxName);
retValue.setClientOrg(line);
retValue.setC_Invoice_ID(line.getC_Invoice_ID());
retValue.setC_Tax_ID(line.getC_Tax_ID());
retValue.setPrecision(precision);
retValue.setIsTaxIncluded(line.isTaxIncluded());
if (s_log.isLoggable(Level.FINE)) s_log.fine("(new) " + retValue);
return retValue;
} // get
/**
* Get Child Tax Lines for Invoice Line
* @param line invoice line
* @param precision currency precision
* @param oldTax if true old tax is returned
* @param trxName transaction name
* @return array of existing or new tax
*/
public static MInvoiceTax[] getChildTaxes(MInvoiceLine line, int precision,
boolean oldTax, String trxName)
{
List invoiceTaxes = new ArrayList();
if (line == null || line.getC_Invoice_ID() == 0)
return invoiceTaxes.toArray(new MInvoiceTax[0]);
int C_Tax_ID = line.getC_Tax_ID();
if (oldTax)
{
Object old = line.get_ValueOld(MInvoiceLine.COLUMNNAME_C_Tax_ID);
if (old == null)
return invoiceTaxes.toArray(new MInvoiceTax[0]);
C_Tax_ID = ((Integer)old).intValue();
}
if (C_Tax_ID == 0)
{
return invoiceTaxes.toArray(new MInvoiceTax[0]);
}
MTax tax = MTax.get(C_Tax_ID);
if (!tax.isSummary())
return invoiceTaxes.toArray(new MInvoiceTax[0]);
MTax[] cTaxes = tax.getChildTaxes(false);
for(MTax cTax : cTaxes) {
MInvoiceTax invoiceTax = new Query(line.getCtx(), Table_Name, "C_Invoice_ID=? AND C_Tax_ID=?", trxName)
.setParameters(line.getC_Invoice_ID(), cTax.getC_Tax_ID())
.firstOnly();
if (invoiceTax != null)
{
invoiceTax.set_TrxName(trxName);
invoiceTax.setPrecision(precision);
invoiceTaxes.add(invoiceTax);
}
// If the old tax was required and there is no MInvoiceTax for that
// return null, and not create another MInvoiceTax - teo_sarca [ 1583825 ]
else
{
if (oldTax)
continue;
}
if (invoiceTax == null)
{
// Create New
invoiceTax = new MInvoiceTax(line.getCtx(), 0, trxName);
invoiceTax.set_TrxName(trxName);
invoiceTax.setClientOrg(line);
invoiceTax.setC_Invoice_ID(line.getC_Invoice_ID());
invoiceTax.setC_Tax_ID(cTax.getC_Tax_ID());
invoiceTax.setPrecision(precision);
invoiceTax.setIsTaxIncluded(line.isTaxIncluded());
invoiceTaxes.add(invoiceTax);
}
}
return invoiceTaxes.toArray(new MInvoiceTax[0]);
}
/** Static Logger */
private static CLogger s_log = CLogger.getCLogger (MInvoiceTax.class);
/**
* UUID based Constructor
* @param ctx Context
* @param C_InvoiceTax_UU UUID key
* @param trxName Transaction
*/
public MInvoiceTax(Properties ctx, String C_InvoiceTax_UU, String trxName) {
super(ctx, C_InvoiceTax_UU, trxName);
if (Util.isEmpty(C_InvoiceTax_UU))
setInitialDefaults();
}
/**
* @param ctx context
* @param ignored ignored
* @param trxName transaction
*/
public MInvoiceTax (Properties ctx, int ignored, String trxName)
{
super(ctx, 0, trxName);
if (ignored != 0)
throw new IllegalArgumentException("Multi-Key");
setInitialDefaults();
} // MInvoiceTax
/**
* Set the initial defaults for a new record
*/
private void setInitialDefaults() {
setTaxAmt (Env.ZERO);
setTaxBaseAmt (Env.ZERO);
setIsTaxIncluded(false);
}
/**
* Load Constructor.
* @param ctx context
* @param rs result set
* @param trxName transaction
*/
public MInvoiceTax (Properties ctx, ResultSet rs, String trxName)
{
super(ctx, rs, trxName);
} // MInvoiceTax
/**
* Copy constructor
* @param copy
*/
public MInvoiceTax(MInvoiceTax copy)
{
this(Env.getCtx(), copy);
}
/**
* Copy constructor
* @param ctx
* @param copy
*/
public MInvoiceTax(Properties ctx, MInvoiceTax copy)
{
this(ctx, copy, (String) null);
}
/**
* Copy constructor
* @param ctx
* @param copy
* @param trxName
*/
public MInvoiceTax(Properties ctx, MInvoiceTax copy, String trxName)
{
this(ctx, 0, trxName);
copyPO(copy);
this.m_precision = copy.m_precision;
this.m_tax = copy.m_tax != null ? new MTax(ctx, copy.m_tax, trxName) : null;
}
/** Tax */
private MTax m_tax = null;
/** Cached Precision */
private Integer m_precision = null;
/**
* Get Precision
* @return precision set or 2
*/
private int getPrecision ()
{
if (m_precision == null)
return 2;
return m_precision.intValue();
} // getPrecision
/**
* Set Precision
* @param precision The precision to set.
*/
protected void setPrecision (int precision)
{
m_precision = Integer.valueOf(precision);
} // setPrecision
/**
* Get Tax (immutable)
* @return tax
*/
protected MTax getTax()
{
if (m_tax == null)
m_tax = MTax.get(getCtx(), getC_Tax_ID());
return m_tax;
} // getTax
/**
* Calculate/Set Tax and Tax Base Amt from Invoice Lines
* @return true if tax calculated
*/
public boolean calculateTaxFromLines ()
{
BigDecimal taxBaseAmt = Env.ZERO;
BigDecimal taxAmt = Env.ZERO;
//
boolean documentLevel = getTax().isDocumentLevel();
MTax tax = getTax();
int parentTaxId = tax.getParent_Tax_ID();
//
String sql = "SELECT il.LineNetAmt, COALESCE(il.TaxAmt,0), i.IsSOTrx "
+ "FROM C_InvoiceLine il"
+ " INNER JOIN C_Invoice i ON (il.C_Invoice_ID=i.C_Invoice_ID) "
+ "WHERE il.C_Invoice_ID=? ";
if (parentTaxId > 0)
sql += "AND il.C_Tax_ID IN (?, ?) ";
else
sql += "AND il.C_Tax_ID=? ";
PreparedStatement pstmt = null;
ResultSet rs = null;
try
{
pstmt = DB.prepareStatement (sql, get_TrxName());
pstmt.setInt (1, getC_Invoice_ID());
pstmt.setInt (2, getC_Tax_ID());
if (parentTaxId > 0)
pstmt.setInt(3, parentTaxId);
rs = pstmt.executeQuery ();
while (rs.next ())
{
// BaseAmt
BigDecimal baseAmt = rs.getBigDecimal(1);
taxBaseAmt = taxBaseAmt.add(baseAmt);
// TaxAmt
BigDecimal amt = rs.getBigDecimal(2);
if (amt == null)
amt = Env.ZERO;
boolean isSOTrx = "Y".equals(rs.getString(3));
//
// phib [ 1702807 ]: manual tax should never be amended
// on line level taxes
if (!documentLevel && amt.signum() != 0 && !isSOTrx) // manually entered
;
else if (documentLevel || baseAmt.signum() == 0)
amt = Env.ZERO;
else // calculate line tax
amt = tax.calculateTax(baseAmt, isTaxIncluded(), getPrecision());
//
taxAmt = taxAmt.add(amt);
}
}
catch (SQLException e)
{
throw new DBException(e, sql);
}
finally
{
DB.close(rs, pstmt);
rs = null; pstmt = null;
}
// Calculate Tax
if (documentLevel || taxAmt.signum() == 0)
taxAmt = tax.calculateTax(taxBaseAmt, isTaxIncluded(), getPrecision());
setTaxAmt(taxAmt);
// Set Base
if (isTaxIncluded())
setTaxBaseAmt (taxBaseAmt.subtract(taxAmt));
else
setTaxBaseAmt (taxBaseAmt);
return true;
} // calculateTaxFromLines
/**
* String Representation
* @return info
*/
@Override
public String toString ()
{
StringBuilder sb = new StringBuilder ("MInvoiceTax[");
sb.append("C_Invoice_ID=").append(getC_Invoice_ID())
.append(",C_Tax_ID=").append(getC_Tax_ID())
.append(", Base=").append(getTaxBaseAmt()).append(",Tax=").append(getTaxAmt())
.append ("]");
return sb.toString ();
} // toString
} // MInvoiceTax