2015-03-03 09:25:30 +00:00
# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
2013-08-05 09:29:54 +00:00
# License: GNU General Public License v3. See license.txt
2013-01-29 06:04:39 +00:00
from __future__ import unicode_literals
2014-02-14 10:17:51 +00:00
import frappe
from frappe import _ , throw
2015-02-23 06:28:15 +00:00
from frappe . utils import today , flt , cint
2014-07-01 12:15:15 +00:00
from erpnext . setup . utils import get_company_currency , get_exchange_rate
2013-12-12 13:42:19 +00:00
from erpnext . accounts . utils import get_fiscal_year , validate_fiscal_year
2014-02-10 09:17:54 +00:00
from erpnext . utilities . transaction_base import TransactionBase
2014-09-21 14:15:49 +00:00
from erpnext . controllers . recurring_document import convert_to_recurring , validate_recurring_document
2013-01-29 06:04:39 +00:00
2013-01-30 07:19:08 +00:00
class AccountsController ( TransactionBase ) :
2013-03-20 07:25:28 +00:00
def validate ( self ) :
2014-07-03 06:55:06 +00:00
if self . get ( " _action " ) and self . _action != " update_after_submit " :
2014-06-05 11:25:31 +00:00
self . set_missing_values ( for_validate = True )
2013-07-11 12:19:18 +00:00
self . validate_date_with_fiscal_year ( )
2013-05-24 13:55:01 +00:00
if self . meta . get_field ( " currency " ) :
2013-05-28 11:53:36 +00:00
self . calculate_taxes_and_totals ( )
2015-02-12 10:39:11 +00:00
self . validate_value ( " base_grand_total " , " >= " , 0 )
2013-05-24 13:55:01 +00:00
self . set_total_in_words ( )
2014-04-08 14:40:03 +00:00
2014-08-27 11:16:33 +00:00
self . validate_due_date ( )
2014-04-08 14:40:03 +00:00
2014-09-21 14:15:49 +00:00
if self . meta . get_field ( " is_recurring " ) :
validate_recurring_document ( self )
2014-10-06 07:50:53 +00:00
if self . meta . get_field ( " taxes_and_charges " ) :
self . validate_enabled_taxes_and_charges ( )
2014-09-21 14:15:49 +00:00
def on_submit ( self ) :
if self . meta . get_field ( " is_recurring " ) :
convert_to_recurring ( self , self . get ( " posting_date " ) or self . get ( " transaction_date " ) )
def on_update_after_submit ( self ) :
if self . meta . get_field ( " is_recurring " ) :
validate_recurring_document ( self )
convert_to_recurring ( self , self . get ( " posting_date " ) or self . get ( " transaction_date " ) )
def before_recurring ( self ) :
self . fiscal_year = None
for fieldname in ( " due_date " , " aging_date " ) :
if self . meta . get_field ( fieldname ) :
self . set ( fieldname , None )
2013-06-14 12:14:03 +00:00
def set_missing_values ( self , for_validate = False ) :
for fieldname in [ " posting_date " , " transaction_date " ] :
2014-03-28 08:25:00 +00:00
if not self . get ( fieldname ) and self . meta . get_field ( fieldname ) :
self . set ( fieldname , today ( ) )
if not self . fiscal_year :
2014-03-31 18:07:40 +00:00
self . fiscal_year = get_fiscal_year ( self . get ( fieldname ) ) [ 0 ]
2014-04-17 20:00:14 +00:00
break
2014-04-08 14:40:03 +00:00
2015-02-17 05:41:11 +00:00
def calculate_taxes_and_totals ( self ) :
2015-02-18 06:53:18 +00:00
from erpnext . controllers . taxes_and_totals import calculate_taxes_and_totals
calculate_taxes_and_totals ( self )
2015-02-17 05:41:11 +00:00
if self . doctype in [ " Quotation " , " Sales Order " , " Delivery Note " , " Sales Invoice " ] :
self . calculate_commission ( )
self . calculate_contribution ( )
2013-07-11 12:19:18 +00:00
def validate_date_with_fiscal_year ( self ) :
if self . meta . get_field ( " fiscal_year " ) :
date_field = " "
if self . meta . get_field ( " posting_date " ) :
date_field = " posting_date "
elif self . meta . get_field ( " transaction_date " ) :
date_field = " transaction_date "
2014-04-08 14:40:03 +00:00
2014-03-31 18:07:40 +00:00
if date_field and self . get ( date_field ) :
2014-04-08 14:40:03 +00:00
validate_fiscal_year ( self . get ( date_field ) , self . fiscal_year ,
2015-02-19 09:21:58 +00:00
self . meta . get_label ( date_field ) , self )
2014-04-08 14:40:03 +00:00
2014-08-27 11:16:33 +00:00
def validate_due_date ( self ) :
from erpnext . accounts . party import validate_due_date
if self . doctype == " Sales Invoice " :
validate_due_date ( self . posting_date , self . due_date , " Customer " , self . customer , self . company )
elif self . doctype == " Purchase Invoice " :
validate_due_date ( self . posting_date , self . due_date , " Supplier " , self . supplier , self . company )
2013-10-17 11:31:14 +00:00
def set_price_list_currency ( self , buying_or_selling ) :
2013-08-09 12:41:35 +00:00
if self . meta . get_field ( " currency " ) :
2014-03-28 08:25:00 +00:00
company_currency = get_company_currency ( self . company )
2014-04-08 14:40:03 +00:00
2013-09-17 04:51:20 +00:00
# price list part
fieldname = " selling_price_list " if buying_or_selling . lower ( ) == " selling " \
else " buying_price_list "
2014-03-28 08:25:00 +00:00
if self . meta . get_field ( fieldname ) and self . get ( fieldname ) :
self . price_list_currency = frappe . db . get_value ( " Price List " ,
self . get ( fieldname ) , " currency " )
2014-04-08 14:40:03 +00:00
2014-03-28 08:25:00 +00:00
if self . price_list_currency == company_currency :
self . plc_conversion_rate = 1.0
2013-09-24 09:06:55 +00:00
2014-03-28 08:25:00 +00:00
elif not self . plc_conversion_rate :
2014-07-01 12:15:15 +00:00
self . plc_conversion_rate = get_exchange_rate (
2014-03-28 08:25:00 +00:00
self . price_list_currency , company_currency )
2014-04-08 14:40:03 +00:00
2013-09-17 04:51:20 +00:00
# currency
2014-03-28 08:25:00 +00:00
if not self . currency :
self . currency = self . price_list_currency
self . conversion_rate = self . plc_conversion_rate
elif self . currency == company_currency :
self . conversion_rate = 1.0
elif not self . conversion_rate :
2014-07-01 12:15:15 +00:00
self . conversion_rate = get_exchange_rate ( self . currency ,
2013-09-17 04:51:20 +00:00
company_currency )
2014-02-11 10:44:52 +00:00
def set_missing_item_details ( self ) :
2013-05-24 13:55:01 +00:00
""" set missing item values """
2014-02-11 10:44:52 +00:00
from erpnext . stock . get_item_details import get_item_details
2015-02-24 11:20:47 +00:00
2014-12-26 07:45:21 +00:00
if hasattr ( self , " items " ) :
2014-06-20 10:29:49 +00:00
parent_dict = { }
2014-04-10 12:23:30 +00:00
for fieldname in self . meta . get_valid_columns ( ) :
parent_dict [ fieldname ] = self . get ( fieldname )
2014-12-26 07:45:21 +00:00
for item in self . get ( " items " ) :
2014-04-03 09:00:42 +00:00
if item . get ( " item_code " ) :
2014-06-20 10:29:49 +00:00
args = parent_dict . copy ( )
args . update ( item . as_dict ( ) )
2015-03-09 09:24:37 +00:00
if not args . get ( " transaction_date " ) :
args [ " transaction_date " ] = args . get ( " posting_date " )
2015-03-20 08:48:09 +00:00
if self . get ( " is_subcontracted " ) :
args [ " is_subcontracted " ] = self . is_subcontracted
2014-04-03 09:00:42 +00:00
ret = get_item_details ( args )
2014-05-28 07:19:20 +00:00
2014-04-03 09:00:42 +00:00
for fieldname , value in ret . items ( ) :
2014-04-03 12:08:54 +00:00
if item . meta . get_field ( fieldname ) and \
2014-04-03 09:00:42 +00:00
item . get ( fieldname ) is None and value is not None :
item . set ( fieldname , value )
2014-04-08 14:40:03 +00:00
2014-12-30 13:03:52 +00:00
if fieldname == " cost_center " and item . meta . get_field ( " cost_center " ) \
and not item . get ( " cost_center " ) and value is not None :
item . set ( fieldname , value )
2014-05-28 07:19:20 +00:00
if ret . get ( " pricing_rule " ) :
for field in [ " base_price_list_rate " , " price_list_rate " ,
" discount_percentage " , " base_rate " , " rate " ] :
item . set ( field , ret . get ( field ) )
2013-07-04 11:43:53 +00:00
def set_taxes ( self , tax_parentfield , tax_master_field ) :
2013-05-24 13:55:01 +00:00
if not self . meta . get_field ( tax_parentfield ) :
return
2014-04-08 14:40:03 +00:00
2013-07-04 11:43:53 +00:00
tax_master_doctype = self . meta . get_field ( tax_master_field ) . options
2014-04-08 14:40:03 +00:00
2014-03-27 10:42:56 +00:00
if not self . get ( tax_parentfield ) :
2014-03-28 08:25:00 +00:00
if not self . get ( tax_master_field ) :
2013-05-24 13:55:01 +00:00
# get the default tax master
2014-03-28 08:25:00 +00:00
self . set ( tax_master_field , frappe . db . get_value ( tax_master_doctype , { " is_default " : 1 } ) )
2014-04-08 14:40:03 +00:00
2013-07-04 11:43:53 +00:00
self . append_taxes_from_master ( tax_parentfield , tax_master_field , tax_master_doctype )
2014-04-08 14:40:03 +00:00
2013-07-04 11:43:53 +00:00
def append_taxes_from_master ( self , tax_parentfield , tax_master_field , tax_master_doctype = None ) :
2014-03-28 08:25:00 +00:00
if self . get ( tax_master_field ) :
2013-07-04 11:43:53 +00:00
if not tax_master_doctype :
tax_master_doctype = self . meta . get_field ( tax_master_field ) . options
2014-04-08 14:40:03 +00:00
2014-05-02 10:15:10 +00:00
self . extend ( tax_parentfield ,
2014-05-04 14:07:45 +00:00
get_taxes_and_charges ( tax_master_doctype , self . get ( tax_master_field ) , tax_parentfield ) )
2014-01-30 08:26:57 +00:00
2014-04-17 20:00:14 +00:00
def set_other_charges ( self ) :
2014-12-25 10:31:55 +00:00
self . set ( " taxes " , [ ] )
self . set_taxes ( " taxes " , " taxes_and_charges " )
2014-04-08 14:40:03 +00:00
2014-10-06 07:50:53 +00:00
def validate_enabled_taxes_and_charges ( self ) :
taxes_and_charges_doctype = self . meta . get_options ( " taxes_and_charges " )
if frappe . db . get_value ( taxes_and_charges_doctype , self . taxes_and_charges , " disabled " ) :
frappe . throw ( _ ( " {0} ' {1} ' is disabled " ) . format ( taxes_and_charges_doctype , self . taxes_and_charges ) )
2013-08-21 12:17:11 +00:00
def get_gl_dict ( self , args ) :
2013-01-29 06:04:39 +00:00
""" this method populates the common properties of a gl entry record """
2014-02-14 10:17:51 +00:00
gl_dict = frappe . _dict ( {
2014-04-08 14:40:03 +00:00
' company ' : self . company ,
2014-03-28 08:25:00 +00:00
' posting_date ' : self . posting_date ,
' voucher_type ' : self . doctype ,
' voucher_no ' : self . name ,
' aging_date ' : self . get ( " aging_date " ) or self . posting_date ,
2014-04-07 06:32:57 +00:00
' remarks ' : self . get ( " remarks " ) ,
2014-03-28 08:25:00 +00:00
' fiscal_year ' : self . fiscal_year ,
2013-01-29 06:04:39 +00:00
' debit ' : 0 ,
' credit ' : 0 ,
2014-03-28 08:25:00 +00:00
' is_opening ' : self . get ( " is_opening " ) or " No " ,
2014-08-29 05:48:32 +00:00
' party_type ' : None ,
' party ' : None
2013-08-28 13:23:11 +00:00
} )
2013-01-29 06:04:39 +00:00
gl_dict . update ( args )
return gl_dict
2014-04-08 14:40:03 +00:00
2013-02-06 12:03:46 +00:00
def clear_unallocated_advances ( self , childtype , parentfield ) :
2014-04-02 12:39:34 +00:00
self . set ( parentfield , self . get ( parentfield , { " allocated_amount " : [ " not in " , [ 0 , None , " " ] ] } ) )
2014-04-08 14:40:03 +00:00
frappe . db . sql ( """ delete from `tab %s ` where parentfield= %s and parent = %s
2014-03-28 08:25:00 +00:00
and ifnull ( allocated_amount , 0 ) = 0 """ % (childtype, ' %s ' , ' %s ' ), (parentfield, self.name))
2014-04-08 14:40:03 +00:00
2014-08-29 05:48:32 +00:00
def get_advances ( self , account_head , party_type , party , child_doctype , parentfield , dr_or_cr , against_order_field ) :
2014-12-25 12:49:39 +00:00
so_list = list ( set ( [ d . get ( against_order_field ) for d in self . get ( " items " ) if d . get ( against_order_field ) ] ) )
2014-09-19 09:01:49 +00:00
cond = " "
if so_list :
cond = " or (ifnull(t2. %s , ' ' ) in ( %s )) " % ( " against_ " + against_order_field , ' , ' . join ( [ ' %s ' ] * len ( so_list ) ) )
2014-09-18 09:33:54 +00:00
2014-05-01 10:13:22 +00:00
res = frappe . db . sql ( """
select
2014-10-21 07:44:55 +00:00
t1 . name as jv_no , t1 . remark , t2 . { 0 } as amount , t2 . name as jv_detail_no , ` against_ { 1 } ` as against_order
2014-05-01 10:13:22 +00:00
from
2014-12-25 11:44:18 +00:00
` tabJournal Entry ` t1 , ` tabJournal Entry Account ` t2
2014-05-01 10:13:22 +00:00
where
2014-08-29 05:48:32 +00:00
t1 . name = t2 . parent and t2 . account = % s
and t2 . party_type = % s and t2 . party = % s
and t2 . is_advance = ' Yes ' and t1 . docstatus = 1
2014-09-18 09:33:54 +00:00
and ( (
ifnull ( t2 . against_voucher , ' ' ) = ' '
and ifnull ( t2 . against_invoice , ' ' ) = ' '
and ifnull ( t2 . against_jv , ' ' ) = ' '
and ifnull ( t2 . against_sales_order , ' ' ) = ' '
and ifnull ( t2 . against_purchase_order , ' ' ) = ' '
2014-10-21 07:44:55 +00:00
) { 2 } )
order by t1 . posting_date """ .format(dr_or_cr, against_order_field, cond),
[ account_head , party_type , party ] + so_list , as_dict = 1 )
2014-09-16 05:38:08 +00:00
2014-03-27 10:42:56 +00:00
self . set ( parentfield , [ ] )
2013-01-30 13:46:13 +00:00
for d in res :
2014-09-18 09:33:54 +00:00
self . append ( parentfield , {
" doctype " : child_doctype ,
2014-12-25 11:44:18 +00:00
" journal_entry " : d . jv_no ,
2014-09-18 09:33:54 +00:00
" jv_detail_no " : d . jv_detail_no ,
" remarks " : d . remark ,
" advance_amount " : flt ( d . amount ) ,
2014-10-13 05:17:14 +00:00
" allocated_amount " : flt ( d . amount ) if d . against_order else 0
2014-09-18 09:33:54 +00:00
} )
def validate_advance_jv ( self , advance_table_fieldname , against_order_field ) :
2014-12-25 12:49:39 +00:00
order_list = list ( set ( [ d . get ( against_order_field ) for d in self . get ( " items " ) if d . get ( against_order_field ) ] ) )
2014-09-18 09:33:54 +00:00
if order_list :
account = self . get ( " debit_to " if self . doctype == " Sales Invoice " else " credit_to " )
jv_against_order = frappe . db . sql ( """ select parent, %s as against_order
2014-12-25 10:31:55 +00:00
from ` tabJournal Entry Account `
2014-09-18 09:33:54 +00:00
where docstatus = 1 and account = % s and ifnull ( is_advance , ' No ' ) = ' Yes '
and ifnull ( against_sales_order , ' ' ) in ( % s )
group by parent , against_sales_order """ %
( " against_ " + against_order_field , ' %s ' , ' , ' . join ( [ ' %s ' ] * len ( order_list ) ) ) ,
tuple ( [ account ] + order_list ) , as_dict = 1 )
if jv_against_order :
order_jv_map = { }
for d in jv_against_order :
order_jv_map . setdefault ( d . against_order , [ ] ) . append ( d . parent )
2014-12-25 11:44:18 +00:00
advance_jv_against_si = [ d . journal_entry for d in self . get ( advance_table_fieldname ) ]
2014-09-18 09:33:54 +00:00
for order , jv_list in order_jv_map . items ( ) :
for jv in jv_list :
if not advance_jv_against_si or jv not in advance_jv_against_si :
2014-12-31 07:54:36 +00:00
frappe . msgprint ( _ ( " Journal Entry {0} is linked against Order {1} , check if it should be pulled as advance in this invoice. " )
2014-09-18 09:33:54 +00:00
. format ( jv , order ) )
2014-04-08 14:40:03 +00:00
2013-07-29 13:05:39 +00:00
def validate_multiple_billing ( self , ref_dt , item_ref_dn , based_on , parentfield ) :
2014-01-23 10:03:30 +00:00
from erpnext . controllers . status_updater import get_tolerance_for
2014-01-03 12:13:19 +00:00
item_tolerance = { }
global_tolerance = None
2014-04-08 14:40:03 +00:00
2014-12-25 12:49:39 +00:00
for item in self . get ( " items " ) :
2014-03-28 08:25:00 +00:00
if item . get ( item_ref_dn ) :
2014-04-08 14:40:03 +00:00
ref_amt = flt ( frappe . db . get_value ( ref_dt + " Item " ,
2014-03-31 18:07:40 +00:00
item . get ( item_ref_dn ) , based_on ) , self . precision ( based_on , item ) )
2014-01-07 07:11:09 +00:00
if not ref_amt :
2014-04-14 13:50:45 +00:00
frappe . msgprint ( _ ( " Warning: System will not check overbilling since amount for Item {0} in {1} is zero " ) . format ( item . item_code , ref_dt ) )
2014-01-07 07:11:09 +00:00
else :
2014-04-08 14:40:03 +00:00
already_billed = frappe . db . sql ( """ select sum( %s ) from `tab %s `
where % s = % s and docstatus = 1 and parent != % s """ %
2014-12-26 07:45:21 +00:00
( based_on , self . doctype + " Item " , item_ref_dn , ' %s ' , ' %s ' ) ,
2014-03-31 18:07:40 +00:00
( item . get ( item_ref_dn ) , self . name ) ) [ 0 ] [ 0 ]
2014-04-08 14:40:03 +00:00
total_billed_amt = flt ( flt ( already_billed ) + flt ( item . get ( based_on ) ) ,
2014-01-07 07:11:09 +00:00
self . precision ( based_on , item ) )
2014-04-08 14:40:03 +00:00
tolerance , item_tolerance , global_tolerance = get_tolerance_for ( item . item_code ,
2014-01-07 07:11:09 +00:00
item_tolerance , global_tolerance )
2014-04-08 14:40:03 +00:00
2014-01-07 07:11:09 +00:00
max_allowed_amt = flt ( ref_amt * ( 100 + tolerance ) / 100 )
2014-04-08 14:40:03 +00:00
2014-01-07 07:11:09 +00:00
if total_billed_amt - max_allowed_amt > 0.01 :
2015-01-06 07:26:37 +00:00
frappe . throw ( _ ( " Cannot overbill for Item {0} in row {1} more than {2} . To allow overbilling, please set in Stock Settings " ) . format ( item . item_code , item . idx , max_allowed_amt ) )
2014-04-08 14:40:03 +00:00
2013-03-25 05:36:00 +00:00
def get_company_default ( self , fieldname ) :
2013-12-12 13:42:19 +00:00
from erpnext . accounts . utils import get_company_default
2014-03-28 08:25:00 +00:00
return get_company_default ( self . company , fieldname )
2014-04-08 14:40:03 +00:00
2013-08-02 09:20:12 +00:00
def get_stock_items ( self ) :
stock_items = [ ]
2014-12-26 07:45:21 +00:00
item_codes = list ( set ( item . item_code for item in self . get ( " items " ) ) )
2013-08-02 09:20:12 +00:00
if item_codes :
2014-02-26 07:05:33 +00:00
stock_items = [ r [ 0 ] for r in frappe . db . sql ( """ select name
2013-08-02 09:20:12 +00:00
from ` tabItem ` where name in ( % s ) and is_stock_item = ' Yes ' """ % \
( " , " . join ( ( [ " %s " ] * len ( item_codes ) ) ) , ) , item_codes ) ]
2014-04-08 14:40:03 +00:00
2013-08-02 09:20:12 +00:00
return stock_items
2014-04-08 14:40:03 +00:00
2014-09-12 09:48:53 +00:00
def set_total_advance_paid ( self ) :
if self . doctype == " Sales Order " :
dr_or_cr = " credit "
against_field = " against_sales_order "
else :
dr_or_cr = " debit "
against_field = " against_purchase_order "
advance_paid = frappe . db . sql ( """
select
sum ( ifnull ( { dr_or_cr } , 0 ) )
from
2014-12-25 10:31:55 +00:00
` tabJournal Entry Account `
2014-09-12 09:48:53 +00:00
where
{ against_field } = % s and docstatus = 1 and is_advance = " Yes " """ .format(dr_or_cr=dr_or_cr, \
against_field = against_field ) , self . name )
if advance_paid :
advance_paid = flt ( advance_paid [ 0 ] [ 0 ] , self . precision ( " advance_paid " ) )
2015-02-12 10:39:11 +00:00
if flt ( self . base_grand_total ) > = advance_paid :
2014-09-12 09:48:53 +00:00
frappe . db . set_value ( self . doctype , self . name , " advance_paid " , advance_paid )
else :
frappe . throw ( _ ( " Total advance ( {0} ) against Order {1} cannot be greater \
than the Grand Total ( { 2 } ) " )
2015-02-12 10:39:11 +00:00
. format ( advance_paid , self . name , self . base_grand_total ) )
2014-09-12 09:48:53 +00:00
2014-08-26 08:59:06 +00:00
@property
def company_abbr ( self ) :
if not hasattr ( self , " _abbr " ) :
self . _abbr = frappe . db . get_value ( " Company " , self . company , " abbr " )
return self . _abbr
2014-02-14 10:17:51 +00:00
@frappe.whitelist ( )
2013-10-10 11:05:09 +00:00
def get_tax_rate ( account_head ) :
2014-04-08 14:40:03 +00:00
return frappe . db . get_value ( " Account " , account_head , " tax_rate " )
2014-05-02 10:15:10 +00:00
@frappe.whitelist ( )
def get_taxes_and_charges ( master_doctype , master_name , tax_parentfield ) :
from frappe . model import default_fields
tax_master = frappe . get_doc ( master_doctype , master_name )
taxes_and_charges = [ ]
for i , tax in enumerate ( tax_master . get ( tax_parentfield ) ) :
tax = tax . as_dict ( )
for fieldname in default_fields :
if fieldname in tax :
del tax [ fieldname ]
taxes_and_charges . append ( tax )
return taxes_and_charges
2014-09-29 06:47:03 +00:00
def validate_conversion_rate ( currency , conversion_rate , conversion_rate_label , company ) :
""" common validation for currency and price list currency """
company_currency = frappe . db . get_value ( " Company " , company , " default_currency " )
if not conversion_rate :
throw ( _ ( " {0} is mandatory. Maybe Currency Exchange record is not created for {1} to {2} . " ) . format (
conversion_rate_label , currency , company_currency ) )
2015-02-23 06:28:15 +00:00
def validate_taxes_and_charges ( tax ) :
2015-02-23 10:31:33 +00:00
if tax . charge_type in [ ' Actual ' , ' On Net Total ' ] and tax . row_id :
2015-02-23 06:28:15 +00:00
frappe . throw ( _ ( " Can refer row only if the charge type is ' On Previous Row Amount ' or ' Previous Row Total ' " ) )
2015-02-23 10:31:33 +00:00
elif tax . charge_type in [ ' On Previous Row Amount ' , ' On Previous Row Total ' ] :
2015-02-23 06:28:15 +00:00
if cint ( tax . idx ) == 1 :
frappe . throw ( _ ( " Cannot select charge type as ' On Previous Row Amount ' or ' On Previous Row Total ' for first row " ) )
elif not tax . row_id :
frappe . throw ( _ ( " Please specify a valid Row ID for row {0} in table {1} " . format ( tax . idx , _ ( tax . doctype ) ) ) )
elif tax . row_id and cint ( tax . row_id ) > = cint ( tax . idx ) :
frappe . throw ( _ ( " Cannot refer row number greater than or equal to current row number for this Charge type " ) )
2015-02-25 09:38:42 +00:00
if tax . charge_type == " Actual " :
2015-02-27 18:10:56 +00:00
tax . rate = None
2015-02-25 09:38:42 +00:00
if not tax . tax_amount :
frappe . throw ( _ ( " Amount is mandatory for charge type ' Actual ' " ) )
2015-02-23 06:28:15 +00:00
def validate_inclusive_tax ( tax , doc ) :
def _on_previous_row_error ( row_range ) :
throw ( _ ( " To include tax in row {0} in Item rate, taxes in rows {1} must also be included " ) . format ( tax . idx ,
row_range ) )
if cint ( getattr ( tax , " included_in_print_rate " , None ) ) :
if tax . charge_type == " Actual " :
# inclusive tax cannot be of type Actual
throw ( _ ( " Charge of type ' Actual ' in row {0} cannot be included in Item Rate " ) . format ( tax . idx ) )
elif tax . charge_type == " On Previous Row Amount " and \
not cint ( doc . get ( " taxes " ) [ cint ( tax . row_id ) - 1 ] . included_in_print_rate ) :
# referred row should also be inclusive
_on_previous_row_error ( tax . row_id )
elif tax . charge_type == " On Previous Row Total " and \
not all ( [ cint ( t . included_in_print_rate ) for t in doc . get ( " taxes " ) [ : cint ( tax . row_id ) - 1 ] ] ) :
# all rows about the reffered tax should be inclusive
_on_previous_row_error ( " 1 - %d " % ( tax . row_id , ) )
2015-02-24 11:38:34 +00:00
elif tax . get ( " category " ) == " Valuation " :
2015-02-24 10:35:19 +00:00
frappe . throw ( _ ( " Valuation type charges can not marked as Inclusive " ) )