############################################################################## # # Copyright (c) 2002 Nexedi SARL and Contributors. All Rights Reserved. # Jean-Paul Smets-Solanes <jp@nexedi.com> # # WARNING: This program as such is intended to be used by professional # programmers who take the whole responsability of assessing all potential # consequences resulting from its eventual inadequacies and bugs # End users who are looking for a ready-to-use solution with commercial # garantees and support are strongly adviced to contract a Free Software # Service Company # # This program is Free Software; you can redistribute it and/or # modify it under the terms of the GNU General Public License # as published by the Free Software Foundation; either version 2 # of the License, or (at your option) any later version. # # 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. # ############################################################################## from AccessControl import ClassSecurityInfo from Products.ERP5Type import Permissions, PropertySheet, Constraint, Interface from Products.ERP5.Document.Invoice import Invoice from zLOG import LOG class PaySheetTransaction(Invoice): """ A paysheet will store data about the salary of an employee """ meta_type = 'ERP5 Pay Sheet Transaction' portal_type = 'Pay Sheet Transaction' add_permission = Permissions.AddPortalContent isPortalContent = 1 isRADContent = 1 # Declarative security security = ClassSecurityInfo() security.declareObjectProtected(Permissions.AccessContentsInformation) # Global variables _transaction_line_portal_type = 'Pay Sheet Transaction Line' # Default Properties property_sheets = ( PropertySheet.Base , PropertySheet.SimpleItem , PropertySheet.CategoryCore , PropertySheet.Task , PropertySheet.Arrow , PropertySheet.Delivery , PropertySheet.PaySheet , PropertySheet.Movement , PropertySheet.Amount , PropertySheet.XMLObject , PropertySheet.TradeCondition , PropertySheet.DefaultAnnotationLine ) # Declarative Interface __implements__ = ( ) security.declareProtected(Permissions.AccessContentsInformation, 'getRatioQuantityFromReference') def getRatioQuantityFromReference(self, ratio_reference=None): """ return the ratio value correponding to the ratio_reference, or description if ratio value is empty, None if ratio_reference not found """ object_ratio_list = self.contentValues(portal_type=\ 'Pay Sheet Model Ratio Line') for object in object_ratio_list: if object.getReference() == ratio_reference: return object.getQuantity() return None security.declareProtected(Permissions.AccessContentsInformation, 'getRatioQuantityList') def getRatioQuantityList(self, ratio_reference_list): """ Return a list of reference_ratio_list correponding values. reference_ratio_list is a list of references to the ratio lines we want to get. """ if not isinstance(ratio_reference_list, list): return [self.getRatioQuantityFromReference(ratio_reference_list)] return [self.getRatioQuantityFromReference(reference) \ for reference in ratio_reference_list] security.declareProtected(Permissions.AddPortalContent, 'createPaySheetLine') def createPaySheetLine(self, cell_list, title='', res='', desc='', base_amount_list=None, **kw): ''' This function register all paysheet informations in paysheet lines and cells. Select good cells only ''' good_cell_list = [] for cell in cell_list: if cell['quantity'] or cell['price']: good_cell_list.append(cell) if len(good_cell_list) == 0: return # Get all variation categories used in cell_list var_cat_list = [] for cell in good_cell_list: # Don't add a variation category if already in it for category in cell['axe_list']: if category not in var_cat_list: var_cat_list.append(category) # Construct the description description = None if len(desc) > 0: description = desc#'\n'.join(desc) # Add a new Pay Sheet Line payline = self.newContent( portal_type = 'Pay Sheet Line', title = title, description = description, source_section = \ self.getPortalObject().restrictedTraverse(res).getSource(), resource = res, destination_section = self.getDestinationSection(), destination = self.getDestinationSection(), variation_base_category_list = ('tax_category', 'salary_range'), variation_category_list = var_cat_list, base_amount_list = base_amount_list, **kw) base_id = 'movement' a = payline.updateCellRange(script_id = 'PaySheetLine_asCellRange', base_id = base_id) # create cell_list for cell in good_cell_list: cell_cat_list = cell['axe_list'] paycell = payline.newCell(base_id = base_id, *cell_cat_list) # if the quantity aven't be completed, it should be set to 1 (=100%) if cell['price']: price = cell['price'] else: price = 1 paycell.edit( mapped_value_property_list = ('price', 'quantity') , quantity = cell['quantity'] , price = price , force_update = 1 , category_list = cell_cat_list ) return payline security.declareProtected(Permissions.AddPortalContent, 'createEditablePaySheetLineList') def createEditablePaySheetLineList(self, listbox, **kw): ''' this script is called by the preview form to ask to the accountable the values of the editables lines and create corresponding PaySheetLines with this values ''' paysheet = self paysheet_items = {} for line in listbox: model_line = paysheet.getPortalObject().restrictedTraverse(\ line['model_line']) service = model_line.getResourceValue() service_id = service.getId() quantity = line['quantity'] price = line['price'] tax_category = line['tax_category_relative_url'] variation_category_list = model_line.getVariationCategoryList(\ base_category_list='salary_range') salary_range_categories = [] #for category in resource_variation_category_list: for category in variation_category_list: if category.startswith('salary_range/'): salary_range_categories.append(category) # perhaps here it will be necesary to raise an error ? if not paysheet_items.has_key(service_id): paysheet_items[service_id] = { 'title' : model_line.getTitleOrId(), 'desc' : [], 'base_amount_list' : model_line.getBaseAmountList(), 'res' : service.getRelativeUrl(), 'cell_list' : [] } # create cells if a value has been entered by accountable if quantity or price: for salary_range in salary_range_categories: # Define an empty new cell new_cell = None new_cell = { 'axe_list' : [tax_category,salary_range] , 'quantity' : quantity , 'price' : price } # Add the cell to the conresponding paysheet item if new_cell: paysheet_items[service_id]['cell_list'].append(new_cell) # Save the comment as description if model_line.getDescription(): paysheet_items[service_id]['desc'].append(\ model_line.getDescription()) else: resource_description = \ model_line.getResourceValue().getDescription() paysheet_items[service_id]['desc'].append(resource_description) # Create a paysheet item for each service with user data in it for item in paysheet_items.values(): if item['cell_list']: if len(item['desc']) > 0: desc = '\n'.join(item['desc']) else: desc = None paysheet.createPaySheetLine( title = item['title'], res = item['res'], desc = desc, cell_list = item['cell_list'], base_amount_list=item['base_amount_list'],) security.declareProtected(Permissions.ModifyPortalContent, 'createNotEditablePaySheetLineList') def createNotEditablePaySheetLineList(self, **kw): ''' get all data required to create not editable paysheet lines and create it editable paysheet lines have been created by a script ''' # Get Precision precision = self.getPriceCurrencyValue().getQuantityPrecision() def getDictList(category_list): ''' This method return a list of dict wich are composed with the given category as key and an integer as value. It's used to find datas in a two-dimensional table ''' list_of_dict = [] for category in category_list: base_amount_list=[x.getRelativeUrl() for x in\ self.portal_categories[category].objectValues()] list_of_dict.append(dict([[base, base_amount_list.index(base)] \ for base in base_amount_list])) return list_of_dict def getEmptyTwoDimentionalTable(nb_columns=0, nb_rows=0): ''' create a two-dimentional table with nb_columns columns and nb_rows rows all item of this table are set to None ex : with nb_columns=2 and nb_rows=4, this function returns : [[None, None, None, None], [None, None, None, None]] ''' base_amount = [] for i in range(nb_columns): line = [] for j in range(nb_rows): line.append(None) base_amount.append(line) return base_amount # this dict permit to have the index of the category in the table column, row = getDictList(['tax_category', 'base_amount']) base_amount_table = getEmptyTwoDimentionalTable(nb_columns=len(column), nb_rows=len(row)) def sortByIntIndex(a, b): return cmp(a.getIntIndex(), b.getIntIndex()) base_amount_list = self.portal_categories['base_amount'].contentValues() base_amount_list.sort(sortByIntIndex) # it's important to get the editable lines to know if they contribute to # a base_amount (this is required to do the calcul later) # get edited lines: paysheetline_list = self.contentValues(portal_type = ['Pay Sheet Line']) for paysheetline in paysheetline_list: service = paysheetline.getResourceValue() base_amount_list = service.getBaseAmountList(base=1) for base_amount in base_amount_list: if not row.has_key(base_amount): raise ValueError, "Unable to find `%s` base_amount category" % \ base_amount paysheetcell_list = paysheetline.contentValues(portal_type = \ ['Pay Sheet Cell']) for paysheetcell in paysheetcell_list: tax_category = paysheetcell.getTaxCategory(base=1) if tax_category and paysheetcell.getQuantity(): old_val = base_amount_table[column[tax_category]][row[base_amount]] if old_val is not None: new_val = old_val + paysheetcell.getQuantity() else: new_val = paysheetcell.getQuantity() base_amount_table[column[tax_category]][row[base_amount]] = new_val # get not editables model lines model = self.getSpecialiseValue() model_line_list = model.contentValues(portal_type='Pay Sheet Model Line', sort_on='int_index') model_line_list = [line for line in model_line_list if not line.getEditable()] pay_sheet_line_list = [] employee_tax_amount = 0 # main loop : find all informations and create cell and PaySheetLines for model_line in model_line_list: cell_list = [] # test with predicate if this model line could be applied if not model_line.test(self,): # This line should not be used continue service = model_line.getResourceValue() title = model_line.getTitleOrId() id = model_line.getId() base_amount_list = model_line.getBaseAmountList() res = service.getRelativeUrl() if model_line.getDescription(): desc = ''.join(model_line.getDescription()) # if the model_line description is empty, the payroll service # description is used else: desc = ''.join(service.getDescription()) variation_share_list = model_line.getVariationCategoryList(\ base_category_list=['tax_category',]) variation_slice_list = model_line.getVariationCategoryList(\ base_category_list=['salary_range',]) for share in variation_share_list: base_application = None for slice in variation_slice_list: #get the amount of application for this line base_application_list = model_line.getBaseAmountList(base=1) if base_application is None: base_application = 0 for base in model_line.getBaseAmountList(base=1): if base_amount_table[column[share]][row[base]] is not None: base_application += \ base_amount_table[column[share]][row[base]] cell = model_line.getCell(slice, share) if cell is not None: # get the slice : model_slice = None model_slice = model_line.getParentValue().getCell(slice) quantity = 0.0 price = 0.0 if model_slice is not None: model_slice_min = model_slice.getQuantityRangeMin() model_slice_max = model_slice.getQuantityRangeMax() quantity = cell.getQuantity() price = cell.getPrice() ###################### # calculation part : # ###################### script_name = model.getLocalizedCalculationScriptId() if script_name is None or \ getattr(self, script_name, None) is None: # if no calculation script found, use a default method : if not quantity: if base_application <= model_slice_max: quantity = base_application else: quantity = model_slice_max else: localized_calculation_script = getattr(self, script_name, None) quantity, price = localized_calculation_script(\ paysheet=self, model_slice_min = model_slice_min, model_slice_max=model_slice_max, quantity=quantity, price=price, model_line=model_line) # Cell creation : # Define an empty new cell new_cell = { 'axe_list' : [share, slice] , 'quantity' : quantity , 'price' : price } cell_list.append(new_cell) #XXX this is a hack to have the net salary base_list = model_line.getResourceValue().getBaseAmountList() if price is not None and 'employee_share' in share and\ not ('base_salary' in base_list or\ 'bonus' in base_list or\ 'gross_salary' in base_list): employee_tax_amount += round((price * quantity), precision) # update base participation base_participation_list = service.getBaseAmountList(base=1) for base_participation in base_participation_list: old_val = \ base_amount_table[column[share]][row[base_participation]] if quantity: if old_val is not None: new_val = old_val + quantity else: new_val = quantity if price: if old_val is not None: new_val = round((old_val + quantity*price), precision) base_amount_table[column[share]][row[base_participation]]= \ new_val base_amount_table[column[share]][row[base_participation]] = \ new_val # decrease the base_application used for this model line if cell is not None : base_application -= quantity if cell_list: # create the PaySheetLine pay_sheet_line = self.createPaySheetLine( title = title, res = res, desc = desc, base_amount_list = base_amount_list, cell_list = cell_list, ) pay_sheet_line_list.append(pay_sheet_line) # create a line of the total tax payed by the employee # this hack permit to calculate the net salary new_cell = { 'axe_list' : [ 'tax_category/employer_share', 'salary_range/france/forfait'] , 'quantity' : employee_tax_amount , 'price' : 1 } pay_sheet_line = self.createPaySheetLine( title = 'total employee contributions' , res = 'payroll_service_module/total_employee_contributions' , desc = 'this line permit to calculate the sum of the employer' ' share and permit to calculate the net salary' , cell_list = [new_cell]) return pay_sheet_line_list