2017-12-12 14:40:52 +05:30
|
|
|
# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
|
|
|
# License: GNU General Public License v3. See license.txt
|
|
|
|
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
import frappe, os, json
|
|
|
|
from frappe.custom.doctype.custom_field.custom_field import create_custom_fields
|
2020-09-28 14:38:45 +05:30
|
|
|
from frappe.permissions import add_permission, update_permission_property
|
2020-10-27 15:44:43 +05:30
|
|
|
from erpnext.payroll.doctype.gratuity_rule.gratuity_rule import get_gratuity_rule
|
2017-12-12 14:40:52 +05:30
|
|
|
|
|
|
|
def setup(company=None, patch=True):
|
|
|
|
make_custom_fields()
|
|
|
|
add_print_formats()
|
2020-09-25 14:24:57 +05:30
|
|
|
add_custom_roles_for_reports()
|
2020-09-28 14:38:45 +05:30
|
|
|
add_permissions()
|
2020-08-18 14:13:54 +05:30
|
|
|
create_gratuity_rule()
|
2017-12-12 14:40:52 +05:30
|
|
|
|
|
|
|
def make_custom_fields():
|
2020-09-25 13:13:38 +05:30
|
|
|
is_zero_rated = dict(fieldname='is_zero_rated', label='Is Zero Rated',
|
|
|
|
fieldtype='Check', fetch_from='item_code.is_zero_rated', insert_after='description',
|
|
|
|
print_hide=1)
|
|
|
|
is_exempt = dict(fieldname='is_exempt', label='Is Exempt',
|
|
|
|
fieldtype='Check', fetch_from='item_code.is_exempt', insert_after='is_zero_rated',
|
|
|
|
print_hide=1)
|
|
|
|
|
2017-12-12 14:40:52 +05:30
|
|
|
invoice_fields = [
|
|
|
|
dict(fieldname='vat_section', label='VAT Details', fieldtype='Section Break',
|
2018-07-03 10:28:20 +05:30
|
|
|
insert_after='group_same_items', print_hide=1, collapsible=1),
|
2017-12-12 14:40:52 +05:30
|
|
|
dict(fieldname='permit_no', label='Permit Number',
|
|
|
|
fieldtype='Data', insert_after='vat_section', print_hide=1),
|
|
|
|
]
|
|
|
|
|
|
|
|
purchase_invoice_fields = [
|
|
|
|
dict(fieldname='company_trn', label='Company TRN',
|
|
|
|
fieldtype='Read Only', insert_after='shipping_address',
|
2019-03-08 18:37:34 +05:30
|
|
|
fetch_from='company.tax_id', print_hide=1),
|
2017-12-12 14:40:52 +05:30
|
|
|
dict(fieldname='supplier_name_in_arabic', label='Supplier Name in Arabic',
|
|
|
|
fieldtype='Read Only', insert_after='supplier_name',
|
2020-09-24 13:21:23 +05:30
|
|
|
fetch_from='supplier.supplier_name_in_arabic', print_hide=1),
|
2020-11-16 17:36:27 +05:30
|
|
|
dict(fieldname='recoverable_standard_rated_expenses', print_hide=1, default='0',
|
|
|
|
label='Recoverable Standard Rated Expenses (AED)', insert_after='permit_no',
|
|
|
|
fieldtype='Currency', ),
|
2020-09-24 13:21:23 +05:30
|
|
|
dict(fieldname='reverse_charge', label='Reverse Charge Applicable',
|
2020-10-19 17:58:50 +05:30
|
|
|
fieldtype='Select', insert_after='recoverable_standard_rated_expenses', print_hide=1,
|
2020-09-24 13:21:23 +05:30
|
|
|
options='Y\nN', default='N'),
|
2020-10-19 17:58:50 +05:30
|
|
|
dict(fieldname='recoverable_reverse_charge', label='Recoverable Reverse Charge (Percentage)',
|
2020-09-24 18:41:43 +05:30
|
|
|
insert_after='reverse_charge', fieldtype='Percent', print_hide=1,
|
|
|
|
depends_on="eval:doc.reverse_charge=='Y'", default='100.000'),
|
2017-12-12 14:40:52 +05:30
|
|
|
]
|
|
|
|
|
|
|
|
sales_invoice_fields = [
|
|
|
|
dict(fieldname='company_trn', label='Company TRN',
|
|
|
|
fieldtype='Read Only', insert_after='company_address',
|
2019-03-08 18:37:34 +05:30
|
|
|
fetch_from='company.tax_id', print_hide=1),
|
2017-12-12 14:40:52 +05:30
|
|
|
dict(fieldname='customer_name_in_arabic', label='Customer Name in Arabic',
|
|
|
|
fieldtype='Read Only', insert_after='customer_name',
|
2019-03-08 18:37:34 +05:30
|
|
|
fetch_from='customer.customer_name_in_arabic', print_hide=1),
|
2020-10-28 10:28:25 +05:30
|
|
|
dict(fieldname='vat_emirate', label='VAT Emirate', insert_after='permit_no', fieldtype='Select',
|
2020-11-16 12:58:38 +05:30
|
|
|
options='\nAbu Dhabi\nAjman\nDubai\nFujairah\nRas Al Khaimah\nSharjah\nUmm Al Quwain',
|
|
|
|
fetch_from='company_address.emirate'),
|
2020-09-24 13:21:23 +05:30
|
|
|
dict(fieldname='tourist_tax_return', label='Tax Refund provided to Tourists (AED)',
|
2020-10-28 10:28:25 +05:30
|
|
|
insert_after='vat_emirate', fieldtype='Currency', print_hide=1, default='0'),
|
2017-12-12 14:40:52 +05:30
|
|
|
]
|
|
|
|
|
2017-12-28 14:20:13 +05:30
|
|
|
invoice_item_fields = [
|
|
|
|
dict(fieldname='tax_code', label='Tax Code',
|
2019-03-08 18:37:34 +05:30
|
|
|
fieldtype='Read Only', fetch_from='item_code.tax_code', insert_after='description',
|
2017-12-28 14:20:13 +05:30
|
|
|
allow_on_submit=1, print_hide=1),
|
|
|
|
dict(fieldname='tax_rate', label='Tax Rate',
|
|
|
|
fieldtype='Float', insert_after='tax_code',
|
|
|
|
print_hide=1, hidden=1, read_only=1),
|
|
|
|
dict(fieldname='tax_amount', label='Tax Amount',
|
|
|
|
fieldtype='Currency', insert_after='tax_rate',
|
|
|
|
print_hide=1, hidden=1, read_only=1, options="currency"),
|
|
|
|
dict(fieldname='total_amount', label='Total Amount',
|
|
|
|
fieldtype='Currency', insert_after='tax_amount',
|
|
|
|
print_hide=1, hidden=1, read_only=1, options="currency"),
|
2017-12-28 18:42:22 +05:30
|
|
|
]
|
|
|
|
|
|
|
|
delivery_date_field = [
|
2017-12-28 14:20:13 +05:30
|
|
|
dict(fieldname='delivery_date', label='Delivery Date',
|
2017-12-28 18:42:22 +05:30
|
|
|
fieldtype='Date', insert_after='item_name', print_hide=1)
|
2017-12-28 14:20:13 +05:30
|
|
|
]
|
2017-12-12 14:40:52 +05:30
|
|
|
|
|
|
|
custom_fields = {
|
|
|
|
'Item': [
|
|
|
|
dict(fieldname='tax_code', label='Tax Code',
|
|
|
|
fieldtype='Data', insert_after='item_group'),
|
2020-09-25 13:13:38 +05:30
|
|
|
dict(fieldname='is_zero_rated', label='Is Zero Rated',
|
|
|
|
fieldtype='Check', insert_after='tax_code',
|
|
|
|
print_hide=1),
|
2020-10-20 16:35:28 +05:30
|
|
|
dict(fieldname='is_exempt', label='Is Exempt',
|
2020-09-25 13:13:38 +05:30
|
|
|
fieldtype='Check', insert_after='is_zero_rated',
|
|
|
|
print_hide=1)
|
2017-12-12 14:40:52 +05:30
|
|
|
],
|
|
|
|
'Customer': [
|
|
|
|
dict(fieldname='customer_name_in_arabic', label='Customer Name in Arabic',
|
|
|
|
fieldtype='Data', insert_after='customer_name'),
|
|
|
|
],
|
|
|
|
'Supplier': [
|
|
|
|
dict(fieldname='supplier_name_in_arabic', label='Supplier Name in Arabic',
|
|
|
|
fieldtype='Data', insert_after='supplier_name'),
|
|
|
|
],
|
2020-11-16 12:58:38 +05:30
|
|
|
'Address': [
|
|
|
|
dict(fieldname='emirate', label='Emirate', fieldtype='Select', insert_after='state',
|
2020-11-16 17:36:27 +05:30
|
|
|
options='\nAbu Dhabi\nAjman\nDubai\nFujairah\nRas Al Khaimah\nSharjah\nUmm Al Quwain')
|
2020-11-16 12:58:38 +05:30
|
|
|
],
|
2017-12-12 14:40:52 +05:30
|
|
|
'Purchase Invoice': purchase_invoice_fields + invoice_fields,
|
2017-12-28 18:42:22 +05:30
|
|
|
'Purchase Order': purchase_invoice_fields + invoice_fields,
|
|
|
|
'Purchase Receipt': purchase_invoice_fields + invoice_fields,
|
2017-12-12 14:40:52 +05:30
|
|
|
'Sales Invoice': sales_invoice_fields + invoice_fields,
|
2021-01-28 17:58:55 +05:30
|
|
|
'POS Invoice': sales_invoice_fields + invoice_fields,
|
2017-12-28 18:42:22 +05:30
|
|
|
'Sales Order': sales_invoice_fields + invoice_fields,
|
|
|
|
'Delivery Note': sales_invoice_fields + invoice_fields,
|
2020-09-25 13:13:38 +05:30
|
|
|
'Sales Invoice Item': invoice_item_fields + delivery_date_field + [is_zero_rated, is_exempt],
|
2021-01-28 17:58:55 +05:30
|
|
|
'POS Invoice Item': invoice_item_fields + delivery_date_field + [is_zero_rated, is_exempt],
|
2017-12-28 18:42:22 +05:30
|
|
|
'Purchase Invoice Item': invoice_item_fields,
|
|
|
|
'Sales Order Item': invoice_item_fields,
|
|
|
|
'Delivery Note Item': invoice_item_fields,
|
|
|
|
'Quotation Item': invoice_item_fields,
|
|
|
|
'Purchase Order Item': invoice_item_fields,
|
|
|
|
'Purchase Receipt Item': invoice_item_fields,
|
|
|
|
'Supplier Quotation Item': invoice_item_fields,
|
2017-12-12 14:40:52 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
create_custom_fields(custom_fields)
|
|
|
|
|
|
|
|
def add_print_formats():
|
|
|
|
frappe.reload_doc("regional", "print_format", "detailed_tax_invoice")
|
|
|
|
frappe.reload_doc("regional", "print_format", "simplified_tax_invoice")
|
2017-12-28 14:20:13 +05:30
|
|
|
frappe.reload_doc("regional", "print_format", "tax_invoice")
|
2017-12-16 10:53:53 +05:30
|
|
|
|
|
|
|
frappe.db.sql(""" update `tabPrint Format` set disabled = 0 where
|
2017-12-28 14:20:13 +05:30
|
|
|
name in('Simplified Tax Invoice', 'Detailed Tax Invoice', 'Tax Invoice') """)
|
2020-09-25 14:24:57 +05:30
|
|
|
|
|
|
|
def add_custom_roles_for_reports():
|
2020-10-14 12:21:36 +05:30
|
|
|
"""Add Access Control to UAE VAT 201."""
|
|
|
|
if not frappe.db.get_value('Custom Role', dict(report='UAE VAT 201')):
|
2020-09-25 14:24:57 +05:30
|
|
|
frappe.get_doc(dict(
|
|
|
|
doctype='Custom Role',
|
2020-10-14 12:21:36 +05:30
|
|
|
report='UAE VAT 201',
|
2020-09-25 14:24:57 +05:30
|
|
|
roles= [
|
|
|
|
dict(role='Accounts User'),
|
|
|
|
dict(role='Accounts Manager'),
|
|
|
|
dict(role='Auditor')
|
|
|
|
]
|
2020-09-28 14:38:45 +05:30
|
|
|
)).insert()
|
|
|
|
|
|
|
|
def add_permissions():
|
2020-09-30 12:15:07 +05:30
|
|
|
"""Add Permissions for UAE VAT Settings and UAE VAT Account."""
|
2020-09-30 21:51:00 +05:30
|
|
|
for doctype in ('UAE VAT Settings', 'UAE VAT Account'):
|
2020-09-28 14:38:45 +05:30
|
|
|
add_permission(doctype, 'All', 0)
|
|
|
|
for role in ('Accounts Manager', 'Accounts User', 'System Manager'):
|
|
|
|
add_permission(doctype, role, 0)
|
|
|
|
update_permission_property(doctype, role, 0, 'write', 1)
|
|
|
|
update_permission_property(doctype, role, 0, 'create', 1)
|
2020-08-18 14:13:54 +05:30
|
|
|
|
|
|
|
def create_gratuity_rule():
|
2020-10-27 15:44:43 +05:30
|
|
|
rule_1 = rule_2 = rule_3 = None
|
2020-08-17 14:50:35 +05:30
|
|
|
|
|
|
|
# Rule Under Limited Contract
|
2020-10-27 15:44:43 +05:30
|
|
|
slabs = get_slab_for_limited_contract()
|
2020-08-18 15:02:32 +05:30
|
|
|
if not frappe.db.exists("Gratuity Rule", "Rule Under Limited Contract (UAE)"):
|
2020-10-27 15:44:43 +05:30
|
|
|
rule_1 = get_gratuity_rule("Rule Under Limited Contract (UAE)", slabs, calculate_gratuity_amount_based_on="Sum of all previous slabs")
|
2020-08-17 14:50:35 +05:30
|
|
|
|
|
|
|
# Rule Under Unlimited Contract on termination
|
2020-10-27 15:44:43 +05:30
|
|
|
slabs = get_slab_for_unlimited_contract_on_termination()
|
2020-08-18 15:02:32 +05:30
|
|
|
if not frappe.db.exists("Gratuity Rule", "Rule Under Unlimited Contract on termination (UAE)"):
|
2020-10-27 15:44:43 +05:30
|
|
|
rule_2 = get_gratuity_rule("Rule Under Unlimited Contract on termination (UAE)", slabs)
|
2020-08-18 15:02:32 +05:30
|
|
|
|
2020-10-27 15:44:43 +05:30
|
|
|
# Rule Under Unlimited Contract on resignation
|
|
|
|
slabs = get_slab_for_unlimited_contract_on_resignation()
|
2020-08-18 15:02:32 +05:30
|
|
|
if not frappe.db.exists("Gratuity Rule", "Rule Under Unlimited Contract on resignation (UAE)"):
|
2020-10-27 15:44:43 +05:30
|
|
|
rule_3 = get_gratuity_rule("Rule Under Unlimited Contract on resignation (UAE)", slabs)
|
2020-08-18 15:02:32 +05:30
|
|
|
|
2020-10-27 15:44:43 +05:30
|
|
|
#for applicable salary component user need to set this by its own
|
|
|
|
if rule_1:
|
2020-08-18 15:02:32 +05:30
|
|
|
rule_1.flags.ignore_mandatory = True
|
|
|
|
rule_1.save()
|
2020-10-27 15:44:43 +05:30
|
|
|
if rule_2:
|
|
|
|
rule_2.flags.ignore_mandatory = True
|
2020-08-18 15:02:32 +05:30
|
|
|
rule_2.save()
|
2020-10-27 15:44:43 +05:30
|
|
|
if rule_3:
|
|
|
|
rule_3.flags.ignore_mandatory = True
|
2020-08-18 15:02:32 +05:30
|
|
|
rule_3.save()
|
2020-08-17 14:50:35 +05:30
|
|
|
|
|
|
|
|
2020-10-27 15:44:43 +05:30
|
|
|
def get_slab_for_limited_contract():
|
|
|
|
return [{
|
|
|
|
"from_year": 0,
|
|
|
|
"to_year":1,
|
|
|
|
"fraction_of_applicable_earnings": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"from_year": 1,
|
|
|
|
"to_year":5,
|
|
|
|
"fraction_of_applicable_earnings": 21/30
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"from_year": 5,
|
|
|
|
"to_year":0,
|
|
|
|
"fraction_of_applicable_earnings": 1
|
|
|
|
}]
|
|
|
|
|
|
|
|
def get_slab_for_unlimited_contract_on_termination():
|
|
|
|
return [{
|
|
|
|
"from_year": 0,
|
|
|
|
"to_year":1,
|
|
|
|
"fraction_of_applicable_earnings": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"from_year": 1,
|
|
|
|
"to_year":5,
|
|
|
|
"fraction_of_applicable_earnings": 21/30
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"from_year": 5,
|
|
|
|
"to_year":0,
|
|
|
|
"fraction_of_applicable_earnings": 1
|
|
|
|
}]
|
|
|
|
|
|
|
|
def get_slab_for_unlimited_contract_on_resignation():
|
|
|
|
fraction_1 = 1/3 * 21/30
|
|
|
|
fraction_2 = 2/3 * 21/30
|
|
|
|
fraction_3 = 21/30
|
|
|
|
|
|
|
|
return [{
|
|
|
|
"from_year": 0,
|
|
|
|
"to_year":1,
|
|
|
|
"fraction_of_applicable_earnings": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"from_year": 1,
|
|
|
|
"to_year":3,
|
|
|
|
"fraction_of_applicable_earnings": fraction_1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"from_year": 3,
|
|
|
|
"to_year":5,
|
|
|
|
"fraction_of_applicable_earnings": fraction_2
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"from_year": 5,
|
|
|
|
"to_year":0,
|
|
|
|
"fraction_of_applicable_earnings": fraction_3
|
|
|
|
}]
|