Merge lp:~carlos-realsystems/openobject-server/trunk-bugfix-954189 into lp:openobject-server

Proposed by Carlos Contreras
Status: Rejected
Rejected by: Fabien (Open ERP)
Proposed branch: lp:~carlos-realsystems/openobject-server/trunk-bugfix-954189
Merge into: lp:openobject-server
Diff against target: 4988936 lines (has conflicts)
Conflict adding file .bzrignore.  Moved existing file to .bzrignore.moved.
To merge this branch: bzr merge lp:~carlos-realsystems/openobject-server/trunk-bugfix-954189
Reviewer Review Type Date Requested Status
Lorenzo Battistini (community) Needs Fixing
OpenERP Core Team Pending
Review via email: mp+127963@code.launchpad.net

Commit message

[FIX] Solves the problem that when you make an purchase order select create and then create an purchase order line.
When you select a product in purchase orders got allways "Selected UOM does not belong to the same category as the product UOM " but the UOM are the same

This error rises because uom_id always have a default UoM PCE and never gets true.

Description of the change

Error #954189:

In 6.1 i have the Problem that when i make an purchase order select create and then create an purchase order line.

When i select then a product i got allways "Selected UOM does not belong to the same category as the product UOM "
but the UOM are the same ?

This error rises because uom_id always have a default UoM PCE and never gets true on line 765:

- if not uom_id:
- uom_id = product_uom_po_id

It solves adding one condition and it doesnt cause harm:

+ if not uom_id or uom_id != product_uom_po_id:
+ uom_id = product_uom_po_id

To post a comment you must log in.
Revision history for this message
Lorenzo Battistini (elbati) wrote :

Hello Carlos, proposal has conflicts.

review: Needs Fixing
Revision history for this message
Lorenzo Battistini (elbati) wrote :

On 10/08/2012 07:48 PM, Carlos Enrique Contreras Vara wrote:
> Dear Lorenzo
>
> How can I do a merge correctly without conflicts because its only one
> line o code of purchase/purchase.py that has been modified.
>

You changed a file from openobject-addons/6.1 branch and proposed to
merge it into openobject-server/trunk.

You should put your branch under the openobject-addons project
(something like
lp:~carlos-realsystems/openobject-addons/6.1-bugfix-954189) and propose
to merge into openobject-addons/6.1 branch.

Regards

--
Lorenzo Battistini <email address hidden>
Via Brüsighell - 6807 - Taverne - Switzerland
Agile Business Group sagl http://www.agilebg.com

Revision history for this message
Fabien (Open ERP) (fp-tinyerp) wrote :

reject it as 4m lines of diff.
Your merge proposal is probably not clean, you should make feature branches.

Unmerged revisions

7011. By Carlos Contreras

[FIX] fixed Selected UOM does not belong to the same category as the product UOM on purchase order

7010. By Launchpad Translations on behalf of openerp

Launchpad automatic translations update.

7009. By Xavier ALT

[MERGE] OPW 50562: stock: fix picking type for chained picking

  forward port of v6.0 revid: <email address hidden>

7008. By Launchpad Translations on behalf of openerp

Launchpad automatic translations update.

7007. By Launchpad Translations on behalf of openerp

Launchpad automatic translations update.

7006. By Launchpad Translations on behalf of openerp

Launchpad automatic translations update.

7005. By Launchpad Translations on behalf of openerp

Launchpad automatic translations update.

7004. By Olivier Dony (Odoo)

[FIX] OPW 574637: account_analytic_plans: remove the strange behavior of instance editing - a thing of the past

7003. By Launchpad Translations on behalf of openerp

Launchpad automatic translations update.

7002. By Launchpad Translations on behalf of openerp

Launchpad automatic translations update.

Preview Diff

[H/L] Next/Prev Comment, [J/K] Next/Prev File, [N/P] Next/Prev Hunk
1=== added file '.bzrignore'
2--- .bzrignore 1970-01-01 00:00:00 +0000
3+++ .bzrignore 2012-10-04 09:12:55 +0000
4@@ -0,0 +1,1 @@
5+.*
6
7=== renamed file '.bzrignore' => '.bzrignore.moved'
8=== added directory 'account'
9=== added file 'account/__init__.py'
10--- account/__init__.py 1970-01-01 00:00:00 +0000
11+++ account/__init__.py 2012-10-04 09:12:55 +0000
12@@ -0,0 +1,40 @@
13+# -*- coding: utf-8 -*-
14+##############################################################################
15+#
16+# OpenERP, Open Source Management Solution
17+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
18+#
19+# This program is free software: you can redistribute it and/or modify
20+# it under the terms of the GNU Affero General Public License as
21+# published by the Free Software Foundation, either version 3 of the
22+# License, or (at your option) any later version.
23+#
24+# This program is distributed in the hope that it will be useful,
25+# but WITHOUT ANY WARRANTY; without even the implied warranty of
26+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
27+# GNU Affero General Public License for more details.
28+#
29+# You should have received a copy of the GNU Affero General Public License
30+# along with this program. If not, see <http://www.gnu.org/licenses/>.
31+#
32+##############################################################################
33+
34+import account
35+import installer
36+import project
37+import partner
38+import account_invoice
39+import account_bank_statement
40+import account_bank
41+import account_cash_statement
42+import account_move_line
43+import account_analytic_line
44+import account_financial_report
45+import wizard
46+import report
47+import product
48+import ir_sequence
49+import company
50+import res_currency
51+import edi
52+# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
53
54=== added file 'account/__openerp__.py'
55--- account/__openerp__.py 1970-01-01 00:00:00 +0000
56+++ account/__openerp__.py 2012-10-04 09:12:55 +0000
57@@ -0,0 +1,157 @@
58+# -*- coding: utf-8 -*-
59+##############################################################################
60+#
61+# OpenERP, Open Source Management Solution
62+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
63+#
64+# This program is free software: you can redistribute it and/or modify
65+# it under the terms of the GNU Affero General Public License as
66+# published by the Free Software Foundation, either version 3 of the
67+# License, or (at your option) any later version.
68+#
69+# This program is distributed in the hope that it will be useful,
70+# but WITHOUT ANY WARRANTY; without even the implied warranty of
71+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
72+# GNU Affero General Public License for more details.
73+#
74+# You should have received a copy of the GNU Affero General Public License
75+# along with this program. If not, see <http://www.gnu.org/licenses/>.
76+#
77+##############################################################################
78+{
79+ "name" : "eInvoicing",
80+ "version" : "1.1",
81+ "author" : "OpenERP SA",
82+ "category": 'Accounting & Finance',
83+ 'complexity': "easy",
84+ "description": """
85+Accounting and Financial Management.
86+====================================
87+
88+Financial and accounting module that covers:
89+--------------------------------------------
90+General accountings
91+Cost / Analytic accounting
92+Third party accounting
93+Taxes management
94+Budgets
95+Customer and Supplier Invoices
96+Bank statements
97+Reconciliation process by partner
98+
99+Creates a dashboard for accountants that includes:
100+--------------------------------------------------
101+* List of Customer Invoice to Approve
102+* Company Analysis
103+* Graph of Aged Receivables
104+* Graph of Treasury
105+
106+The processes like maintaining of general ledger is done through the defined financial Journals (entry move line or
107+grouping is maintained through journal) for a particular financial year and for preparation of vouchers there is a
108+module named account_voucher.
109+ """,
110+ 'website': 'http://www.openerp.com',
111+ 'images' : ['images/accounts.jpeg','images/bank_statement.jpeg','images/cash_register.jpeg','images/chart_of_accounts.jpeg','images/customer_invoice.jpeg','images/journal_entries.jpeg'],
112+ 'init_xml': [],
113+ "depends" : ["base_setup", "product", "analytic", "process", "board", "edi"],
114+ 'update_xml': [
115+ 'security/account_security.xml',
116+ 'security/ir.model.access.csv',
117+ 'account_menuitem.xml',
118+ 'report/account_invoice_report_view.xml',
119+ 'report/account_entries_report_view.xml',
120+ 'report/account_treasury_report_view.xml',
121+ 'report/account_report_view.xml',
122+ 'report/account_analytic_entries_report_view.xml',
123+ 'wizard/account_move_bank_reconcile_view.xml',
124+ 'wizard/account_use_model_view.xml',
125+ 'account_installer.xml',
126+ 'wizard/account_period_close_view.xml',
127+ 'account_view.xml',
128+ 'account_report.xml',
129+ 'account_financial_report_data.xml',
130+ 'wizard/account_report_common_view.xml',
131+ 'wizard/account_invoice_refund_view.xml',
132+ 'wizard/account_fiscalyear_close_state.xml',
133+ 'wizard/account_chart_view.xml',
134+ 'wizard/account_tax_chart_view.xml',
135+ 'wizard/account_move_journal_view.xml',
136+ 'wizard/account_move_line_reconcile_select_view.xml',
137+ 'wizard/account_open_closed_fiscalyear_view.xml',
138+ 'wizard/account_move_line_unreconcile_select_view.xml',
139+ 'wizard/account_vat_view.xml',
140+ 'wizard/account_report_print_journal_view.xml',
141+ 'wizard/account_report_general_journal_view.xml',
142+ 'wizard/account_report_central_journal_view.xml',
143+ 'wizard/account_subscription_generate_view.xml',
144+ 'wizard/account_fiscalyear_close_view.xml',
145+ 'wizard/account_state_open_view.xml',
146+ 'wizard/account_journal_select_view.xml',
147+ 'wizard/account_change_currency_view.xml',
148+ 'wizard/account_validate_move_view.xml',
149+ 'wizard/account_unreconcile_view.xml',
150+ 'wizard/account_report_general_ledger_view.xml',
151+ 'wizard/account_invoice_state_view.xml',
152+ 'wizard/account_report_partner_balance_view.xml',
153+ 'wizard/account_report_account_balance_view.xml',
154+ 'wizard/account_report_aged_partner_balance_view.xml',
155+ 'wizard/account_report_partner_ledger_view.xml',
156+ 'wizard/account_reconcile_view.xml',
157+ 'wizard/account_reconcile_partner_process_view.xml',
158+ 'wizard/account_automatic_reconcile_view.xml',
159+ 'wizard/account_financial_report_view.xml',
160+ 'project/wizard/project_account_analytic_line_view.xml',
161+ 'account_end_fy.xml',
162+ 'account_invoice_view.xml',
163+ 'partner_view.xml',
164+ 'data/account_data.xml',
165+ 'data/data_account_type.xml',
166+ 'account_invoice_workflow.xml',
167+ 'project/project_view.xml',
168+ 'project/project_report.xml',
169+ 'project/wizard/account_analytic_balance_report_view.xml',
170+ 'project/wizard/account_analytic_cost_ledger_view.xml',
171+ 'project/wizard/account_analytic_inverted_balance_report.xml',
172+ 'project/wizard/account_analytic_journal_report_view.xml',
173+ 'project/wizard/account_analytic_cost_ledger_for_journal_report_view.xml',
174+ 'project/wizard/account_analytic_chart_view.xml',
175+ 'product_view.xml',
176+ 'account_assert_test.xml',
177+ 'process/statement_process.xml',
178+ 'process/customer_invoice_process.xml',
179+ 'process/supplier_invoice_process.xml',
180+ 'ir_sequence_view.xml',
181+ 'company_view.xml',
182+ 'board_account_view.xml',
183+ "edi/invoice_action_data.xml",
184+ "account_bank_view.xml",
185+ "account_pre_install.yml"
186+ ],
187+ 'demo_xml': [
188+ 'demo/account_demo.xml',
189+ 'project/project_demo.xml',
190+ 'project/analytic_account_demo.xml',
191+ 'demo/account_minimal.xml',
192+ 'demo/account_invoice_demo.xml',
193+ #'account_unit_test.xml',
194+ ],
195+ 'test': [
196+ 'test/account_customer_invoice.yml',
197+ 'test/account_supplier_invoice.yml',
198+ 'test/account_change_currency.yml',
199+ 'test/chart_of_account.yml',
200+ 'test/account_period_close.yml',
201+ 'test/account_use_model.yml',
202+ 'test/account_validate_account_move.yml',
203+ 'test/account_fiscalyear_close.yml',
204+ 'test/account_bank_statement.yml',
205+ 'test/account_cash_statement.yml',
206+ 'test/test_edi_invoice.yml',
207+ 'test/account_report.yml',
208+ 'test/account_fiscalyear_close_state.yml', #last test, as it will definitively close the demo fiscalyear
209+ ],
210+ 'installable': True,
211+ 'auto_install': False,
212+ 'certificate': '0080331923549',
213+}
214+# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
215
216=== added file 'account/account.py'
217--- account/account.py 1970-01-01 00:00:00 +0000
218+++ account/account.py 2012-10-04 09:12:55 +0000
219@@ -0,0 +1,3499 @@
220+# -*- coding: utf-8 -*-
221+##############################################################################
222+#
223+# OpenERP, Open Source Management Solution
224+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
225+#
226+# This program is free software: you can redistribute it and/or modify
227+# it under the terms of the GNU Affero General Public License as
228+# published by the Free Software Foundation, either version 3 of the
229+# License, or (at your option) any later version.
230+#
231+# This program is distributed in the hope that it will be useful,
232+# but WITHOUT ANY WARRANTY; without even the implied warranty of
233+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
234+# GNU Affero General Public License for more details.
235+#
236+# You should have received a copy of the GNU Affero General Public License
237+# along with this program. If not, see <http://www.gnu.org/licenses/>.
238+#
239+##############################################################################
240+
241+import time
242+from datetime import datetime
243+from dateutil.relativedelta import relativedelta
244+from operator import itemgetter
245+
246+import netsvc
247+import pooler
248+from osv import fields, osv
249+import decimal_precision as dp
250+from tools.translate import _
251+
252+def check_cycle(self, cr, uid, ids, context=None):
253+ """ climbs the ``self._table.parent_id`` chains for 100 levels or
254+ until it can't find any more parent(s)
255+
256+ Returns true if it runs out of parents (no cycle), false if
257+ it can recurse 100 times without ending all chains
258+ """
259+ level = 100
260+ while len(ids):
261+ cr.execute('SELECT DISTINCT parent_id '\
262+ 'FROM '+self._table+' '\
263+ 'WHERE id IN %s '\
264+ 'AND parent_id IS NOT NULL',(tuple(ids),))
265+ ids = map(itemgetter(0), cr.fetchall())
266+ if not level:
267+ return False
268+ level -= 1
269+ return True
270+
271+class account_payment_term(osv.osv):
272+ _name = "account.payment.term"
273+ _description = "Payment Term"
274+ _columns = {
275+ 'name': fields.char('Payment Term', size=64, translate=True, required=True),
276+ 'active': fields.boolean('Active', help="If the active field is set to False, it will allow you to hide the payment term without removing it."),
277+ 'note': fields.text('Description', translate=True),
278+ 'line_ids': fields.one2many('account.payment.term.line', 'payment_id', 'Terms'),
279+ }
280+ _defaults = {
281+ 'active': 1,
282+ }
283+ _order = "name"
284+
285+ def compute(self, cr, uid, id, value, date_ref=False, context=None):
286+ if not date_ref:
287+ date_ref = datetime.now().strftime('%Y-%m-%d')
288+ pt = self.browse(cr, uid, id, context=context)
289+ amount = value
290+ result = []
291+ obj_precision = self.pool.get('decimal.precision')
292+ for line in pt.line_ids:
293+ prec = obj_precision.precision_get(cr, uid, 'Account')
294+ if line.value == 'fixed':
295+ amt = round(line.value_amount, prec)
296+ elif line.value == 'procent':
297+ amt = round(value * line.value_amount, prec)
298+ elif line.value == 'balance':
299+ amt = round(amount, prec)
300+ if amt:
301+ next_date = (datetime.strptime(date_ref, '%Y-%m-%d') + relativedelta(days=line.days))
302+ if line.days2 < 0:
303+ next_first_date = next_date + relativedelta(day=1,months=1) #Getting 1st of next month
304+ next_date = next_first_date + relativedelta(days=line.days2)
305+ if line.days2 > 0:
306+ next_date += relativedelta(day=line.days2, months=1)
307+ result.append( (next_date.strftime('%Y-%m-%d'), amt) )
308+ amount -= amt
309+ return result
310+
311+account_payment_term()
312+
313+class account_payment_term_line(osv.osv):
314+ _name = "account.payment.term.line"
315+ _description = "Payment Term Line"
316+ _columns = {
317+ 'name': fields.char('Line Name', size=32, required=True),
318+ 'sequence': fields.integer('Sequence', required=True, help="The sequence field is used to order the payment term lines from the lowest sequences to the higher ones"),
319+ 'value': fields.selection([('procent', 'Percent'),
320+ ('balance', 'Balance'),
321+ ('fixed', 'Fixed Amount')], 'Valuation',
322+ required=True, help="""Select here the kind of valuation related to this payment term line. Note that you should have your last line with the type 'Balance' to ensure that the whole amount will be threated."""),
323+
324+ 'value_amount': fields.float('Amount To Pay', digits_compute=dp.get_precision('Payment Term'), help="For percent enter a ratio between 0-1."),
325+ 'days': fields.integer('Number of Days', required=True, help="Number of days to add before computation of the day of month." \
326+ "If Date=15/01, Number of Days=22, Day of Month=-1, then the due date is 28/02."),
327+ 'days2': fields.integer('Day of the Month', required=True, help="Day of the month, set -1 for the last day of the current month. If it's positive, it gives the day of the next month. Set 0 for net days (otherwise it's based on the beginning of the month)."),
328+ 'payment_id': fields.many2one('account.payment.term', 'Payment Term', required=True, select=True),
329+ }
330+ _defaults = {
331+ 'value': 'balance',
332+ 'sequence': 5,
333+ 'days2': 0,
334+ }
335+ _order = "sequence"
336+
337+ def _check_percent(self, cr, uid, ids, context=None):
338+ obj = self.browse(cr, uid, ids[0], context=context)
339+ if obj.value == 'procent' and ( obj.value_amount < 0.0 or obj.value_amount > 1.0):
340+ return False
341+ return True
342+
343+ _constraints = [
344+ (_check_percent, 'Percentages for Payment Term Line must be between 0 and 1, Example: 0.02 for 2% ', ['value_amount']),
345+ ]
346+
347+account_payment_term_line()
348+
349+class account_account_type(osv.osv):
350+ _name = "account.account.type"
351+ _description = "Account Type"
352+
353+ def _get_current_report_type(self, cr, uid, ids, name, arg, context=None):
354+ obj_data = self.pool.get('ir.model.data')
355+ obj_financial_report = self.pool.get('account.financial.report')
356+ res = {}
357+ financial_report_ref = {
358+ 'asset': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_assets0')[1], context=context),
359+ 'liability': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_liability0')[1], context=context),
360+ 'income': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_income0')[1], context=context),
361+ 'expense': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_expense0')[1], context=context),
362+ }
363+ for record in self.browse(cr, uid, ids, context=context):
364+ res[record.id] = 'none'
365+ for key, financial_report in financial_report_ref.items():
366+ list_ids = [x.id for x in financial_report.account_type_ids]
367+ if record.id in list_ids:
368+ res[record.id] = key
369+ return res
370+
371+ def _save_report_type(self, cr, uid, account_type_id, field_name, field_value, arg, context=None):
372+ obj_data = self.pool.get('ir.model.data')
373+ obj_financial_report = self.pool.get('account.financial.report')
374+ #unlink if it exists somewhere in the financial reports related to BS or PL
375+ financial_report_ref = {
376+ 'asset': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_assets0')[1], context=context),
377+ 'liability': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_liability0')[1], context=context),
378+ 'income': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_income0')[1], context=context),
379+ 'expense': obj_financial_report.browse(cr, uid, obj_data.get_object_reference(cr, uid, 'account','account_financial_report_expense0')[1], context=context),
380+ }
381+ for key, financial_report in financial_report_ref.items():
382+ list_ids = [x.id for x in financial_report.account_type_ids]
383+ if account_type_id in list_ids:
384+ obj_financial_report.write(cr, uid, [financial_report.id], {'account_type_ids': [(3, account_type_id)]})
385+ #write it in the good place
386+ if field_value != 'none':
387+ return obj_financial_report.write(cr, uid, [financial_report_ref[field_value].id], {'account_type_ids': [(4, account_type_id)]})
388+
389+ _columns = {
390+ 'name': fields.char('Account Type', size=64, required=True, translate=True),
391+ 'code': fields.char('Code', size=32, required=True, select=True),
392+ 'close_method': fields.selection([('none', 'None'), ('balance', 'Balance'), ('detail', 'Detail'), ('unreconciled', 'Unreconciled')], 'Deferral Method', required=True, help="""Set here the method that will be used to generate the end of year journal entries for all the accounts of this type.
393+
394+ 'None' means that nothing will be done.
395+ 'Balance' will generally be used for cash accounts.
396+ 'Detail' will copy each existing journal item of the previous year, even the reconciled ones.
397+ 'Unreconciled' will copy only the journal items that were unreconciled on the first day of the new fiscal year."""),
398+ 'report_type': fields.function(_get_current_report_type, fnct_inv=_save_report_type, type='selection', string='P&L / BS Category',
399+ selection= [('none','/'),
400+ ('income', _('Profit & Loss (Income account)')),
401+ ('expense', _('Profit & Loss (Expense account)')),
402+ ('asset', _('Balance Sheet (Asset account)')),
403+ ('liability', _('Balance Sheet (Liability account)'))], help="This field is used to generate legal reports: profit and loss, balance sheet.", required=True),
404+ 'note': fields.text('Description'),
405+ }
406+ _defaults = {
407+ 'close_method': 'none',
408+ 'report_type': 'none',
409+ }
410+ _order = "code"
411+
412+account_account_type()
413+
414+def _code_get(self, cr, uid, context=None):
415+ acc_type_obj = self.pool.get('account.account.type')
416+ ids = acc_type_obj.search(cr, uid, [])
417+ res = acc_type_obj.read(cr, uid, ids, ['code', 'name'], context=context)
418+ return [(r['code'], r['name']) for r in res]
419+
420+#----------------------------------------------------------
421+# Accounts
422+#----------------------------------------------------------
423+
424+class account_tax(osv.osv):
425+ _name = 'account.tax'
426+account_tax()
427+
428+class account_account(osv.osv):
429+ _order = "parent_left"
430+ _parent_order = "code"
431+ _name = "account.account"
432+ _description = "Account"
433+ _parent_store = True
434+ logger = netsvc.Logger()
435+
436+ def search(self, cr, uid, args, offset=0, limit=None, order=None,
437+ context=None, count=False):
438+ if context is None:
439+ context = {}
440+ pos = 0
441+
442+ while pos < len(args):
443+
444+ if args[pos][0] == 'code' and args[pos][1] in ('like', 'ilike') and args[pos][2]:
445+ args[pos] = ('code', '=like', str(args[pos][2].replace('%', ''))+'%')
446+ if args[pos][0] == 'journal_id':
447+ if not args[pos][2]:
448+ del args[pos]
449+ continue
450+ jour = self.pool.get('account.journal').browse(cr, uid, args[pos][2], context=context)
451+ if (not (jour.account_control_ids or jour.type_control_ids)) or not args[pos][2]:
452+ args[pos] = ('type','not in',('consolidation','view'))
453+ continue
454+ ids3 = map(lambda x: x.id, jour.type_control_ids)
455+ ids1 = super(account_account, self).search(cr, uid, [('user_type', 'in', ids3)])
456+ ids1 += map(lambda x: x.id, jour.account_control_ids)
457+ args[pos] = ('id', 'in', ids1)
458+ pos += 1
459+
460+ if context and context.has_key('consolidate_children'): #add consolidated children of accounts
461+ ids = super(account_account, self).search(cr, uid, args, offset, limit,
462+ order, context=context, count=count)
463+ for consolidate_child in self.browse(cr, uid, context['account_id'], context=context).child_consol_ids:
464+ ids.append(consolidate_child.id)
465+ return ids
466+
467+ return super(account_account, self).search(cr, uid, args, offset, limit,
468+ order, context=context, count=count)
469+
470+ def _get_children_and_consol(self, cr, uid, ids, context=None):
471+ #this function search for all the children and all consolidated children (recursively) of the given account ids
472+ ids2 = self.search(cr, uid, [('parent_id', 'child_of', ids)], context=context)
473+ ids3 = []
474+ for rec in self.browse(cr, uid, ids2, context=context):
475+ for child in rec.child_consol_ids:
476+ ids3.append(child.id)
477+ if ids3:
478+ ids3 = self._get_children_and_consol(cr, uid, ids3, context)
479+ return ids2 + ids3
480+
481+ def __compute(self, cr, uid, ids, field_names, arg=None, context=None,
482+ query='', query_params=()):
483+ """ compute the balance, debit and/or credit for the provided
484+ account ids
485+ Arguments:
486+ `ids`: account ids
487+ `field_names`: the fields to compute (a list of any of
488+ 'balance', 'debit' and 'credit')
489+ `arg`: unused fields.function stuff
490+ `query`: additional query filter (as a string)
491+ `query_params`: parameters for the provided query string
492+ (__compute will handle their escaping) as a
493+ tuple
494+ """
495+ mapping = {
496+ 'balance': "COALESCE(SUM(l.debit),0) - COALESCE(SUM(l.credit), 0) as balance",
497+ 'debit': "COALESCE(SUM(l.debit), 0) as debit",
498+ 'credit': "COALESCE(SUM(l.credit), 0) as credit",
499+ # by convention, foreign_balance is 0 when the account has no secondary currency, because the amounts may be in different currencies
500+ 'foreign_balance': "(SELECT CASE WHEN currency_id IS NULL THEN 0 ELSE COALESCE(SUM(l.amount_currency), 0) END FROM account_account WHERE id IN (l.account_id)) as foreign_balance",
501+ }
502+ #get all the necessary accounts
503+ children_and_consolidated = self._get_children_and_consol(cr, uid, ids, context=context)
504+ #compute for each account the balance/debit/credit from the move lines
505+ accounts = {}
506+ res = {}
507+ null_result = dict((fn, 0.0) for fn in field_names)
508+ if children_and_consolidated:
509+ aml_query = self.pool.get('account.move.line')._query_get(cr, uid, context=context)
510+
511+ wheres = [""]
512+ if query.strip():
513+ wheres.append(query.strip())
514+ if aml_query.strip():
515+ wheres.append(aml_query.strip())
516+ filters = " AND ".join(wheres)
517+ self.logger.notifyChannel('addons.'+self._name, netsvc.LOG_DEBUG,
518+ 'Filters: %s'%filters)
519+ # IN might not work ideally in case there are too many
520+ # children_and_consolidated, in that case join on a
521+ # values() e.g.:
522+ # SELECT l.account_id as id FROM account_move_line l
523+ # INNER JOIN (VALUES (id1), (id2), (id3), ...) AS tmp (id)
524+ # ON l.account_id = tmp.id
525+ # or make _get_children_and_consol return a query and join on that
526+ request = ("SELECT l.account_id as id, " +\
527+ ', '.join(mapping.values()) +
528+ " FROM account_move_line l" \
529+ " WHERE l.account_id IN %s " \
530+ + filters +
531+ " GROUP BY l.account_id")
532+ params = (tuple(children_and_consolidated),) + query_params
533+ cr.execute(request, params)
534+ self.logger.notifyChannel('addons.'+self._name, netsvc.LOG_DEBUG,
535+ 'Status: %s'%cr.statusmessage)
536+
537+ for res in cr.dictfetchall():
538+ accounts[res['id']] = res
539+
540+ # consolidate accounts with direct children
541+ children_and_consolidated.reverse()
542+ brs = list(self.browse(cr, uid, children_and_consolidated, context=context))
543+ sums = {}
544+ currency_obj = self.pool.get('res.currency')
545+ while brs:
546+ current = brs.pop(0)
547+# can_compute = True
548+# for child in current.child_id:
549+# if child.id not in sums:
550+# can_compute = False
551+# try:
552+# brs.insert(0, brs.pop(brs.index(child)))
553+# except ValueError:
554+# brs.insert(0, child)
555+# if can_compute:
556+ for fn in field_names:
557+ sums.setdefault(current.id, {})[fn] = accounts.get(current.id, {}).get(fn, 0.0)
558+ for child in current.child_id:
559+ if child.company_id.currency_id.id == current.company_id.currency_id.id:
560+ sums[current.id][fn] += sums[child.id][fn]
561+ else:
562+ sums[current.id][fn] += currency_obj.compute(cr, uid, child.company_id.currency_id.id, current.company_id.currency_id.id, sums[child.id][fn], context=context)
563+
564+ # as we have to relay on values computed before this is calculated separately than previous fields
565+ if current.currency_id and current.exchange_rate and \
566+ ('adjusted_balance' in field_names or 'unrealized_gain_loss' in field_names):
567+ # Computing Adjusted Balance and Unrealized Gains and losses
568+ # Adjusted Balance = Foreign Balance / Exchange Rate
569+ # Unrealized Gains and losses = Adjusted Balance - Balance
570+ adj_bal = sums[current.id].get('foreign_balance', 0.0) / current.exchange_rate
571+ sums[current.id].update({'adjusted_balance': adj_bal, 'unrealized_gain_loss': adj_bal - sums[current.id].get('balance', 0.0)})
572+
573+ for id in ids:
574+ res[id] = sums.get(id, null_result)
575+ else:
576+ for id in ids:
577+ res[id] = null_result
578+ return res
579+
580+ def _get_company_currency(self, cr, uid, ids, field_name, arg, context=None):
581+ result = {}
582+ for rec in self.browse(cr, uid, ids, context=context):
583+ result[rec.id] = (rec.company_id.currency_id.id,rec.company_id.currency_id.symbol)
584+ return result
585+
586+ def _get_child_ids(self, cr, uid, ids, field_name, arg, context=None):
587+ result = {}
588+ for record in self.browse(cr, uid, ids, context=context):
589+ if record.child_parent_ids:
590+ result[record.id] = [x.id for x in record.child_parent_ids]
591+ else:
592+ result[record.id] = []
593+
594+ if record.child_consol_ids:
595+ for acc in record.child_consol_ids:
596+ if acc.id not in result[record.id]:
597+ result[record.id].append(acc.id)
598+
599+ return result
600+
601+ def _get_level(self, cr, uid, ids, field_name, arg, context=None):
602+ res = {}
603+ for account in self.browse(cr, uid, ids, context=context):
604+ #we may not know the level of the parent at the time of computation, so we
605+ # can't simply do res[account.id] = account.parent_id.level + 1
606+ level = 0
607+ parent = account.parent_id
608+ while parent:
609+ level += 1
610+ parent = parent.parent_id
611+ res[account.id] = level
612+ return res
613+
614+ def _set_credit_debit(self, cr, uid, account_id, name, value, arg, context=None):
615+ if context.get('config_invisible', True):
616+ return True
617+
618+ account = self.browse(cr, uid, account_id, context=context)
619+ diff = value - getattr(account,name)
620+ if not diff:
621+ return True
622+
623+ journal_obj = self.pool.get('account.journal')
624+ jids = journal_obj.search(cr, uid, [('type','=','situation'),('centralisation','=',1),('company_id','=',account.company_id.id)], context=context)
625+ if not jids:
626+ raise osv.except_osv(_('Error!'),_("You need an Opening journal with centralisation checked to set the initial balance!"))
627+
628+ period_obj = self.pool.get('account.period')
629+ pids = period_obj.search(cr, uid, [('special','=',True),('company_id','=',account.company_id.id)], context=context)
630+ if not pids:
631+ raise osv.except_osv(_('Error!'),_("No opening/closing period defined, please create one to set the initial balance!"))
632+
633+ move_obj = self.pool.get('account.move.line')
634+ move_id = move_obj.search(cr, uid, [
635+ ('journal_id','=',jids[0]),
636+ ('period_id','=',pids[0]),
637+ ('account_id','=', account_id),
638+ (name,'>', 0.0),
639+ ('name','=', _('Opening Balance'))
640+ ], context=context)
641+ if move_id:
642+ move = move_obj.browse(cr, uid, move_id[0], context=context)
643+ move_obj.write(cr, uid, move_id[0], {
644+ name: diff+getattr(move,name)
645+ }, context=context)
646+ else:
647+ if diff<0.0:
648+ raise osv.except_osv(_('Error!'),_("Unable to adapt the initial balance (negative value)!"))
649+ nameinv = (name=='credit' and 'debit') or 'credit'
650+ move_id = move_obj.create(cr, uid, {
651+ 'name': _('Opening Balance'),
652+ 'account_id': account_id,
653+ 'journal_id': jids[0],
654+ 'period_id': pids[0],
655+ name: diff,
656+ nameinv: 0.0
657+ }, context=context)
658+ return True
659+
660+ _columns = {
661+ 'name': fields.char('Name', size=256, required=True, select=True),
662+ 'currency_id': fields.many2one('res.currency', 'Secondary Currency', help="Forces all moves for this account to have this secondary currency."),
663+ 'code': fields.char('Code', size=64, required=True, select=1),
664+ 'type': fields.selection([
665+ ('view', 'View'),
666+ ('other', 'Regular'),
667+ ('receivable', 'Receivable'),
668+ ('payable', 'Payable'),
669+ ('liquidity','Liquidity'),
670+ ('consolidation', 'Consolidation'),
671+ ('closed', 'Closed'),
672+ ], 'Internal Type', required=True, help="The 'Internal Type' is used for features available on "\
673+ "different types of accounts: view can not have journal items, consolidation are accounts that "\
674+ "can have children accounts for multi-company consolidations, payable/receivable are for "\
675+ "partners accounts (for debit/credit computations), closed for depreciated accounts."),
676+ 'user_type': fields.many2one('account.account.type', 'Account Type', required=True,
677+ help="Account Type is used for information purpose, to generate "
678+ "country-specific legal reports, and set the rules to close a fiscal year and generate opening entries."),
679+ 'financial_report_ids': fields.many2many('account.financial.report', 'account_account_financial_report', 'account_id', 'report_line_id', 'Financial Reports'),
680+ 'parent_id': fields.many2one('account.account', 'Parent', ondelete='cascade', domain=[('type','=','view')]),
681+ 'child_parent_ids': fields.one2many('account.account','parent_id','Children'),
682+ 'child_consol_ids': fields.many2many('account.account', 'account_account_consol_rel', 'child_id', 'parent_id', 'Consolidated Children'),
683+ 'child_id': fields.function(_get_child_ids, type='many2many', relation="account.account", string="Child Accounts"),
684+ 'balance': fields.function(__compute, digits_compute=dp.get_precision('Account'), string='Balance', multi='balance'),
685+ 'credit': fields.function(__compute, fnct_inv=_set_credit_debit, digits_compute=dp.get_precision('Account'), string='Credit', multi='balance'),
686+ 'debit': fields.function(__compute, fnct_inv=_set_credit_debit, digits_compute=dp.get_precision('Account'), string='Debit', multi='balance'),
687+ 'foreign_balance': fields.function(__compute, digits_compute=dp.get_precision('Account'), string='Foreign Balance', multi='balance',
688+ help="Total amount (in Secondary currency) for transactions held in secondary currency for this account."),
689+ 'adjusted_balance': fields.function(__compute, digits_compute=dp.get_precision('Account'), string='Adjusted Balance', multi='balance',
690+ help="Total amount (in Company currency) for transactions held in secondary currency for this account."),
691+ 'unrealized_gain_loss': fields.function(__compute, digits_compute=dp.get_precision('Account'), string='Unrealized Gain or Loss', multi='balance',
692+ help="Value of Loss or Gain due to changes in exchange rate when doing multi-currency transactions."),
693+ 'reconcile': fields.boolean('Allow Reconciliation', help="Check this box if this account allows reconciliation of journal items."),
694+ 'exchange_rate': fields.related('currency_id', 'rate', type='float', string='Exchange Rate', digits=(12,6)),
695+ 'shortcut': fields.char('Shortcut', size=12),
696+ 'tax_ids': fields.many2many('account.tax', 'account_account_tax_default_rel',
697+ 'account_id', 'tax_id', 'Default Taxes'),
698+ 'note': fields.text('Note'),
699+ 'company_currency_id': fields.function(_get_company_currency, type='many2one', relation='res.currency', string='Company Currency'),
700+ 'company_id': fields.many2one('res.company', 'Company', required=True),
701+ 'active': fields.boolean('Active', select=2, help="If the active field is set to False, it will allow you to hide the account without removing it."),
702+
703+ 'parent_left': fields.integer('Parent Left', select=1),
704+ 'parent_right': fields.integer('Parent Right', select=1),
705+ 'currency_mode': fields.selection([('current', 'At Date'), ('average', 'Average Rate')], 'Outgoing Currencies Rate',
706+ help=
707+ 'This will select how the current currency rate for outgoing transactions is computed. '\
708+ 'In most countries the legal method is "average" but only a few software systems are able to '\
709+ 'manage this. So if you import from another software system you may have to use the rate at date. ' \
710+ 'Incoming transactions always use the rate at date.', \
711+ required=True),
712+ 'level': fields.function(_get_level, string='Level', method=True, type='integer',
713+ store={
714+ 'account.account': (_get_children_and_consol, ['level', 'parent_id'], 10),
715+ }),
716+ }
717+
718+ _defaults = {
719+ 'type': 'other',
720+ 'reconcile': False,
721+ 'active': True,
722+ 'currency_mode': 'current',
723+ 'company_id': lambda s, cr, uid, c: s.pool.get('res.company')._company_default_get(cr, uid, 'account.account', context=c),
724+ }
725+
726+ def _check_recursion(self, cr, uid, ids, context=None):
727+ obj_self = self.browse(cr, uid, ids[0], context=context)
728+ p_id = obj_self.parent_id and obj_self.parent_id.id
729+ if (obj_self in obj_self.child_consol_ids) or (p_id and (p_id is obj_self.id)):
730+ return False
731+ while(ids):
732+ cr.execute('SELECT DISTINCT child_id '\
733+ 'FROM account_account_consol_rel '\
734+ 'WHERE parent_id IN %s', (tuple(ids),))
735+ child_ids = map(itemgetter(0), cr.fetchall())
736+ c_ids = child_ids
737+ if (p_id and (p_id in c_ids)) or (obj_self.id in c_ids):
738+ return False
739+ while len(c_ids):
740+ s_ids = self.search(cr, uid, [('parent_id', 'in', c_ids)])
741+ if p_id and (p_id in s_ids):
742+ return False
743+ c_ids = s_ids
744+ ids = child_ids
745+ return True
746+
747+ def _check_type(self, cr, uid, ids, context=None):
748+ if context is None:
749+ context = {}
750+ accounts = self.browse(cr, uid, ids, context=context)
751+ for account in accounts:
752+ if account.child_id and account.type not in ('view', 'consolidation'):
753+ return False
754+ return True
755+
756+ def _check_account_type(self, cr, uid, ids, context=None):
757+ for account in self.browse(cr, uid, ids, context=context):
758+ if account.type in ('receivable', 'payable') and account.user_type.close_method != 'unreconciled':
759+ return False
760+ return True
761+
762+ _constraints = [
763+ (_check_recursion, 'Error ! You can not create recursive accounts.', ['parent_id']),
764+ (_check_type, 'Configuration Error! \nYou can not define children to an account with internal type different of "View"! ', ['type']),
765+ (_check_account_type, 'Configuration Error! \nYou can not select an account type with a deferral method different of "Unreconciled" for accounts with internal type "Payable/Receivable"! ', ['user_type','type']),
766+ ]
767+ _sql_constraints = [
768+ ('code_company_uniq', 'unique (code,company_id)', 'The code of the account must be unique per company !')
769+ ]
770+ def name_search(self, cr, user, name, args=None, operator='ilike', context=None, limit=100):
771+ if not args:
772+ args = []
773+ args = args[:]
774+ ids = []
775+ try:
776+ if name and str(name).startswith('partner:'):
777+ part_id = int(name.split(':')[1])
778+ part = self.pool.get('res.partner').browse(cr, user, part_id, context=context)
779+ args += [('id', 'in', (part.property_account_payable.id, part.property_account_receivable.id))]
780+ name = False
781+ if name and str(name).startswith('type:'):
782+ type = name.split(':')[1]
783+ args += [('type', '=', type)]
784+ name = False
785+ except:
786+ pass
787+ if name:
788+ ids = self.search(cr, user, [('code', '=like', name+"%")]+args, limit=limit)
789+ if not ids:
790+ ids = self.search(cr, user, [('shortcut', '=', name)]+ args, limit=limit)
791+ if not ids:
792+ ids = self.search(cr, user, [('name', operator, name)]+ args, limit=limit)
793+ if not ids and len(name.split()) >= 2:
794+ #Separating code and name of account for searching
795+ operand1,operand2 = name.split(' ',1) #name can contain spaces e.g. OpenERP S.A.
796+ ids = self.search(cr, user, [('code', operator, operand1), ('name', operator, operand2)]+ args, limit=limit)
797+ else:
798+ ids = self.search(cr, user, args, context=context, limit=limit)
799+ return self.name_get(cr, user, ids, context=context)
800+
801+ def name_get(self, cr, uid, ids, context=None):
802+ if not ids:
803+ return []
804+ reads = self.read(cr, uid, ids, ['name', 'code'], context=context)
805+ res = []
806+ for record in reads:
807+ name = record['name']
808+ if record['code']:
809+ name = record['code'] + ' ' + name
810+ res.append((record['id'], name))
811+ return res
812+
813+ def copy(self, cr, uid, id, default={}, context=None, done_list=[], local=False):
814+ account = self.browse(cr, uid, id, context=context)
815+ new_child_ids = []
816+ if not default:
817+ default = {}
818+ default = default.copy()
819+ default['code'] = (account['code'] or '') + '(copy)'
820+ if not local:
821+ done_list = []
822+ if account.id in done_list:
823+ return False
824+ done_list.append(account.id)
825+ if account:
826+ for child in account.child_id:
827+ child_ids = self.copy(cr, uid, child.id, default, context=context, done_list=done_list, local=True)
828+ if child_ids:
829+ new_child_ids.append(child_ids)
830+ default['child_parent_ids'] = [(6, 0, new_child_ids)]
831+ else:
832+ default['child_parent_ids'] = False
833+ return super(account_account, self).copy(cr, uid, id, default, context=context)
834+
835+ def _check_moves(self, cr, uid, ids, method, context=None):
836+ line_obj = self.pool.get('account.move.line')
837+ account_ids = self.search(cr, uid, [('id', 'child_of', ids)])
838+
839+ if line_obj.search(cr, uid, [('account_id', 'in', account_ids)]):
840+ if method == 'write':
841+ raise osv.except_osv(_('Error !'), _('You can not desactivate an account that contains some journal items.'))
842+ elif method == 'unlink':
843+ raise osv.except_osv(_('Error !'), _('You can not remove an account containing journal items.'))
844+ #Checking whether the account is set as a property to any Partner or not
845+ value = 'account.account,' + str(ids[0])
846+ partner_prop_acc = self.pool.get('ir.property').search(cr, uid, [('value_reference','=',value)], context=context)
847+ if partner_prop_acc:
848+ raise osv.except_osv(_('Warning !'), _('You can not remove/desactivate an account which is set on a customer or supplier.'))
849+ return True
850+
851+ def _check_allow_type_change(self, cr, uid, ids, new_type, context=None):
852+ group1 = ['payable', 'receivable', 'other']
853+ group2 = ['consolidation','view']
854+ line_obj = self.pool.get('account.move.line')
855+ for account in self.browse(cr, uid, ids, context=context):
856+ old_type = account.type
857+ account_ids = self.search(cr, uid, [('id', 'child_of', [account.id])])
858+ if line_obj.search(cr, uid, [('account_id', 'in', account_ids)]):
859+ #Check for 'Closed' type
860+ if old_type == 'closed' and new_type !='closed':
861+ raise osv.except_osv(_('Warning !'), _("You cannot change the type of account from 'Closed' to any other type which contains journal items!"))
862+ #Check for change From group1 to group2 and vice versa
863+ if (old_type in group1 and new_type in group2) or (old_type in group2 and new_type in group1):
864+ raise osv.except_osv(_('Warning !'), _("You cannot change the type of account from '%s' to '%s' type as it contains journal items!") % (old_type,new_type,))
865+ return True
866+
867+ def write(self, cr, uid, ids, vals, context=None):
868+
869+ if context is None:
870+ context = {}
871+ if not ids:
872+ return True
873+ if isinstance(ids, (int, long)):
874+ ids = [ids]
875+
876+ # Dont allow changing the company_id when account_move_line already exist
877+ if 'company_id' in vals:
878+ move_lines = self.pool.get('account.move.line').search(cr, uid, [('account_id', 'in', ids)])
879+ if move_lines:
880+ # Allow the write if the value is the same
881+ for i in [i['company_id'][0] for i in self.read(cr,uid,ids,['company_id'])]:
882+ if vals['company_id']!=i:
883+ raise osv.except_osv(_('Warning !'), _('You cannot change the owner company of an account that already contains journal items.'))
884+ if 'active' in vals and not vals['active']:
885+ self._check_moves(cr, uid, ids, "write", context=context)
886+ if 'type' in vals.keys():
887+ self._check_allow_type_change(cr, uid, ids, vals['type'], context=context)
888+ return super(account_account, self).write(cr, uid, ids, vals, context=context)
889+
890+ def unlink(self, cr, uid, ids, context=None):
891+ self._check_moves(cr, uid, ids, "unlink", context=context)
892+ return super(account_account, self).unlink(cr, uid, ids, context=context)
893+
894+account_account()
895+
896+class account_journal_view(osv.osv):
897+ _name = "account.journal.view"
898+ _description = "Journal View"
899+ _columns = {
900+ 'name': fields.char('Journal View', size=64, required=True),
901+ 'columns_id': fields.one2many('account.journal.column', 'view_id', 'Columns')
902+ }
903+ _order = "name"
904+
905+account_journal_view()
906+
907+
908+class account_journal_column(osv.osv):
909+
910+ def _col_get(self, cr, user, context=None):
911+ result = []
912+ cols = self.pool.get('account.move.line')._columns
913+ for col in cols:
914+ if col in ('period_id', 'journal_id'):
915+ continue
916+ result.append( (col, cols[col].string) )
917+ result.sort()
918+ return result
919+
920+ _name = "account.journal.column"
921+ _description = "Journal Column"
922+ _columns = {
923+ 'name': fields.char('Column Name', size=64, required=True),
924+ 'field': fields.selection(_col_get, 'Field Name', required=True, size=32),
925+ 'view_id': fields.many2one('account.journal.view', 'Journal View', select=True),
926+ 'sequence': fields.integer('Sequence', help="Gives the sequence order to journal column.", readonly=True),
927+ 'required': fields.boolean('Required'),
928+ 'readonly': fields.boolean('Readonly'),
929+ }
930+ _order = "view_id, sequence"
931+
932+account_journal_column()
933+
934+class account_journal(osv.osv):
935+ _name = "account.journal"
936+ _description = "Journal"
937+ _columns = {
938+ 'name': fields.char('Journal Name', size=64, required=True),
939+ 'code': fields.char('Code', size=5, required=True, help="The code will be displayed on reports."),
940+ 'type': fields.selection([('sale', 'Sale'),('sale_refund','Sale Refund'), ('purchase', 'Purchase'), ('purchase_refund','Purchase Refund'), ('cash', 'Cash'), ('bank', 'Bank and Cheques'), ('general', 'General'), ('situation', 'Opening/Closing Situation')], 'Type', size=32, required=True,
941+ help="Select 'Sale' for customer invoices journals."\
942+ " Select 'Purchase' for supplier invoices journals."\
943+ " Select 'Cash' or 'Bank' for journals that are used in customer or supplier payments."\
944+ " Select 'General' for miscellaneous operations journals."\
945+ " Select 'Opening/Closing Situation' for entries generated for new fiscal years."),
946+ 'type_control_ids': fields.many2many('account.account.type', 'account_journal_type_rel', 'journal_id','type_id', 'Type Controls', domain=[('code','<>','view'), ('code', '<>', 'closed')]),
947+ 'account_control_ids': fields.many2many('account.account', 'account_account_type_rel', 'journal_id','account_id', 'Account', domain=[('type','<>','view'), ('type', '<>', 'closed')]),
948+ 'view_id': fields.many2one('account.journal.view', 'Display Mode', required=True, help="Gives the view used when writing or browsing entries in this journal. The view tells OpenERP which fields should be visible, required or readonly and in which order. You can create your own view for a faster encoding in each journal."),
949+ 'default_credit_account_id': fields.many2one('account.account', 'Default Credit Account', domain="[('type','!=','view')]", help="It acts as a default account for credit amount"),
950+ 'default_debit_account_id': fields.many2one('account.account', 'Default Debit Account', domain="[('type','!=','view')]", help="It acts as a default account for debit amount"),
951+ 'centralisation': fields.boolean('Centralised counterpart', help="Check this box to determine that each entry of this journal won't create a new counterpart but will share the same counterpart. This is used in fiscal year closing."),
952+ 'update_posted': fields.boolean('Allow Cancelling Entries', help="Check this box if you want to allow the cancellation the entries related to this journal or of the invoice related to this journal"),
953+ 'group_invoice_lines': fields.boolean('Group Invoice Lines', help="If this box is checked, the system will try to group the accounting lines when generating them from invoices."),
954+ 'sequence_id': fields.many2one('ir.sequence', 'Entry Sequence', help="This field contains the informatin related to the numbering of the journal entries of this journal.", required=True),
955+ 'user_id': fields.many2one('res.users', 'User', help="The user responsible for this journal"),
956+ 'groups_id': fields.many2many('res.groups', 'account_journal_group_rel', 'journal_id', 'group_id', 'Groups'),
957+ 'currency': fields.many2one('res.currency', 'Currency', help='The currency used to enter statement'),
958+ 'entry_posted': fields.boolean('Skip \'Draft\' State for Manual Entries', help='Check this box if you don\'t want new journal entries to pass through the \'draft\' state and instead goes directly to the \'posted state\' without any manual validation. \nNote that journal entries that are automatically created by the system are always skipping that state.'),
959+ 'company_id': fields.many2one('res.company', 'Company', required=True, select=1, help="Company related to this journal"),
960+ 'allow_date':fields.boolean('Check Date in Period', help= 'If set to True then do not accept the entry if the entry date is not into the period dates'),
961+ }
962+
963+ _defaults = {
964+ 'user_id': lambda self, cr, uid, context: uid,
965+ 'company_id': lambda self, cr, uid, c: self.pool.get('res.users').browse(cr, uid, uid, c).company_id.id,
966+ }
967+ _sql_constraints = [
968+ ('code_company_uniq', 'unique (code, company_id)', 'The code of the journal must be unique per company !'),
969+ ('name_company_uniq', 'unique (name, company_id)', 'The name of the journal must be unique per company !'),
970+ ]
971+
972+ _order = 'code'
973+
974+ def _check_currency(self, cr, uid, ids, context=None):
975+ for journal in self.browse(cr, uid, ids, context=context):
976+ if journal.currency:
977+ if journal.default_credit_account_id and not journal.default_credit_account_id.currency_id.id == journal.currency.id:
978+ return False
979+ if journal.default_debit_account_id and not journal.default_debit_account_id.currency_id.id == journal.currency.id:
980+ return False
981+ return True
982+
983+ _constraints = [
984+ (_check_currency, 'Configuration error! The currency chosen should be shared by the default accounts too.', ['currency','default_debit_account_id','default_credit_account_id']),
985+ ]
986+
987+ def copy(self, cr, uid, id, default={}, context=None, done_list=[], local=False):
988+ journal = self.browse(cr, uid, id, context=context)
989+ if not default:
990+ default = {}
991+ default = default.copy()
992+ default['code'] = (journal['code'] or '') + '(copy)'
993+ default['name'] = (journal['name'] or '') + '(copy)'
994+ default['sequence_id'] = False
995+ return super(account_journal, self).copy(cr, uid, id, default, context=context)
996+
997+ def write(self, cr, uid, ids, vals, context=None):
998+ if context is None:
999+ context = {}
1000+ if isinstance(ids, (int, long)):
1001+ ids = [ids]
1002+ for journal in self.browse(cr, uid, ids, context=context):
1003+ if 'company_id' in vals and journal.company_id.id != vals['company_id']:
1004+ move_lines = self.pool.get('account.move.line').search(cr, uid, [('journal_id', 'in', ids)])
1005+ if move_lines:
1006+ raise osv.except_osv(_('Warning !'), _('You can not modify the company of this journal as its related record exist in journal items'))
1007+ return super(account_journal, self).write(cr, uid, ids, vals, context=context)
1008+
1009+ def create_sequence(self, cr, uid, vals, context=None):
1010+ """ Create new no_gap entry sequence for every new Joural
1011+ """
1012+ # in account.journal code is actually the prefix of the sequence
1013+ # whereas ir.sequence code is a key to lookup global sequences.
1014+ prefix = vals['code'].upper()
1015+
1016+ seq = {
1017+ 'name': vals['name'],
1018+ 'implementation':'no_gap',
1019+ 'prefix': prefix + "/%(year)s/",
1020+ 'padding': 4,
1021+ 'number_increment': 1
1022+ }
1023+ if 'company_id' in vals:
1024+ seq['company_id'] = vals['company_id']
1025+ return self.pool.get('ir.sequence').create(cr, uid, seq)
1026+
1027+ def create(self, cr, uid, vals, context=None):
1028+ if not 'sequence_id' in vals or not vals['sequence_id']:
1029+ # if we have the right to create a journal, we should be able to
1030+ # create it's sequence.
1031+ vals.update({'sequence_id': self.create_sequence(cr, 1, vals, context)})
1032+ return super(account_journal, self).create(cr, uid, vals, context)
1033+
1034+ def name_get(self, cr, user, ids, context=None):
1035+ """
1036+ Returns a list of tupples containing id, name.
1037+ result format: {[(id, name), (id, name), ...]}
1038+
1039+ @param cr: A database cursor
1040+ @param user: ID of the user currently logged in
1041+ @param ids: list of ids for which name should be read
1042+ @param context: context arguments, like lang, time zone
1043+
1044+ @return: Returns a list of tupples containing id, name
1045+ """
1046+ result = self.browse(cr, user, ids, context=context)
1047+ res = []
1048+ for rs in result:
1049+ if rs.currency:
1050+ currency = rs.currency
1051+ else:
1052+ currency = rs.company_id.currency_id
1053+ name = "%s (%s)" % (rs.name, currency.name)
1054+ res += [(rs.id, name)]
1055+ return res
1056+
1057+ def name_search(self, cr, user, name, args=None, operator='ilike', context=None, limit=100):
1058+ if not args:
1059+ args = []
1060+ if context is None:
1061+ context = {}
1062+ ids = []
1063+ if context.get('journal_type', False):
1064+ args += [('type','=',context.get('journal_type'))]
1065+ if name:
1066+ ids = self.search(cr, user, [('code', 'ilike', name)]+ args, limit=limit, context=context)
1067+ if not ids:
1068+ ids = self.search(cr, user, [('name', 'ilike', name)]+ args, limit=limit, context=context)#fix it ilike should be replace with operator
1069+
1070+ return self.name_get(cr, user, ids, context=context)
1071+
1072+ def onchange_type(self, cr, uid, ids, type, currency, context=None):
1073+ obj_data = self.pool.get('ir.model.data')
1074+ user_pool = self.pool.get('res.users')
1075+
1076+ type_map = {
1077+ 'sale':'account_sp_journal_view',
1078+ 'sale_refund':'account_sp_refund_journal_view',
1079+ 'purchase':'account_sp_journal_view',
1080+ 'purchase_refund':'account_sp_refund_journal_view',
1081+ 'cash':'account_journal_bank_view',
1082+ 'bank':'account_journal_bank_view',
1083+ 'general':'account_journal_view',
1084+ 'situation':'account_journal_view'
1085+ }
1086+
1087+ res = {}
1088+ view_id = type_map.get(type, 'account_journal_view')
1089+ user = user_pool.browse(cr, uid, uid)
1090+ if type in ('cash', 'bank') and currency and user.company_id.currency_id.id != currency:
1091+ view_id = 'account_journal_bank_view_multi'
1092+ data_id = obj_data.search(cr, uid, [('model','=','account.journal.view'), ('name','=',view_id)])
1093+ data = obj_data.browse(cr, uid, data_id[0], context=context)
1094+
1095+ res.update({
1096+ 'centralisation':type == 'situation',
1097+ 'view_id':data.res_id,
1098+ })
1099+ return {
1100+ 'value':res
1101+ }
1102+
1103+account_journal()
1104+
1105+class account_fiscalyear(osv.osv):
1106+ _name = "account.fiscalyear"
1107+ _description = "Fiscal Year"
1108+ _columns = {
1109+ 'name': fields.char('Fiscal Year', size=64, required=True),
1110+ 'code': fields.char('Code', size=6, required=True),
1111+ 'company_id': fields.many2one('res.company', 'Company', required=True),
1112+ 'date_start': fields.date('Start Date', required=True),
1113+ 'date_stop': fields.date('End Date', required=True),
1114+ 'period_ids': fields.one2many('account.period', 'fiscalyear_id', 'Periods'),
1115+ 'state': fields.selection([('draft','Open'), ('done','Closed')], 'State', readonly=True),
1116+ }
1117+ _defaults = {
1118+ 'state': 'draft',
1119+ 'company_id': lambda self,cr,uid,c: self.pool.get('res.users').browse(cr, uid, uid, c).company_id.id,
1120+ }
1121+ _order = "date_start, id"
1122+
1123+
1124+ def _check_duration(self, cr, uid, ids, context=None):
1125+ obj_fy = self.browse(cr, uid, ids[0], context=context)
1126+ if obj_fy.date_stop < obj_fy.date_start:
1127+ return False
1128+ return True
1129+
1130+ _constraints = [
1131+ (_check_duration, 'Error! The start date of the fiscal year must be before his end date.', ['date_start','date_stop'])
1132+ ]
1133+
1134+ def create_period3(self, cr, uid, ids, context=None):
1135+ return self.create_period(cr, uid, ids, context, 3)
1136+
1137+ def create_period(self, cr, uid, ids, context=None, interval=1):
1138+ period_obj = self.pool.get('account.period')
1139+ for fy in self.browse(cr, uid, ids, context=context):
1140+ ds = datetime.strptime(fy.date_start, '%Y-%m-%d')
1141+ period_obj.create(cr, uid, {
1142+ 'name': "%s %s" % (_('Opening Period'), ds.strftime('%Y')),
1143+ 'code': ds.strftime('00/%Y'),
1144+ 'date_start': ds,
1145+ 'date_stop': ds,
1146+ 'special': True,
1147+ 'fiscalyear_id': fy.id,
1148+ })
1149+ while ds.strftime('%Y-%m-%d') < fy.date_stop:
1150+ de = ds + relativedelta(months=interval, days=-1)
1151+
1152+ if de.strftime('%Y-%m-%d') > fy.date_stop:
1153+ de = datetime.strptime(fy.date_stop, '%Y-%m-%d')
1154+
1155+ period_obj.create(cr, uid, {
1156+ 'name': ds.strftime('%m/%Y'),
1157+ 'code': ds.strftime('%m/%Y'),
1158+ 'date_start': ds.strftime('%Y-%m-%d'),
1159+ 'date_stop': de.strftime('%Y-%m-%d'),
1160+ 'fiscalyear_id': fy.id,
1161+ })
1162+ ds = ds + relativedelta(months=interval)
1163+ return True
1164+
1165+ def find(self, cr, uid, dt=None, exception=True, context=None):
1166+ res = self.finds(cr, uid, dt, exception, context=context)
1167+ return res and res[0] or False
1168+
1169+ def finds(self, cr, uid, dt=None, exception=True, context=None):
1170+ if context is None: context = {}
1171+ if not dt:
1172+ dt = fields.date.context_today(self,cr,uid,context=context)
1173+ args = [('date_start', '<=' ,dt), ('date_stop', '>=', dt)]
1174+ if context.get('company_id', False):
1175+ company_id = context['company_id']
1176+ else:
1177+ company_id = self.pool.get('res.users').browse(cr, uid, uid, context=context).company_id.id
1178+ args.append(('company_id', '=', company_id))
1179+ ids = self.search(cr, uid, args, context=context)
1180+ if not ids:
1181+ if exception:
1182+ raise osv.except_osv(_('Error !'), _('No fiscal year defined for this date !\nPlease create one from the configuration of the accounting menu.'))
1183+ else:
1184+ return []
1185+ return ids
1186+
1187+ def name_search(self, cr, user, name, args=None, operator='ilike', context=None, limit=80):
1188+ if args is None:
1189+ args = []
1190+ if context is None:
1191+ context = {}
1192+ ids = []
1193+ if name:
1194+ ids = self.search(cr, user, [('code', 'ilike', name)]+ args, limit=limit)
1195+ if not ids:
1196+ ids = self.search(cr, user, [('name', operator, name)]+ args, limit=limit)
1197+ return self.name_get(cr, user, ids, context=context)
1198+
1199+account_fiscalyear()
1200+
1201+class account_period(osv.osv):
1202+ _name = "account.period"
1203+ _description = "Account period"
1204+ _columns = {
1205+ 'name': fields.char('Period Name', size=64, required=True),
1206+ 'code': fields.char('Code', size=12),
1207+ 'special': fields.boolean('Opening/Closing Period', size=12,
1208+ help="These periods can overlap."),
1209+ 'date_start': fields.date('Start of Period', required=True, states={'done':[('readonly',True)]}),
1210+ 'date_stop': fields.date('End of Period', required=True, states={'done':[('readonly',True)]}),
1211+ 'fiscalyear_id': fields.many2one('account.fiscalyear', 'Fiscal Year', required=True, states={'done':[('readonly',True)]}, select=True),
1212+ 'state': fields.selection([('draft','Open'), ('done','Closed')], 'State', readonly=True,
1213+ help='When monthly periods are created. The state is \'Draft\'. At the end of monthly period it is in \'Done\' state.'),
1214+ 'company_id': fields.related('fiscalyear_id', 'company_id', type='many2one', relation='res.company', string='Company', store=True, readonly=True)
1215+ }
1216+ _defaults = {
1217+ 'state': 'draft',
1218+ }
1219+ _order = "date_start, special desc"
1220+ _sql_constraints = [
1221+ ('name_company_uniq', 'unique(name, company_id)', 'The name of the period must be unique per company!'),
1222+ ]
1223+
1224+ def _check_duration(self,cr,uid,ids,context=None):
1225+ obj_period = self.browse(cr, uid, ids[0], context=context)
1226+ if obj_period.date_stop < obj_period.date_start:
1227+ return False
1228+ return True
1229+
1230+ def _check_year_limit(self,cr,uid,ids,context=None):
1231+ for obj_period in self.browse(cr, uid, ids, context=context):
1232+ if obj_period.special:
1233+ continue
1234+
1235+ if obj_period.fiscalyear_id.date_stop < obj_period.date_stop or \
1236+ obj_period.fiscalyear_id.date_stop < obj_period.date_start or \
1237+ obj_period.fiscalyear_id.date_start > obj_period.date_start or \
1238+ obj_period.fiscalyear_id.date_start > obj_period.date_stop:
1239+ return False
1240+
1241+ pids = self.search(cr, uid, [('date_stop','>=',obj_period.date_start),('date_start','<=',obj_period.date_stop),('special','=',False),('id','<>',obj_period.id)])
1242+ for period in self.browse(cr, uid, pids):
1243+ if period.fiscalyear_id.company_id.id==obj_period.fiscalyear_id.company_id.id:
1244+ return False
1245+ return True
1246+
1247+ _constraints = [
1248+ (_check_duration, 'Error ! The duration of the Period(s) is/are invalid. ', ['date_stop']),
1249+ (_check_year_limit, 'Invalid period ! Some periods overlap or the date period is not in the scope of the fiscal year. ', ['date_stop'])
1250+ ]
1251+
1252+ def next(self, cr, uid, period, step, context=None):
1253+ ids = self.search(cr, uid, [('date_start','>',period.date_start)])
1254+ if len(ids)>=step:
1255+ return ids[step-1]
1256+ return False
1257+
1258+ def find(self, cr, uid, dt=None, context=None):
1259+ if context is None: context = {}
1260+ if not dt:
1261+ dt = fields.date.context_today(self,cr,uid,context=context)
1262+#CHECKME: shouldn't we check the state of the period?
1263+ args = [('date_start', '<=' ,dt), ('date_stop', '>=', dt)]
1264+ if context.get('company_id', False):
1265+ args.append(('company_id', '=', context['company_id']))
1266+ else:
1267+ company_id = self.pool.get('res.users').browse(cr, uid, uid, context=context).company_id.id
1268+ args.append(('company_id', '=', company_id))
1269+ result = []
1270+ if context.get('account_period_prefer_normal'):
1271+ # look for non-special periods first, and fallback to all if no result is found
1272+ result = self.search(cr, uid, args + [('special', '=', False)], context=context)
1273+ if not result:
1274+ result = self.search(cr, uid, args, context=context)
1275+ if not result:
1276+ raise osv.except_osv(_('Error !'), _('No period defined for this date: %s !\nPlease create one.')%dt)
1277+ return result
1278+
1279+ def action_draft(self, cr, uid, ids, *args):
1280+ mode = 'draft'
1281+ cr.execute('update account_journal_period set state=%s where period_id in %s', (mode, tuple(ids),))
1282+ cr.execute('update account_period set state=%s where id in %s', (mode, tuple(ids),))
1283+ return True
1284+
1285+ def name_search(self, cr, user, name, args=None, operator='ilike', context=None, limit=100):
1286+ if args is None:
1287+ args = []
1288+ if context is None:
1289+ context = {}
1290+ ids = []
1291+ if name:
1292+ ids = self.search(cr, user, [('code','ilike',name)]+ args, limit=limit)
1293+ if not ids:
1294+ ids = self.search(cr, user, [('name',operator,name)]+ args, limit=limit)
1295+ return self.name_get(cr, user, ids, context=context)
1296+
1297+ def write(self, cr, uid, ids, vals, context=None):
1298+ if 'company_id' in vals:
1299+ move_lines = self.pool.get('account.move.line').search(cr, uid, [('period_id', 'in', ids)])
1300+ if move_lines:
1301+ raise osv.except_osv(_('Warning !'), _('You can not modify company of this period as some journal items exists.'))
1302+ return super(account_period, self).write(cr, uid, ids, vals, context=context)
1303+
1304+ def build_ctx_periods(self, cr, uid, period_from_id, period_to_id):
1305+ if period_from_id == period_to_id:
1306+ return [period_from_id]
1307+ period_from = self.browse(cr, uid, period_from_id)
1308+ period_date_start = period_from.date_start
1309+ company1_id = period_from.company_id.id
1310+ period_to = self.browse(cr, uid, period_to_id)
1311+ period_date_stop = period_to.date_stop
1312+ company2_id = period_to.company_id.id
1313+ if company1_id != company2_id:
1314+ raise osv.except_osv(_('Error'), _('You should have chosen periods that belongs to the same company'))
1315+ if period_date_start > period_date_stop:
1316+ raise osv.except_osv(_('Error'), _('Start period should be smaller then End period'))
1317+ #for period from = january, we want to exclude the opening period (but it has same date_from, so we have to check if period_from is special or not to include that clause or not in the search).
1318+ if period_from.special:
1319+ return self.search(cr, uid, [('date_start', '>=', period_date_start), ('date_stop', '<=', period_date_stop), ('company_id', '=', company1_id)])
1320+ return self.search(cr, uid, [('date_start', '>=', period_date_start), ('date_stop', '<=', period_date_stop), ('company_id', '=', company1_id), ('special', '=', False)])
1321+
1322+account_period()
1323+
1324+class account_journal_period(osv.osv):
1325+ _name = "account.journal.period"
1326+ _description = "Journal Period"
1327+
1328+ def _icon_get(self, cr, uid, ids, field_name, arg=None, context=None):
1329+ result = {}.fromkeys(ids, 'STOCK_NEW')
1330+ for r in self.read(cr, uid, ids, ['state']):
1331+ result[r['id']] = {
1332+ 'draft': 'STOCK_NEW',
1333+ 'printed': 'STOCK_PRINT_PREVIEW',
1334+ 'done': 'STOCK_DIALOG_AUTHENTICATION',
1335+ }.get(r['state'], 'STOCK_NEW')
1336+ return result
1337+
1338+ _columns = {
1339+ 'name': fields.char('Journal-Period Name', size=64, required=True),
1340+ 'journal_id': fields.many2one('account.journal', 'Journal', required=True, ondelete="cascade"),
1341+ 'period_id': fields.many2one('account.period', 'Period', required=True, ondelete="cascade"),
1342+ 'icon': fields.function(_icon_get, string='Icon', type='char', size=32),
1343+ 'active': fields.boolean('Active', required=True, help="If the active field is set to False, it will allow you to hide the journal period without removing it."),
1344+ 'state': fields.selection([('draft','Draft'), ('printed','Printed'), ('done','Done')], 'State', required=True, readonly=True,
1345+ help='When journal period is created. The state is \'Draft\'. If a report is printed it comes to \'Printed\' state. When all transactions are done, it comes in \'Done\' state.'),
1346+ 'fiscalyear_id': fields.related('period_id', 'fiscalyear_id', string='Fiscal Year', type='many2one', relation='account.fiscalyear'),
1347+ 'company_id': fields.related('journal_id', 'company_id', type='many2one', relation='res.company', string='Company', store=True, readonly=True)
1348+ }
1349+
1350+ def _check(self, cr, uid, ids, context=None):
1351+ for obj in self.browse(cr, uid, ids, context=context):
1352+ cr.execute('select * from account_move_line where journal_id=%s and period_id=%s limit 1', (obj.journal_id.id, obj.period_id.id))
1353+ res = cr.fetchall()
1354+ if res:
1355+ raise osv.except_osv(_('Error !'), _('You can not modify/delete a journal with entries for this period !'))
1356+ return True
1357+
1358+ def write(self, cr, uid, ids, vals, context=None):
1359+ self._check(cr, uid, ids, context=context)
1360+ return super(account_journal_period, self).write(cr, uid, ids, vals, context=context)
1361+
1362+ def create(self, cr, uid, vals, context=None):
1363+ period_id = vals.get('period_id',False)
1364+ if period_id:
1365+ period = self.pool.get('account.period').browse(cr, uid, period_id, context=context)
1366+ vals['state']=period.state
1367+ return super(account_journal_period, self).create(cr, uid, vals, context)
1368+
1369+ def unlink(self, cr, uid, ids, context=None):
1370+ self._check(cr, uid, ids, context=context)
1371+ return super(account_journal_period, self).unlink(cr, uid, ids, context=context)
1372+
1373+ _defaults = {
1374+ 'state': 'draft',
1375+ 'active': True,
1376+ }
1377+ _order = "period_id"
1378+
1379+account_journal_period()
1380+
1381+class account_fiscalyear(osv.osv):
1382+ _inherit = "account.fiscalyear"
1383+ _description = "Fiscal Year"
1384+ _columns = {
1385+ 'end_journal_period_id':fields.many2one('account.journal.period','End of Year Entries Journal', readonly=True),
1386+ }
1387+
1388+ def copy(self, cr, uid, id, default={}, context=None):
1389+ default.update({
1390+ 'period_ids': [],
1391+ 'end_journal_period_id': False
1392+ })
1393+ return super(account_fiscalyear, self).copy(cr, uid, id, default=default, context=context)
1394+
1395+account_fiscalyear()
1396+#----------------------------------------------------------
1397+# Entries
1398+#----------------------------------------------------------
1399+class account_move(osv.osv):
1400+ _name = "account.move"
1401+ _description = "Account Entry"
1402+ _order = 'id desc'
1403+
1404+ def name_search(self, cr, user, name, args=None, operator='ilike', context=None, limit=80):
1405+ """
1406+ Returns a list of tupples containing id, name, as internally it is called {def name_get}
1407+ result format: {[(id, name), (id, name), ...]}
1408+
1409+ @param cr: A database cursor
1410+ @param user: ID of the user currently logged in
1411+ @param name: name to search
1412+ @param args: other arguments
1413+ @param operator: default operator is 'ilike', it can be changed
1414+ @param context: context arguments, like lang, time zone
1415+ @param limit: Returns first 'n' ids of complete result, default is 80.
1416+
1417+ @return: Returns a list of tuples containing id and name
1418+ """
1419+
1420+ if not args:
1421+ args = []
1422+ ids = []
1423+ if name:
1424+ ids += self.search(cr, user, [('name','ilike',name)]+args, limit=limit, context=context)
1425+
1426+ if not ids and name and type(name) == int:
1427+ ids += self.search(cr, user, [('id','=',name)]+args, limit=limit, context=context)
1428+
1429+ if not ids:
1430+ ids += self.search(cr, user, args, limit=limit, context=context)
1431+
1432+ return self.name_get(cr, user, ids, context=context)
1433+
1434+ def name_get(self, cursor, user, ids, context=None):
1435+ if isinstance(ids, (int, long)):
1436+ ids = [ids]
1437+ if not ids:
1438+ return []
1439+ res = []
1440+ data_move = self.pool.get('account.move').browse(cursor, user, ids, context=context)
1441+ for move in data_move:
1442+ if move.state=='draft':
1443+ name = '*' + str(move.id)
1444+ else:
1445+ name = move.name
1446+ res.append((move.id, name))
1447+ return res
1448+
1449+ def _get_period(self, cr, uid, context=None):
1450+ ctx = dict(context or {}, account_period_prefer_normal=True)
1451+ period_ids = self.pool.get('account.period').find(cr, uid, context=ctx)
1452+ return period_ids[0]
1453+
1454+ def _amount_compute(self, cr, uid, ids, name, args, context, where =''):
1455+ if not ids: return {}
1456+ cr.execute( 'SELECT move_id, SUM(debit) '\
1457+ 'FROM account_move_line '\
1458+ 'WHERE move_id IN %s '\
1459+ 'GROUP BY move_id', (tuple(ids),))
1460+ result = dict(cr.fetchall())
1461+ for id in ids:
1462+ result.setdefault(id, 0.0)
1463+ return result
1464+
1465+ def _search_amount(self, cr, uid, obj, name, args, context):
1466+ ids = set()
1467+ for cond in args:
1468+ amount = cond[2]
1469+ if isinstance(cond[2],(list,tuple)):
1470+ if cond[1] in ['in','not in']:
1471+ amount = tuple(cond[2])
1472+ else:
1473+ continue
1474+ else:
1475+ if cond[1] in ['=like', 'like', 'not like', 'ilike', 'not ilike', 'in', 'not in', 'child_of']:
1476+ continue
1477+
1478+ cr.execute("select move_id from account_move_line group by move_id having sum(debit) %s %%s" % (cond[1]),(amount,))
1479+ res_ids = set(id[0] for id in cr.fetchall())
1480+ ids = ids and (ids & res_ids) or res_ids
1481+ if ids:
1482+ return [('id', 'in', tuple(ids))]
1483+ return [('id', '=', '0')]
1484+
1485+ _columns = {
1486+ 'name': fields.char('Number', size=64, required=True),
1487+ 'ref': fields.char('Reference', size=64),
1488+ 'period_id': fields.many2one('account.period', 'Period', required=True, states={'posted':[('readonly',True)]}),
1489+ 'journal_id': fields.many2one('account.journal', 'Journal', required=True, states={'posted':[('readonly',True)]}),
1490+ 'state': fields.selection([('draft','Unposted'), ('posted','Posted')], 'State', required=True, readonly=True,
1491+ help='All manually created new journal entries are usually in the state \'Unposted\', but you can set the option to skip that state on the related journal. In that case, they will be behave as journal entries automatically created by the system on document validation (invoices, bank statements...) and will be created in \'Posted\' state.'),
1492+ 'line_id': fields.one2many('account.move.line', 'move_id', 'Entries', states={'posted':[('readonly',True)]}),
1493+ 'to_check': fields.boolean('To Review', help='Check this box if you are unsure of that journal entry and if you want to note it as \'to be reviewed\' by an accounting expert.'),
1494+ 'partner_id': fields.related('line_id', 'partner_id', type="many2one", relation="res.partner", string="Partner", store=True),
1495+ 'amount': fields.function(_amount_compute, string='Amount', digits_compute=dp.get_precision('Account'), type='float', fnct_search=_search_amount),
1496+ 'date': fields.date('Date', required=True, states={'posted':[('readonly',True)]}, select=True),
1497+ 'narration':fields.text('Internal Note'),
1498+ 'company_id': fields.related('journal_id','company_id',type='many2one',relation='res.company',string='Company', store=True, readonly=True),
1499+ 'balance': fields.float('balance', digits_compute=dp.get_precision('Account'), help="This is a field only used for internal purpose and shouldn't be displayed"),
1500+ }
1501+
1502+ _defaults = {
1503+ 'name': '/',
1504+ 'state': 'draft',
1505+ 'period_id': _get_period,
1506+ 'date': fields.date.context_today,
1507+ 'company_id': lambda self, cr, uid, c: self.pool.get('res.users').browse(cr, uid, uid, c).company_id.id,
1508+ }
1509+
1510+ def _check_centralisation(self, cursor, user, ids, context=None):
1511+ for move in self.browse(cursor, user, ids, context=context):
1512+ if move.journal_id.centralisation:
1513+ move_ids = self.search(cursor, user, [
1514+ ('period_id', '=', move.period_id.id),
1515+ ('journal_id', '=', move.journal_id.id),
1516+ ])
1517+ if len(move_ids) > 1:
1518+ return False
1519+ return True
1520+
1521+ _constraints = [
1522+ (_check_centralisation,
1523+ 'You can not create more than one move per period on centralized journal',
1524+ ['journal_id']),
1525+ ]
1526+
1527+ def post(self, cr, uid, ids, context=None):
1528+ if context is None:
1529+ context = {}
1530+ invoice = context.get('invoice', False)
1531+ valid_moves = self.validate(cr, uid, ids, context)
1532+
1533+ if not valid_moves:
1534+ raise osv.except_osv(_('Integrity Error !'), _('You can not validate a non-balanced entry !\nMake sure you have configured payment terms properly !\nThe latest payment term line should be of the type "Balance" !'))
1535+ obj_sequence = self.pool.get('ir.sequence')
1536+ for move in self.browse(cr, uid, valid_moves, context=context):
1537+ if move.name =='/':
1538+ new_name = False
1539+ journal = move.journal_id
1540+
1541+ if invoice and invoice.internal_number:
1542+ new_name = invoice.internal_number
1543+ else:
1544+ if journal.sequence_id:
1545+ c = {'fiscalyear_id': move.period_id.fiscalyear_id.id}
1546+ new_name = obj_sequence.next_by_id(cr, uid, journal.sequence_id.id, c)
1547+ else:
1548+ raise osv.except_osv(_('Error'), _('No sequence defined on the journal !'))
1549+
1550+ if new_name:
1551+ self.write(cr, uid, [move.id], {'name':new_name})
1552+
1553+ cr.execute('UPDATE account_move '\
1554+ 'SET state=%s '\
1555+ 'WHERE id IN %s',
1556+ ('posted', tuple(valid_moves),))
1557+ return True
1558+
1559+ def button_validate(self, cursor, user, ids, context=None):
1560+ for move in self.browse(cursor, user, ids, context=context):
1561+ # check that all accounts have the same topmost ancestor
1562+ top_common = None
1563+ for line in move.line_id:
1564+ account = line.account_id
1565+ top_account = account
1566+ while top_account.parent_id:
1567+ top_account = top_account.parent_id
1568+ if not top_common:
1569+ top_common = top_account
1570+ elif top_account.id != top_common.id:
1571+ raise osv.except_osv(_('Error !'),
1572+ _('You cannot validate this journal entry because account "%s" does not belong to chart of accounts "%s"!') % (account.name, top_common.name))
1573+ return self.post(cursor, user, ids, context=context)
1574+
1575+ def button_cancel(self, cr, uid, ids, context=None):
1576+ for line in self.browse(cr, uid, ids, context=context):
1577+ if line.period_id.state == 'done':
1578+ raise osv.except_osv(_('Error !'), _('You can not modify a posted entry of closed periods'))
1579+ elif not line.journal_id.update_posted:
1580+ raise osv.except_osv(_('Error !'), _('You can not modify a posted entry of this journal !\nYou should set the journal to allow cancelling entries if you want to do that.'))
1581+ if ids:
1582+ cr.execute('UPDATE account_move '\
1583+ 'SET state=%s '\
1584+ 'WHERE id IN %s', ('draft', tuple(ids),))
1585+ return True
1586+
1587+ def onchange_line_id(self, cr, uid, ids, line_ids, context=None):
1588+ balance = 0.0
1589+ for line in line_ids:
1590+ if line[2]:
1591+ balance += (line[2]['debit'] or 0.00)- (line[2]['credit'] or 0.00)
1592+ return {'value': {'balance': balance}}
1593+
1594+ def write(self, cr, uid, ids, vals, context=None):
1595+ if context is None:
1596+ context = {}
1597+ c = context.copy()
1598+ c['novalidate'] = True
1599+ result = super(account_move, self).write(cr, uid, ids, vals, c)
1600+ self.validate(cr, uid, ids, context=context)
1601+ return result
1602+
1603+ #
1604+ # TODO: Check if period is closed !
1605+ #
1606+ def create(self, cr, uid, vals, context=None):
1607+ if context is None:
1608+ context = {}
1609+ if 'line_id' in vals and context.get('copy'):
1610+ for l in vals['line_id']:
1611+ if not l[0]:
1612+ l[2].update({
1613+ 'reconcile_id':False,
1614+ 'reconcil_partial_id':False,
1615+ 'analytic_lines':False,
1616+ 'invoice':False,
1617+ 'ref':False,
1618+ 'balance':False,
1619+ 'account_tax_id':False,
1620+ })
1621+
1622+ if 'journal_id' in vals and vals.get('journal_id', False):
1623+ for l in vals['line_id']:
1624+ if not l[0]:
1625+ l[2]['journal_id'] = vals['journal_id']
1626+ context['journal_id'] = vals['journal_id']
1627+ if 'period_id' in vals:
1628+ for l in vals['line_id']:
1629+ if not l[0]:
1630+ l[2]['period_id'] = vals['period_id']
1631+ context['period_id'] = vals['period_id']
1632+ else:
1633+ default_period = self._get_period(cr, uid, context)
1634+ for l in vals['line_id']:
1635+ if not l[0]:
1636+ l[2]['period_id'] = default_period
1637+ context['period_id'] = default_period
1638+
1639+ if 'line_id' in vals:
1640+ c = context.copy()
1641+ c['novalidate'] = True
1642+ result = super(account_move, self).create(cr, uid, vals, c)
1643+ self.validate(cr, uid, [result], context)
1644+ else:
1645+ result = super(account_move, self).create(cr, uid, vals, context)
1646+ return result
1647+
1648+ def copy(self, cr, uid, id, default={}, context=None):
1649+ if context is None:
1650+ context = {}
1651+ default.update({
1652+ 'state':'draft',
1653+ 'name':'/',
1654+ })
1655+ context.update({
1656+ 'copy':True
1657+ })
1658+ return super(account_move, self).copy(cr, uid, id, default, context)
1659+
1660+ def unlink(self, cr, uid, ids, context=None, check=True):
1661+ if context is None:
1662+ context = {}
1663+ toremove = []
1664+ obj_move_line = self.pool.get('account.move.line')
1665+ for move in self.browse(cr, uid, ids, context=context):
1666+ if move['state'] != 'draft':
1667+ raise osv.except_osv(_('UserError'),
1668+ _('You can not delete a posted journal entry "%s"!') % \
1669+ move['name'])
1670+ line_ids = map(lambda x: x.id, move.line_id)
1671+ context['journal_id'] = move.journal_id.id
1672+ context['period_id'] = move.period_id.id
1673+ obj_move_line._update_check(cr, uid, line_ids, context)
1674+ obj_move_line.unlink(cr, uid, line_ids, context=context)
1675+ toremove.append(move.id)
1676+ result = super(account_move, self).unlink(cr, uid, toremove, context)
1677+ return result
1678+
1679+ def _compute_balance(self, cr, uid, id, context=None):
1680+ move = self.browse(cr, uid, id, context=context)
1681+ amount = 0
1682+ for line in move.line_id:
1683+ amount+= (line.debit - line.credit)
1684+ return amount
1685+
1686+ def _centralise(self, cr, uid, move, mode, context=None):
1687+ assert mode in ('debit', 'credit'), 'Invalid Mode' #to prevent sql injection
1688+ currency_obj = self.pool.get('res.currency')
1689+ if context is None:
1690+ context = {}
1691+
1692+ if mode=='credit':
1693+ account_id = move.journal_id.default_debit_account_id.id
1694+ mode2 = 'debit'
1695+ if not account_id:
1696+ raise osv.except_osv(_('UserError'),
1697+ _('There is no default default debit account defined \n' \
1698+ 'on journal "%s"') % move.journal_id.name)
1699+ else:
1700+ account_id = move.journal_id.default_credit_account_id.id
1701+ mode2 = 'credit'
1702+ if not account_id:
1703+ raise osv.except_osv(_('UserError'),
1704+ _('There is no default default credit account defined \n' \
1705+ 'on journal "%s"') % move.journal_id.name)
1706+
1707+ # find the first line of this move with the current mode
1708+ # or create it if it doesn't exist
1709+ cr.execute('select id from account_move_line where move_id=%s and centralisation=%s limit 1', (move.id, mode))
1710+ res = cr.fetchone()
1711+ if res:
1712+ line_id = res[0]
1713+ else:
1714+ context.update({'journal_id': move.journal_id.id, 'period_id': move.period_id.id})
1715+ line_id = self.pool.get('account.move.line').create(cr, uid, {
1716+ 'name': _(mode.capitalize()+' Centralisation'),
1717+ 'centralisation': mode,
1718+ 'account_id': account_id,
1719+ 'move_id': move.id,
1720+ 'journal_id': move.journal_id.id,
1721+ 'period_id': move.period_id.id,
1722+ 'date': move.period_id.date_stop,
1723+ 'debit': 0.0,
1724+ 'credit': 0.0,
1725+ }, context)
1726+
1727+ # find the first line of this move with the other mode
1728+ # so that we can exclude it from our calculation
1729+ cr.execute('select id from account_move_line where move_id=%s and centralisation=%s limit 1', (move.id, mode2))
1730+ res = cr.fetchone()
1731+ if res:
1732+ line_id2 = res[0]
1733+ else:
1734+ line_id2 = 0
1735+
1736+ cr.execute('SELECT SUM(%s) FROM account_move_line WHERE move_id=%%s AND id!=%%s' % (mode,), (move.id, line_id2))
1737+ result = cr.fetchone()[0] or 0.0
1738+ cr.execute('update account_move_line set '+mode2+'=%s where id=%s', (result, line_id))
1739+
1740+ #adjust also the amount in currency if needed
1741+ cr.execute("select currency_id, sum(amount_currency) as amount_currency from account_move_line where move_id = %s and currency_id is not null group by currency_id", (move.id,))
1742+ for row in cr.dictfetchall():
1743+ currency_id = currency_obj.browse(cr, uid, row['currency_id'], context=context)
1744+ if not currency_obj.is_zero(cr, uid, currency_id, row['amount_currency']):
1745+ amount_currency = row['amount_currency'] * -1
1746+ account_id = amount_currency > 0 and move.journal_id.default_debit_account_id.id or move.journal_id.default_credit_account_id.id
1747+ cr.execute('select id from account_move_line where move_id=%s and centralisation=\'currency\' and currency_id = %slimit 1', (move.id, row['currency_id']))
1748+ res = cr.fetchone()
1749+ if res:
1750+ cr.execute('update account_move_line set amount_currency=%s , account_id=%s where id=%s', (amount_currency, account_id, res[0]))
1751+ else:
1752+ context.update({'journal_id': move.journal_id.id, 'period_id': move.period_id.id})
1753+ line_id = self.pool.get('account.move.line').create(cr, uid, {
1754+ 'name': _('Currency Adjustment'),
1755+ 'centralisation': 'currency',
1756+ 'account_id': account_id,
1757+ 'move_id': move.id,
1758+ 'journal_id': move.journal_id.id,
1759+ 'period_id': move.period_id.id,
1760+ 'date': move.period_id.date_stop,
1761+ 'debit': 0.0,
1762+ 'credit': 0.0,
1763+ 'currency_id': row['currency_id'],
1764+ 'amount_currency': amount_currency,
1765+ }, context)
1766+
1767+ return True
1768+
1769+ #
1770+ # Validate a balanced move. If it is a centralised journal, create a move.
1771+ #
1772+ def validate(self, cr, uid, ids, context=None):
1773+ if context and ('__last_update' in context):
1774+ del context['__last_update']
1775+
1776+ valid_moves = [] #Maintains a list of moves which can be responsible to create analytic entries
1777+ obj_analytic_line = self.pool.get('account.analytic.line')
1778+ obj_move_line = self.pool.get('account.move.line')
1779+ for move in self.browse(cr, uid, ids, context):
1780+ # Unlink old analytic lines on move_lines
1781+ for obj_line in move.line_id:
1782+ for obj in obj_line.analytic_lines:
1783+ obj_analytic_line.unlink(cr,uid,obj.id)
1784+
1785+ journal = move.journal_id
1786+ amount = 0
1787+ line_ids = []
1788+ line_draft_ids = []
1789+ company_id = None
1790+ for line in move.line_id:
1791+ amount += line.debit - line.credit
1792+ line_ids.append(line.id)
1793+ if line.state=='draft':
1794+ line_draft_ids.append(line.id)
1795+
1796+ if not company_id:
1797+ company_id = line.account_id.company_id.id
1798+ if not company_id == line.account_id.company_id.id:
1799+ raise osv.except_osv(_('Error'), _("Couldn't create move between different companies"))
1800+
1801+ if line.account_id.currency_id and line.currency_id:
1802+ if line.account_id.currency_id.id != line.currency_id.id and (line.account_id.currency_id.id != line.account_id.company_id.currency_id.id):
1803+ raise osv.except_osv(_('Error'), _("""Couldn't create move with currency different from the secondary currency of the account "%s - %s". Clear the secondary currency field of the account definition if you want to accept all currencies.""") % (line.account_id.code, line.account_id.name))
1804+
1805+ if abs(amount) < 10 ** -4:
1806+ # If the move is balanced
1807+ # Add to the list of valid moves
1808+ # (analytic lines will be created later for valid moves)
1809+ valid_moves.append(move)
1810+
1811+ # Check whether the move lines are confirmed
1812+
1813+ if not line_draft_ids:
1814+ continue
1815+ # Update the move lines (set them as valid)
1816+
1817+ obj_move_line.write(cr, uid, line_draft_ids, {
1818+ 'state': 'valid'
1819+ }, context, check=False)
1820+
1821+ account = {}
1822+ account2 = {}
1823+
1824+ if journal.type in ('purchase','sale'):
1825+ for line in move.line_id:
1826+ code = amount = 0
1827+ key = (line.account_id.id, line.tax_code_id.id)
1828+ if key in account2:
1829+ code = account2[key][0]
1830+ amount = account2[key][1] * (line.debit + line.credit)
1831+ elif line.account_id.id in account:
1832+ code = account[line.account_id.id][0]
1833+ amount = account[line.account_id.id][1] * (line.debit + line.credit)
1834+ if (code or amount) and not (line.tax_code_id or line.tax_amount):
1835+ obj_move_line.write(cr, uid, [line.id], {
1836+ 'tax_code_id': code,
1837+ 'tax_amount': amount
1838+ }, context, check=False)
1839+ elif journal.centralisation:
1840+ # If the move is not balanced, it must be centralised...
1841+
1842+ # Add to the list of valid moves
1843+ # (analytic lines will be created later for valid moves)
1844+ valid_moves.append(move)
1845+
1846+ #
1847+ # Update the move lines (set them as valid)
1848+ #
1849+ self._centralise(cr, uid, move, 'debit', context=context)
1850+ self._centralise(cr, uid, move, 'credit', context=context)
1851+ obj_move_line.write(cr, uid, line_draft_ids, {
1852+ 'state': 'valid'
1853+ }, context, check=False)
1854+ else:
1855+ # We can't validate it (it's unbalanced)
1856+ # Setting the lines as draft
1857+ obj_move_line.write(cr, uid, line_ids, {
1858+ 'state': 'draft'
1859+ }, context, check=False)
1860+ # Create analytic lines for the valid moves
1861+ for record in valid_moves:
1862+ obj_move_line.create_analytic_lines(cr, uid, [line.id for line in record.line_id], context)
1863+
1864+ valid_moves = [move.id for move in valid_moves]
1865+ return len(valid_moves) > 0 and valid_moves or False
1866+
1867+account_move()
1868+
1869+class account_move_reconcile(osv.osv):
1870+ _name = "account.move.reconcile"
1871+ _description = "Account Reconciliation"
1872+ _columns = {
1873+ 'name': fields.char('Name', size=64, required=True),
1874+ 'type': fields.char('Type', size=16, required=True),
1875+ 'line_id': fields.one2many('account.move.line', 'reconcile_id', 'Entry Lines'),
1876+ 'line_partial_ids': fields.one2many('account.move.line', 'reconcile_partial_id', 'Partial Entry lines'),
1877+ 'create_date': fields.date('Creation date', readonly=True),
1878+ }
1879+ _defaults = {
1880+ 'name': lambda self,cr,uid,ctx=None: self.pool.get('ir.sequence').get(cr, uid, 'account.reconcile', context=ctx) or '/',
1881+ }
1882+
1883+ def reconcile_partial_check(self, cr, uid, ids, type='auto', context=None):
1884+ total = 0.0
1885+ for rec in self.browse(cr, uid, ids, context=context):
1886+ for line in rec.line_partial_ids:
1887+ if line.account_id.currency_id:
1888+ total += line.amount_currency
1889+ else:
1890+ total += (line.debit or 0.0) - (line.credit or 0.0)
1891+ if not total:
1892+ self.pool.get('account.move.line').write(cr, uid,
1893+ map(lambda x: x.id, rec.line_partial_ids),
1894+ {'reconcile_id': rec.id }
1895+ )
1896+ return True
1897+
1898+ def name_get(self, cr, uid, ids, context=None):
1899+ if not ids:
1900+ return []
1901+ result = []
1902+ for r in self.browse(cr, uid, ids, context=context):
1903+ total = reduce(lambda y,t: (t.debit or 0.0) - (t.credit or 0.0) + y, r.line_partial_ids, 0.0)
1904+ if total:
1905+ name = '%s (%.2f)' % (r.name, total)
1906+ result.append((r.id,name))
1907+ else:
1908+ result.append((r.id,r.name))
1909+ return result
1910+
1911+account_move_reconcile()
1912+
1913+#----------------------------------------------------------
1914+# Tax
1915+#----------------------------------------------------------
1916+"""
1917+a documenter
1918+child_depend: la taxe depend des taxes filles
1919+"""
1920+class account_tax_code(osv.osv):
1921+ """
1922+ A code for the tax object.
1923+
1924+ This code is used for some tax declarations.
1925+ """
1926+ def _sum(self, cr, uid, ids, name, args, context, where ='', where_params=()):
1927+ parent_ids = tuple(self.search(cr, uid, [('parent_id', 'child_of', ids)]))
1928+ if context.get('based_on', 'invoices') == 'payments':
1929+ cr.execute('SELECT line.tax_code_id, sum(line.tax_amount) \
1930+ FROM account_move_line AS line, \
1931+ account_move AS move \
1932+ LEFT JOIN account_invoice invoice ON \
1933+ (invoice.move_id = move.id) \
1934+ WHERE line.tax_code_id IN %s '+where+' \
1935+ AND move.id = line.move_id \
1936+ AND ((invoice.state = \'paid\') \
1937+ OR (invoice.id IS NULL)) \
1938+ GROUP BY line.tax_code_id',
1939+ (parent_ids,) + where_params)
1940+ else:
1941+ cr.execute('SELECT line.tax_code_id, sum(line.tax_amount) \
1942+ FROM account_move_line AS line, \
1943+ account_move AS move \
1944+ WHERE line.tax_code_id IN %s '+where+' \
1945+ AND move.id = line.move_id \
1946+ GROUP BY line.tax_code_id',
1947+ (parent_ids,) + where_params)
1948+ res=dict(cr.fetchall())
1949+ obj_precision = self.pool.get('decimal.precision')
1950+ res2 = {}
1951+ for record in self.browse(cr, uid, ids, context=context):
1952+ def _rec_get(record):
1953+ amount = res.get(record.id, 0.0)
1954+ for rec in record.child_ids:
1955+ amount += _rec_get(rec) * rec.sign
1956+ return amount
1957+ res2[record.id] = round(_rec_get(record), obj_precision.precision_get(cr, uid, 'Account'))
1958+ return res2
1959+
1960+ def _sum_year(self, cr, uid, ids, name, args, context=None):
1961+ if context is None:
1962+ context = {}
1963+ move_state = ('posted', )
1964+ if context.get('state', 'all') == 'all':
1965+ move_state = ('draft', 'posted', )
1966+ if context.get('fiscalyear_id', False):
1967+ fiscalyear_id = [context['fiscalyear_id']]
1968+ else:
1969+ fiscalyear_id = self.pool.get('account.fiscalyear').finds(cr, uid, exception=False)
1970+ where = ''
1971+ where_params = ()
1972+ if fiscalyear_id:
1973+ pids = []
1974+ for fy in fiscalyear_id:
1975+ pids += map(lambda x: str(x.id), self.pool.get('account.fiscalyear').browse(cr, uid, fy).period_ids)
1976+ if pids:
1977+ where = ' AND line.period_id IN %s AND move.state IN %s '
1978+ where_params = (tuple(pids), move_state)
1979+ return self._sum(cr, uid, ids, name, args, context,
1980+ where=where, where_params=where_params)
1981+
1982+ def _sum_period(self, cr, uid, ids, name, args, context):
1983+ if context is None:
1984+ context = {}
1985+ move_state = ('posted', )
1986+ if context.get('state', False) == 'all':
1987+ move_state = ('draft', 'posted', )
1988+ if context.get('period_id', False):
1989+ period_id = context['period_id']
1990+ else:
1991+ period_id = self.pool.get('account.period').find(cr, uid)
1992+ if not period_id:
1993+ return dict.fromkeys(ids, 0.0)
1994+ period_id = period_id[0]
1995+ return self._sum(cr, uid, ids, name, args, context,
1996+ where=' AND line.period_id=%s AND move.state IN %s', where_params=(period_id, move_state))
1997+
1998+ _name = 'account.tax.code'
1999+ _description = 'Tax Code'
2000+ _rec_name = 'code'
2001+ _columns = {
2002+ 'name': fields.char('Tax Case Name', size=64, required=True, translate=True),
2003+ 'code': fields.char('Case Code', size=64),
2004+ 'info': fields.text('Description'),
2005+ 'sum': fields.function(_sum_year, string="Year Sum"),
2006+ 'sum_period': fields.function(_sum_period, string="Period Sum"),
2007+ 'parent_id': fields.many2one('account.tax.code', 'Parent Code', select=True),
2008+ 'child_ids': fields.one2many('account.tax.code', 'parent_id', 'Child Codes'),
2009+ 'line_ids': fields.one2many('account.move.line', 'tax_code_id', 'Lines'),
2010+ 'company_id': fields.many2one('res.company', 'Company', required=True),
2011+ 'sign': fields.float('Coefficent for parent', required=True, help='You can specify here the coefficient that will be used when consolidating the amount of this case into its parent. For example, set 1/-1 if you want to add/substract it.'),
2012+ 'notprintable':fields.boolean("Not Printable in Invoice", help="Check this box if you don't want any VAT related to this Tax Code to appear on invoices"),
2013+ 'sequence': fields.integer('Sequence', help="Determine the display order in the report 'Accounting \ Reporting \ Generic Reporting \ Taxes \ Taxes Report'"),
2014+ }
2015+
2016+ def name_search(self, cr, user, name, args=None, operator='ilike', context=None, limit=80):
2017+ if not args:
2018+ args = []
2019+ if context is None:
2020+ context = {}
2021+ ids = self.search(cr, user, ['|',('name',operator,name),('code',operator,name)] + args, limit=limit, context=context)
2022+ return self.name_get(cr, user, ids, context)
2023+
2024+ def name_get(self, cr, uid, ids, context=None):
2025+ if isinstance(ids, (int, long)):
2026+ ids = [ids]
2027+ if not ids:
2028+ return []
2029+ if isinstance(ids, (int, long)):
2030+ ids = [ids]
2031+ reads = self.read(cr, uid, ids, ['name','code'], context, load='_classic_write')
2032+ return [(x['id'], (x['code'] and (x['code'] + ' - ') or '') + x['name']) \
2033+ for x in reads]
2034+
2035+ def _default_company(self, cr, uid, context=None):
2036+ user = self.pool.get('res.users').browse(cr, uid, uid, context=context)
2037+ if user.company_id:
2038+ return user.company_id.id
2039+ return self.pool.get('res.company').search(cr, uid, [('parent_id', '=', False)])[0]
2040+ _defaults = {
2041+ 'company_id': _default_company,
2042+ 'sign': 1.0,
2043+ 'notprintable': False,
2044+ }
2045+
2046+ def copy(self, cr, uid, id, default=None, context=None):
2047+ if default is None:
2048+ default = {}
2049+ default = default.copy()
2050+ default.update({'line_ids': []})
2051+ return super(account_tax_code, self).copy(cr, uid, id, default, context)
2052+
2053+ _check_recursion = check_cycle
2054+ _constraints = [
2055+ (_check_recursion, 'Error ! You can not create recursive accounts.', ['parent_id'])
2056+ ]
2057+ _order = 'code'
2058+
2059+account_tax_code()
2060+
2061+class account_tax(osv.osv):
2062+ """
2063+ A tax object.
2064+
2065+ Type: percent, fixed, none, code
2066+ PERCENT: tax = price * amount
2067+ FIXED: tax = price + amount
2068+ NONE: no tax line
2069+ CODE: execute python code. localcontext = {'price_unit':pu, 'address':address_object}
2070+ return result in the context
2071+ Ex: result=round(price_unit*0.21,4)
2072+ """
2073+
2074+ def get_precision_tax():
2075+ def change_digit_tax(cr):
2076+ res = pooler.get_pool(cr.dbname).get('decimal.precision').precision_get(cr, 1, 'Account')
2077+ return (16, res+2)
2078+ return change_digit_tax
2079+
2080+ _name = 'account.tax'
2081+ _description = 'Tax'
2082+ _columns = {
2083+ 'name': fields.char('Tax Name', size=64, required=True, translate=True, help="This name will be displayed on reports"),
2084+ 'sequence': fields.integer('Sequence', required=True, help="The sequence field is used to order the tax lines from the lowest sequences to the higher ones. The order is important if you have a tax with several tax children. In this case, the evaluation order is important."),
2085+ 'amount': fields.float('Amount', required=True, digits_compute=get_precision_tax(), help="For taxes of type percentage, enter % ratio between 0-1."),
2086+ 'active': fields.boolean('Active', help="If the active field is set to False, it will allow you to hide the tax without removing it."),
2087+ 'type': fields.selection( [('percent','Percentage'), ('fixed','Fixed Amount'), ('none','None'), ('code','Python Code'), ('balance','Balance')], 'Tax Type', required=True,
2088+ help="The computation method for the tax amount."),
2089+ 'applicable_type': fields.selection( [('true','Always'), ('code','Given by Python Code')], 'Applicability', required=True,
2090+ help="If not applicable (computed through a Python code), the tax won't appear on the invoice."),
2091+ 'domain':fields.char('Domain', size=32, help="This field is only used if you develop your own module allowing developers to create specific taxes in a custom domain."),
2092+ 'account_collected_id':fields.many2one('account.account', 'Invoice Tax Account'),
2093+ 'account_paid_id':fields.many2one('account.account', 'Refund Tax Account'),
2094+ 'parent_id':fields.many2one('account.tax', 'Parent Tax Account', select=True),
2095+ 'child_ids':fields.one2many('account.tax', 'parent_id', 'Child Tax Accounts'),
2096+ 'child_depend':fields.boolean('Tax on Children', help="Set if the tax computation is based on the computation of child taxes rather than on the total amount."),
2097+ 'python_compute':fields.text('Python Code'),
2098+ 'python_compute_inv':fields.text('Python Code (reverse)'),
2099+ 'python_applicable':fields.text('Python Code'),
2100+
2101+ #
2102+ # Fields used for the VAT declaration
2103+ #
2104+ 'base_code_id': fields.many2one('account.tax.code', 'Account Base Code', help="Use this code for the VAT declaration."),
2105+ 'tax_code_id': fields.many2one('account.tax.code', 'Account Tax Code', help="Use this code for the VAT declaration."),
2106+ 'base_sign': fields.float('Base Code Sign', help="Usually 1 or -1."),
2107+ 'tax_sign': fields.float('Tax Code Sign', help="Usually 1 or -1."),
2108+
2109+ # Same fields for refund invoices
2110+
2111+ 'ref_base_code_id': fields.many2one('account.tax.code', 'Refund Base Code', help="Use this code for the VAT declaration."),
2112+ 'ref_tax_code_id': fields.many2one('account.tax.code', 'Refund Tax Code', help="Use this code for the VAT declaration."),
2113+ 'ref_base_sign': fields.float('Base Code Sign', help="Usually 1 or -1."),
2114+ 'ref_tax_sign': fields.float('Tax Code Sign', help="Usually 1 or -1."),
2115+ 'include_base_amount': fields.boolean('Included in base amount', help="Indicates if the amount of tax must be included in the base amount for the computation of the next taxes"),
2116+ 'company_id': fields.many2one('res.company', 'Company', required=True),
2117+ 'description': fields.char('Tax Code',size=32),
2118+ 'price_include': fields.boolean('Tax Included in Price', help="Check this if the price you use on the product and invoices includes this tax."),
2119+ 'type_tax_use': fields.selection([('sale','Sale'),('purchase','Purchase'),('all','All')], 'Tax Application', required=True)
2120+
2121+ }
2122+ _sql_constraints = [
2123+ ('name_company_uniq', 'unique(name, company_id)', 'Tax Name must be unique per company!'),
2124+ ]
2125+
2126+ def name_search(self, cr, user, name, args=None, operator='ilike', context=None, limit=80):
2127+ """
2128+ Returns a list of tupples containing id, name, as internally it is called {def name_get}
2129+ result format: {[(id, name), (id, name), ...]}
2130+
2131+ @param cr: A database cursor
2132+ @param user: ID of the user currently logged in
2133+ @param name: name to search
2134+ @param args: other arguments
2135+ @param operator: default operator is 'ilike', it can be changed
2136+ @param context: context arguments, like lang, time zone
2137+ @param limit: Returns first 'n' ids of complete result, default is 80.
2138+
2139+ @return: Returns a list of tupples containing id and name
2140+ """
2141+ if not args:
2142+ args = []
2143+ if context is None:
2144+ context = {}
2145+ ids = []
2146+ if name:
2147+ ids = self.search(cr, user, [('description', '=', name)] + args, limit=limit, context=context)
2148+ if not ids:
2149+ ids = self.search(cr, user, [('name', operator, name)] + args, limit=limit, context=context)
2150+ else:
2151+ ids = self.search(cr, user, args, limit=limit, context=context or {})
2152+ return self.name_get(cr, user, ids, context=context)
2153+
2154+ def write(self, cr, uid, ids, vals, context=None):
2155+ if vals.get('type', False) and vals['type'] in ('none', 'code'):
2156+ vals.update({'amount': 0.0})
2157+ return super(account_tax, self).write(cr, uid, ids, vals, context=context)
2158+
2159+ def search(self, cr, uid, args, offset=0, limit=None, order=None, context=None, count=False):
2160+ journal_pool = self.pool.get('account.journal')
2161+
2162+ if context and context.has_key('type'):
2163+ if context.get('type') in ('out_invoice','out_refund'):
2164+ args += [('type_tax_use','in',['sale','all'])]
2165+ elif context.get('type') in ('in_invoice','in_refund'):
2166+ args += [('type_tax_use','in',['purchase','all'])]
2167+
2168+ if context and context.has_key('journal_id'):
2169+ journal = journal_pool.browse(cr, uid, context.get('journal_id'))
2170+ if journal.type in ('sale', 'purchase'):
2171+ args += [('type_tax_use','in',[journal.type,'all'])]
2172+
2173+ return super(account_tax, self).search(cr, uid, args, offset, limit, order, context, count)
2174+
2175+ def name_get(self, cr, uid, ids, context=None):
2176+ if not ids:
2177+ return []
2178+ res = []
2179+ for record in self.read(cr, uid, ids, ['description','name'], context=context):
2180+ name = record['description'] and record['description'] or record['name']
2181+ res.append((record['id'],name ))
2182+ return res
2183+
2184+ def _default_company(self, cr, uid, context=None):
2185+ user = self.pool.get('res.users').browse(cr, uid, uid, context=context)
2186+ if user.company_id:
2187+ return user.company_id.id
2188+ return self.pool.get('res.company').search(cr, uid, [('parent_id', '=', False)])[0]
2189+
2190+ _defaults = {
2191+ 'python_compute': '''# price_unit\n# address: res.partner.address object or False\n# product: product.product object or None\n# partner: res.partner object or None\n\nresult = price_unit * 0.10''',
2192+ 'python_compute_inv': '''# price_unit\n# address: res.partner.address object or False\n# product: product.product object or False\n\nresult = price_unit * 0.10''',
2193+ 'applicable_type': 'true',
2194+ 'type': 'percent',
2195+ 'amount': 0,
2196+ 'price_include': 0,
2197+ 'active': 1,
2198+ 'type_tax_use': 'all',
2199+ 'sequence': 1,
2200+ 'ref_tax_sign': 1,
2201+ 'ref_base_sign': 1,
2202+ 'tax_sign': 1,
2203+ 'base_sign': 1,
2204+ 'include_base_amount': False,
2205+ 'company_id': _default_company,
2206+ }
2207+ _order = 'sequence'
2208+
2209+ def _applicable(self, cr, uid, taxes, price_unit, address_id=None, product=None, partner=None):
2210+ res = []
2211+ obj_partener_address = self.pool.get('res.partner.address')
2212+ for tax in taxes:
2213+ if tax.applicable_type=='code':
2214+ localdict = {'price_unit':price_unit, 'address':obj_partener_address.browse(cr, uid, address_id), 'product':product, 'partner':partner}
2215+ exec tax.python_applicable in localdict
2216+ if localdict.get('result', False):
2217+ res.append(tax)
2218+ else:
2219+ res.append(tax)
2220+ return res
2221+
2222+ def _unit_compute(self, cr, uid, taxes, price_unit, address_id=None, product=None, partner=None, quantity=0):
2223+ taxes = self._applicable(cr, uid, taxes, price_unit, address_id, product, partner)
2224+ res = []
2225+ cur_price_unit=price_unit
2226+ obj_partener_address = self.pool.get('res.partner.address')
2227+ for tax in taxes:
2228+ # we compute the amount for the current tax object and append it to the result
2229+ data = {'id':tax.id,
2230+ 'name':tax.description and tax.description + " - " + tax.name or tax.name,
2231+ 'account_collected_id':tax.account_collected_id.id,
2232+ 'account_paid_id':tax.account_paid_id.id,
2233+ 'base_code_id': tax.base_code_id.id,
2234+ 'ref_base_code_id': tax.ref_base_code_id.id,
2235+ 'sequence': tax.sequence,
2236+ 'base_sign': tax.base_sign,
2237+ 'tax_sign': tax.tax_sign,
2238+ 'ref_base_sign': tax.ref_base_sign,
2239+ 'ref_tax_sign': tax.ref_tax_sign,
2240+ 'price_unit': cur_price_unit,
2241+ 'tax_code_id': tax.tax_code_id.id,
2242+ 'ref_tax_code_id': tax.ref_tax_code_id.id,
2243+ }
2244+ res.append(data)
2245+ if tax.type=='percent':
2246+ amount = cur_price_unit * tax.amount
2247+ data['amount'] = amount
2248+
2249+ elif tax.type=='fixed':
2250+ data['amount'] = tax.amount
2251+ data['tax_amount']=quantity
2252+ # data['amount'] = quantity
2253+ elif tax.type=='code':
2254+ address = address_id and obj_partener_address.browse(cr, uid, address_id) or None
2255+ localdict = {'price_unit':cur_price_unit, 'address':address, 'product':product, 'partner':partner}
2256+ exec tax.python_compute in localdict
2257+ amount = localdict['result']
2258+ data['amount'] = amount
2259+ elif tax.type=='balance':
2260+ data['amount'] = cur_price_unit - reduce(lambda x,y: y.get('amount',0.0)+x, res, 0.0)
2261+ data['balance'] = cur_price_unit
2262+
2263+ amount2 = data.get('amount', 0.0)
2264+ if tax.child_ids:
2265+ if tax.child_depend:
2266+ latest = res.pop()
2267+ amount = amount2
2268+ child_tax = self._unit_compute(cr, uid, tax.child_ids, amount, address_id, product, partner, quantity)
2269+ res.extend(child_tax)
2270+ if tax.child_depend:
2271+ for r in res:
2272+ for name in ('base','ref_base'):
2273+ if latest[name+'_code_id'] and latest[name+'_sign'] and not r[name+'_code_id']:
2274+ r[name+'_code_id'] = latest[name+'_code_id']
2275+ r[name+'_sign'] = latest[name+'_sign']
2276+ r['price_unit'] = latest['price_unit']
2277+ latest[name+'_code_id'] = False
2278+ for name in ('tax','ref_tax'):
2279+ if latest[name+'_code_id'] and latest[name+'_sign'] and not r[name+'_code_id']:
2280+ r[name+'_code_id'] = latest[name+'_code_id']
2281+ r[name+'_sign'] = latest[name+'_sign']
2282+ r['amount'] = data['amount']
2283+ latest[name+'_code_id'] = False
2284+ if tax.include_base_amount:
2285+ cur_price_unit+=amount2
2286+ return res
2287+
2288+ def compute_all(self, cr, uid, taxes, price_unit, quantity, address_id=None, product=None, partner=None, force_excluded=False):
2289+ """
2290+ :param force_excluded: boolean used to say that we don't want to consider the value of field price_include of
2291+ tax. It's used in encoding by line where you don't matter if you encoded a tax with that boolean to True or
2292+ False
2293+ RETURN: {
2294+ 'total': 0.0, # Total without taxes
2295+ 'total_included: 0.0, # Total with taxes
2296+ 'taxes': [] # List of taxes, see compute for the format
2297+ }
2298+ """
2299+ precision = self.pool.get('decimal.precision').precision_get(cr, uid, 'Account')
2300+ totalin = totalex = round(price_unit * quantity, precision)
2301+ tin = []
2302+ tex = []
2303+ for tax in taxes:
2304+ if not tax.price_include or force_excluded:
2305+ tex.append(tax)
2306+ else:
2307+ tin.append(tax)
2308+ tin = self.compute_inv(cr, uid, tin, price_unit, quantity, address_id=address_id, product=product, partner=partner)
2309+ for r in tin:
2310+ totalex -= r.get('amount', 0.0)
2311+ totlex_qty = 0.0
2312+ try:
2313+ totlex_qty = totalex/quantity
2314+ except:
2315+ pass
2316+ tex = self._compute(cr, uid, tex, totlex_qty, quantity, address_id=address_id, product=product, partner=partner)
2317+ for r in tex:
2318+ totalin += r.get('amount', 0.0)
2319+ return {
2320+ 'total': totalex,
2321+ 'total_included': totalin,
2322+ 'taxes': tin + tex
2323+ }
2324+
2325+ def compute(self, cr, uid, taxes, price_unit, quantity, address_id=None, product=None, partner=None):
2326+ logger = netsvc.Logger()
2327+ logger.notifyChannel("warning", netsvc.LOG_WARNING,
2328+ "Deprecated, use compute_all(...)['taxes'] instead of compute(...) to manage prices with tax included")
2329+ return self._compute(cr, uid, taxes, price_unit, quantity, address_id, product, partner)
2330+
2331+ def _compute(self, cr, uid, taxes, price_unit, quantity, address_id=None, product=None, partner=None):
2332+ """
2333+ Compute tax values for given PRICE_UNIT, QUANTITY and a buyer/seller ADDRESS_ID.
2334+
2335+ RETURN:
2336+ [ tax ]
2337+ tax = {'name':'', 'amount':0.0, 'account_collected_id':1, 'account_paid_id':2}
2338+ one tax for each tax id in IDS and their children
2339+ """
2340+ res = self._unit_compute(cr, uid, taxes, price_unit, address_id, product, partner, quantity)
2341+ total = 0.0
2342+ precision_pool = self.pool.get('decimal.precision')
2343+ for r in res:
2344+ if r.get('balance',False):
2345+ r['amount'] = round(r.get('balance', 0.0) * quantity, precision_pool.precision_get(cr, uid, 'Account')) - total
2346+ else:
2347+ r['amount'] = round(r.get('amount', 0.0) * quantity, precision_pool.precision_get(cr, uid, 'Account'))
2348+ total += r['amount']
2349+ return res
2350+
2351+ def _unit_compute_inv(self, cr, uid, taxes, price_unit, address_id=None, product=None, partner=None):
2352+ taxes = self._applicable(cr, uid, taxes, price_unit, address_id, product, partner)
2353+ obj_partener_address = self.pool.get('res.partner.address')
2354+ res = []
2355+ taxes.reverse()
2356+ cur_price_unit = price_unit
2357+
2358+ tax_parent_tot = 0.0
2359+ for tax in taxes:
2360+ if (tax.type=='percent') and not tax.include_base_amount:
2361+ tax_parent_tot += tax.amount
2362+
2363+ for tax in taxes:
2364+ if (tax.type=='fixed') and not tax.include_base_amount:
2365+ cur_price_unit -= tax.amount
2366+
2367+ for tax in taxes:
2368+ if tax.type=='percent':
2369+ if tax.include_base_amount:
2370+ amount = cur_price_unit - (cur_price_unit / (1 + tax.amount))
2371+ else:
2372+ amount = (cur_price_unit / (1 + tax_parent_tot)) * tax.amount
2373+
2374+ elif tax.type=='fixed':
2375+ amount = tax.amount
2376+
2377+ elif tax.type=='code':
2378+ address = address_id and obj_partener_address.browse(cr, uid, address_id) or None
2379+ localdict = {'price_unit':cur_price_unit, 'address':address, 'product':product, 'partner':partner}
2380+ exec tax.python_compute_inv in localdict
2381+ amount = localdict['result']
2382+ elif tax.type=='balance':
2383+ amount = cur_price_unit - reduce(lambda x,y: y.get('amount',0.0)+x, res, 0.0)
2384+
2385+ if tax.include_base_amount:
2386+ cur_price_unit -= amount
2387+ todo = 0
2388+ else:
2389+ todo = 1
2390+ res.append({
2391+ 'id': tax.id,
2392+ 'todo': todo,
2393+ 'name': tax.name,
2394+ 'amount': amount,
2395+ 'account_collected_id': tax.account_collected_id.id,
2396+ 'account_paid_id': tax.account_paid_id.id,
2397+ 'base_code_id': tax.base_code_id.id,
2398+ 'ref_base_code_id': tax.ref_base_code_id.id,
2399+ 'sequence': tax.sequence,
2400+ 'base_sign': tax.base_sign,
2401+ 'tax_sign': tax.tax_sign,
2402+ 'ref_base_sign': tax.ref_base_sign,
2403+ 'ref_tax_sign': tax.ref_tax_sign,
2404+ 'price_unit': cur_price_unit,
2405+ 'tax_code_id': tax.tax_code_id.id,
2406+ 'ref_tax_code_id': tax.ref_tax_code_id.id,
2407+ })
2408+ if tax.child_ids:
2409+ if tax.child_depend:
2410+ del res[-1]
2411+ amount = price_unit
2412+
2413+ parent_tax = self._unit_compute_inv(cr, uid, tax.child_ids, amount, address_id, product, partner)
2414+ res.extend(parent_tax)
2415+
2416+ total = 0.0
2417+ for r in res:
2418+ if r['todo']:
2419+ total += r['amount']
2420+ for r in res:
2421+ r['price_unit'] -= total
2422+ r['todo'] = 0
2423+ return res
2424+
2425+ def compute_inv(self, cr, uid, taxes, price_unit, quantity, address_id=None, product=None, partner=None):
2426+ """
2427+ Compute tax values for given PRICE_UNIT, QUANTITY and a buyer/seller ADDRESS_ID.
2428+ Price Unit is a VAT included price
2429+
2430+ RETURN:
2431+ [ tax ]
2432+ tax = {'name':'', 'amount':0.0, 'account_collected_id':1, 'account_paid_id':2}
2433+ one tax for each tax id in IDS and their children
2434+ """
2435+ res = self._unit_compute_inv(cr, uid, taxes, price_unit, address_id, product, partner=None)
2436+ total = 0.0
2437+ obj_precision = self.pool.get('decimal.precision')
2438+ for r in res:
2439+ prec = obj_precision.precision_get(cr, uid, 'Account')
2440+ if r.get('balance',False):
2441+ r['amount'] = round(r['balance'] * quantity, prec) - total
2442+ else:
2443+ r['amount'] = round(r['amount'] * quantity, prec)
2444+ total += r['amount']
2445+ return res
2446+
2447+account_tax()
2448+
2449+# ---------------------------------------------------------
2450+# Account Entries Models
2451+# ---------------------------------------------------------
2452+
2453+class account_model(osv.osv):
2454+ _name = "account.model"
2455+ _description = "Account Model"
2456+ _columns = {
2457+ 'name': fields.char('Model Name', size=64, required=True, help="This is a model for recurring accounting entries"),
2458+ 'journal_id': fields.many2one('account.journal', 'Journal', required=True),
2459+ 'company_id': fields.related('journal_id', 'company_id', type='many2one', relation='res.company', string='Company', store=True, readonly=True),
2460+ 'lines_id': fields.one2many('account.model.line', 'model_id', 'Model Entries'),
2461+ 'legend': fields.text('Legend', readonly=True, size=100),
2462+ }
2463+
2464+ _defaults = {
2465+ 'legend': lambda self, cr, uid, context:_('You can specify year, month and date in the name of the model using the following labels:\n\n%(year)s: To Specify Year \n%(month)s: To Specify Month \n%(date)s: Current Date\n\ne.g. My model on %(date)s'),
2466+ }
2467+ def generate(self, cr, uid, ids, datas={}, context=None):
2468+ move_ids = []
2469+ entry = {}
2470+ account_move_obj = self.pool.get('account.move')
2471+ account_move_line_obj = self.pool.get('account.move.line')
2472+ pt_obj = self.pool.get('account.payment.term')
2473+ period_obj = self.pool.get('account.period')
2474+
2475+ if context is None:
2476+ context = {}
2477+
2478+ if datas.get('date', False):
2479+ context.update({'date': datas['date']})
2480+
2481+ move_date = context.get('date', time.strftime('%Y-%m-%d'))
2482+ move_date = datetime.strptime(move_date,"%Y-%m-%d")
2483+ for model in self.browse(cr, uid, ids, context=context):
2484+ ctx = context.copy()
2485+ ctx.update({'company_id': model.company_id.id})
2486+ period_ids = period_obj.find(cr, uid, dt=context.get('date', False), context=ctx)
2487+ period_id = period_ids and period_ids[0] or False
2488+ ctx.update({'journal_id': model.journal_id.id,'period_id': period_id})
2489+ try:
2490+ entry['name'] = model.name%{'year': move_date.strftime('%Y'), 'month': move_date.strftime('%m'), 'date': move_date.strftime('%Y-%m')}
2491+ except:
2492+ raise osv.except_osv(_('Wrong model !'), _('You have a wrong expression "%(...)s" in your model !'))
2493+ move_id = account_move_obj.create(cr, uid, {
2494+ 'ref': entry['name'],
2495+ 'period_id': period_id,
2496+ 'journal_id': model.journal_id.id,
2497+ 'date': context.get('date', fields.date.context_today(self,cr,uid,context=context))
2498+ })
2499+ move_ids.append(move_id)
2500+ for line in model.lines_id:
2501+ analytic_account_id = False
2502+ if line.analytic_account_id:
2503+ if not model.journal_id.analytic_journal_id:
2504+ raise osv.except_osv(_('No Analytic Journal !'),_("You have to define an analytic journal on the '%s' journal!") % (model.journal_id.name,))
2505+ analytic_account_id = line.analytic_account_id.id
2506+ val = {
2507+ 'move_id': move_id,
2508+ 'journal_id': model.journal_id.id,
2509+ 'period_id': period_id,
2510+ 'analytic_account_id': analytic_account_id
2511+ }
2512+
2513+ date_maturity = context.get('date',time.strftime('%Y-%m-%d'))
2514+ if line.date_maturity == 'partner':
2515+ if not line.partner_id:
2516+ raise osv.except_osv(_('Error !'), _("Maturity date of entry line generated by model line '%s' of model '%s' is based on partner payment term!" \
2517+ "\nPlease define partner on it!")%(line.name, model.name))
2518+ if line.partner_id.property_payment_term:
2519+ payment_term_id = line.partner_id.property_payment_term.id
2520+ pterm_list = pt_obj.compute(cr, uid, payment_term_id, value=1, date_ref=date_maturity)
2521+ if pterm_list:
2522+ pterm_list = [l[0] for l in pterm_list]
2523+ pterm_list.sort()
2524+ date_maturity = pterm_list[-1]
2525+
2526+ val.update({
2527+ 'name': line.name,
2528+ 'quantity': line.quantity,
2529+ 'debit': line.debit,
2530+ 'credit': line.credit,
2531+ 'account_id': line.account_id.id,
2532+ 'move_id': move_id,
2533+ 'partner_id': line.partner_id.id,
2534+ 'date': context.get('date', fields.date.context_today(self,cr,uid,context=context)),
2535+ 'date_maturity': date_maturity
2536+ })
2537+ account_move_line_obj.create(cr, uid, val, context=ctx)
2538+
2539+ return move_ids
2540+
2541+account_model()
2542+
2543+class account_model_line(osv.osv):
2544+ _name = "account.model.line"
2545+ _description = "Account Model Entries"
2546+ _columns = {
2547+ 'name': fields.char('Name', size=64, required=True),
2548+ 'sequence': fields.integer('Sequence', required=True, help="The sequence field is used to order the resources from lower sequences to higher ones."),
2549+ 'quantity': fields.float('Quantity', digits_compute=dp.get_precision('Account'), help="The optional quantity on entries."),
2550+ 'debit': fields.float('Debit', digits_compute=dp.get_precision('Account')),
2551+ 'credit': fields.float('Credit', digits_compute=dp.get_precision('Account')),
2552+ 'account_id': fields.many2one('account.account', 'Account', required=True, ondelete="cascade"),
2553+ 'analytic_account_id': fields.many2one('account.analytic.account', 'Analytic Account', ondelete="cascade"),
2554+ 'model_id': fields.many2one('account.model', 'Model', required=True, ondelete="cascade", select=True),
2555+ 'amount_currency': fields.float('Amount Currency', help="The amount expressed in an optional other currency."),
2556+ 'currency_id': fields.many2one('res.currency', 'Currency'),
2557+ 'partner_id': fields.many2one('res.partner', 'Partner'),
2558+ 'date_maturity': fields.selection([('today','Date of the day'), ('partner','Partner Payment Term')], 'Maturity Date', help="The maturity date of the generated entries for this model. You can choose between the creation date or the creation date of the entries plus the partner payment terms."),
2559+ }
2560+ _order = 'sequence'
2561+ _sql_constraints = [
2562+ ('credit_debit1', 'CHECK (credit*debit=0)', 'Wrong credit or debit value in model, they must be positive!'),
2563+ ('credit_debit2', 'CHECK (credit+debit>=0)', 'Wrong credit or debit value in model, they must be positive!'),
2564+ ]
2565+account_model_line()
2566+
2567+# ---------------------------------------------------------
2568+# Account Subscription
2569+# ---------------------------------------------------------
2570+
2571+
2572+class account_subscription(osv.osv):
2573+ _name = "account.subscription"
2574+ _description = "Account Subscription"
2575+ _columns = {
2576+ 'name': fields.char('Name', size=64, required=True),
2577+ 'ref': fields.char('Reference', size=16),
2578+ 'model_id': fields.many2one('account.model', 'Model', required=True),
2579+ 'date_start': fields.date('Start Date', required=True),
2580+ 'period_total': fields.integer('Number of Periods', required=True),
2581+ 'period_nbr': fields.integer('Period', required=True),
2582+ 'period_type': fields.selection([('day','days'),('month','month'),('year','year')], 'Period Type', required=True),
2583+ 'state': fields.selection([('draft','Draft'),('running','Running'),('done','Done')], 'State', required=True, readonly=True),
2584+ 'lines_id': fields.one2many('account.subscription.line', 'subscription_id', 'Subscription Lines')
2585+ }
2586+ _defaults = {
2587+ 'date_start': fields.date.context_today,
2588+ 'period_type': 'month',
2589+ 'period_total': 12,
2590+ 'period_nbr': 1,
2591+ 'state': 'draft',
2592+ }
2593+ def state_draft(self, cr, uid, ids, context=None):
2594+ self.write(cr, uid, ids, {'state':'draft'})
2595+ return False
2596+
2597+ def check(self, cr, uid, ids, context=None):
2598+ todone = []
2599+ for sub in self.browse(cr, uid, ids, context=context):
2600+ ok = True
2601+ for line in sub.lines_id:
2602+ if not line.move_id.id:
2603+ ok = False
2604+ break
2605+ if ok:
2606+ todone.append(sub.id)
2607+ if todone:
2608+ self.write(cr, uid, todone, {'state':'done'})
2609+ return False
2610+
2611+ def remove_line(self, cr, uid, ids, context=None):
2612+ toremove = []
2613+ for sub in self.browse(cr, uid, ids, context=context):
2614+ for line in sub.lines_id:
2615+ if not line.move_id.id:
2616+ toremove.append(line.id)
2617+ if toremove:
2618+ self.pool.get('account.subscription.line').unlink(cr, uid, toremove)
2619+ self.write(cr, uid, ids, {'state':'draft'})
2620+ return False
2621+
2622+ def compute(self, cr, uid, ids, context=None):
2623+ for sub in self.browse(cr, uid, ids, context=context):
2624+ ds = sub.date_start
2625+ for i in range(sub.period_total):
2626+ self.pool.get('account.subscription.line').create(cr, uid, {
2627+ 'date': ds,
2628+ 'subscription_id': sub.id,
2629+ })
2630+ if sub.period_type=='day':
2631+ ds = (datetime.strptime(ds, '%Y-%m-%d') + relativedelta(days=sub.period_nbr)).strftime('%Y-%m-%d')
2632+ if sub.period_type=='month':
2633+ ds = (datetime.strptime(ds, '%Y-%m-%d') + relativedelta(months=sub.period_nbr)).strftime('%Y-%m-%d')
2634+ if sub.period_type=='year':
2635+ ds = (datetime.strptime(ds, '%Y-%m-%d') + relativedelta(years=sub.period_nbr)).strftime('%Y-%m-%d')
2636+ self.write(cr, uid, ids, {'state':'running'})
2637+ return True
2638+
2639+account_subscription()
2640+
2641+class account_subscription_line(osv.osv):
2642+ _name = "account.subscription.line"
2643+ _description = "Account Subscription Line"
2644+ _columns = {
2645+ 'subscription_id': fields.many2one('account.subscription', 'Subscription', required=True, select=True),
2646+ 'date': fields.date('Date', required=True),
2647+ 'move_id': fields.many2one('account.move', 'Entry'),
2648+ }
2649+
2650+ def move_create(self, cr, uid, ids, context=None):
2651+ tocheck = {}
2652+ all_moves = []
2653+ obj_model = self.pool.get('account.model')
2654+ for line in self.browse(cr, uid, ids, context=context):
2655+ datas = {
2656+ 'date': line.date,
2657+ }
2658+ move_ids = obj_model.generate(cr, uid, [line.subscription_id.model_id.id], datas, context)
2659+ tocheck[line.subscription_id.id] = True
2660+ self.write(cr, uid, [line.id], {'move_id':move_ids[0]})
2661+ all_moves.extend(move_ids)
2662+ if tocheck:
2663+ self.pool.get('account.subscription').check(cr, uid, tocheck.keys(), context)
2664+ return all_moves
2665+
2666+ _rec_name = 'date'
2667+
2668+account_subscription_line()
2669+
2670+# ---------------------------------------------------------------
2671+# Account Templates: Account, Tax, Tax Code and chart. + Wizard
2672+# ---------------------------------------------------------------
2673+
2674+class account_tax_template(osv.osv):
2675+ _name = 'account.tax.template'
2676+account_tax_template()
2677+
2678+class account_account_template(osv.osv):
2679+ _order = "code"
2680+ _name = "account.account.template"
2681+ _description ='Templates for Accounts'
2682+
2683+ _columns = {
2684+ 'name': fields.char('Name', size=256, required=True, select=True),
2685+ 'currency_id': fields.many2one('res.currency', 'Secondary Currency', help="Forces all moves for this account to have this secondary currency."),
2686+ 'code': fields.char('Code', size=64, required=True, select=1),
2687+ 'type': fields.selection([
2688+ ('receivable','Receivable'),
2689+ ('payable','Payable'),
2690+ ('view','View'),
2691+ ('consolidation','Consolidation'),
2692+ ('liquidity','Liquidity'),
2693+ ('other','Regular'),
2694+ ('closed','Closed'),
2695+ ], 'Internal Type', required=True,help="This type is used to differentiate types with "\
2696+ "special effects in OpenERP: view can not have entries, consolidation are accounts that "\
2697+ "can have children accounts for multi-company consolidations, payable/receivable are for "\
2698+ "partners accounts (for debit/credit computations), closed for depreciated accounts."),
2699+ 'user_type': fields.many2one('account.account.type', 'Account Type', required=True,
2700+ help="These types are defined according to your country. The type contains more information "\
2701+ "about the account and its specificities."),
2702+ 'financial_report_ids': fields.many2many('account.financial.report', 'account_template_financial_report', 'account_template_id', 'report_line_id', 'Financial Reports'),
2703+ 'reconcile': fields.boolean('Allow Reconciliation', help="Check this option if you want the user to reconcile entries in this account."),
2704+ 'shortcut': fields.char('Shortcut', size=12),
2705+ 'note': fields.text('Note'),
2706+ 'parent_id': fields.many2one('account.account.template', 'Parent Account Template', ondelete='cascade'),
2707+ 'child_parent_ids':fields.one2many('account.account.template', 'parent_id', 'Children'),
2708+ 'tax_ids': fields.many2many('account.tax.template', 'account_account_template_tax_rel', 'account_id', 'tax_id', 'Default Taxes'),
2709+ 'nocreate': fields.boolean('Optional create', help="If checked, the new chart of accounts will not contain this by default."),
2710+ 'chart_template_id': fields.many2one('account.chart.template', 'Chart Template', help="This optional field allow you to link an account template to a specific chart template that may differ from the one its root parent belongs to. This allow you to define chart templates that extend another and complete it with few new accounts (You don't need to define the whole structure that is common to both several times)."),
2711+ }
2712+
2713+ _defaults = {
2714+ 'reconcile': False,
2715+ 'type': 'view',
2716+ 'nocreate': False,
2717+ }
2718+
2719+ def _check_type(self, cr, uid, ids, context=None):
2720+ if context is None:
2721+ context = {}
2722+ accounts = self.browse(cr, uid, ids, context=context)
2723+ for account in accounts:
2724+ if account.parent_id and account.parent_id.type != 'view':
2725+ return False
2726+ return True
2727+
2728+ _check_recursion = check_cycle
2729+ _constraints = [
2730+ (_check_recursion, 'Error ! You can not create recursive account templates.', ['parent_id']),
2731+ (_check_type, 'Configuration Error!\nYou can not define children to an account with internal type different of "View"! ', ['type']),
2732+
2733+ ]
2734+
2735+ def name_get(self, cr, uid, ids, context=None):
2736+ if not ids:
2737+ return []
2738+ reads = self.read(cr, uid, ids, ['name','code'], context=context)
2739+ res = []
2740+ for record in reads:
2741+ name = record['name']
2742+ if record['code']:
2743+ name = record['code']+' '+name
2744+ res.append((record['id'],name ))
2745+ return res
2746+
2747+ def generate_account(self, cr, uid, chart_template_id, tax_template_ref, acc_template_ref, code_digits, company_id, context=None):
2748+ """
2749+ This method for generating accounts from templates.
2750+
2751+ :param chart_template_id: id of the chart template chosen in the wizard
2752+ :param tax_template_ref: Taxes templates reference for write taxes_id in account_account.
2753+ :paramacc_template_ref: dictionary with the mappping between the account templates and the real accounts.
2754+ :param code_digits: number of digits got from wizard.multi.charts.accounts, this is use for account code.
2755+ :param company_id: company_id selected from wizard.multi.charts.accounts.
2756+ :returns: return acc_template_ref for reference purpose.
2757+ :rtype: dict
2758+ """
2759+ if context is None:
2760+ context = {}
2761+ obj_acc = self.pool.get('account.account')
2762+ company_name = self.pool.get('res.company').browse(cr, uid, company_id, context=context).name
2763+ template = self.pool.get('account.chart.template').browse(cr, uid, chart_template_id, context=context)
2764+ #deactivate the parent_store functionnality on account_account for rapidity purpose
2765+ ctx = context.copy()
2766+ ctx.update({'defer_parent_store_computation': True})
2767+ level_ref = {}
2768+ children_acc_criteria = [('chart_template_id','=', chart_template_id)]
2769+ if template.account_root_id.id:
2770+ children_acc_criteria = ['|'] + children_acc_criteria + ['&',('parent_id','child_of', [template.account_root_id.id]),('chart_template_id','=', False)]
2771+ children_acc_template = self.search(cr, uid, [('nocreate','!=',True)] + children_acc_criteria, order='id')
2772+ for account_template in self.browse(cr, uid, children_acc_template, context=context):
2773+ # skip the root of COA if it's not the main one
2774+ if (template.account_root_id.id == account_template.id) and template.parent_id:
2775+ continue
2776+ tax_ids = []
2777+ for tax in account_template.tax_ids:
2778+ tax_ids.append(tax_template_ref[tax.id])
2779+
2780+ code_main = account_template.code and len(account_template.code) or 0
2781+ code_acc = account_template.code or ''
2782+ if code_main > 0 and code_main <= code_digits and account_template.type != 'view':
2783+ code_acc = str(code_acc) + (str('0'*(code_digits-code_main)))
2784+ parent_id = account_template.parent_id and ((account_template.parent_id.id in acc_template_ref) and acc_template_ref[account_template.parent_id.id]) or False
2785+ #the level as to be given as well at the creation time, because of the defer_parent_store_computation in
2786+ #context. Indeed because of this, the parent_left and parent_right are not computed and thus the child_of
2787+ #operator does not return the expected values, with result of having the level field not computed at all.
2788+ if parent_id:
2789+ level = parent_id in level_ref and level_ref[parent_id] + 1 or obj_acc._get_level(cr, uid, [parent_id], 'level', None, context=context)[parent_id] + 1
2790+ else:
2791+ level = 0
2792+ vals={
2793+ 'name': (template.account_root_id.id == account_template.id) and company_name or account_template.name,
2794+ 'currency_id': account_template.currency_id and account_template.currency_id.id or False,
2795+ 'code': code_acc,
2796+ 'type': account_template.type,
2797+ 'user_type': account_template.user_type and account_template.user_type.id or False,
2798+ 'reconcile': account_template.reconcile,
2799+ 'shortcut': account_template.shortcut,
2800+ 'note': account_template.note,
2801+ 'financial_report_ids': account_template.financial_report_ids and [(6,0,[x.id for x in account_template.financial_report_ids])] or False,
2802+ 'parent_id': parent_id,
2803+ 'tax_ids': [(6,0,tax_ids)],
2804+ 'company_id': company_id,
2805+ 'level': level,
2806+ }
2807+ new_account = obj_acc.create(cr, uid, vals, context=ctx)
2808+ acc_template_ref[account_template.id] = new_account
2809+ level_ref[new_account] = level
2810+
2811+ #reactivate the parent_store functionnality on account_account
2812+ obj_acc._parent_store_compute(cr)
2813+ return acc_template_ref
2814+
2815+account_account_template()
2816+
2817+class account_add_tmpl_wizard(osv.osv_memory):
2818+ """Add one more account from the template.
2819+
2820+ With the 'nocreate' option, some accounts may not be created. Use this to add them later."""
2821+ _name = 'account.addtmpl.wizard'
2822+
2823+ def _get_def_cparent(self, cr, uid, context=None):
2824+ acc_obj = self.pool.get('account.account')
2825+ tmpl_obj = self.pool.get('account.account.template')
2826+ tids = tmpl_obj.read(cr, uid, [context['tmpl_ids']], ['parent_id'])
2827+ if not tids or not tids[0]['parent_id']:
2828+ return False
2829+ ptids = tmpl_obj.read(cr, uid, [tids[0]['parent_id'][0]], ['code'])
2830+ res = None
2831+ if not ptids or not ptids[0]['code']:
2832+ raise osv.except_osv(_('Error !'), _('I can not locate a parent code for the template account!'))
2833+ res = acc_obj.search(cr, uid, [('code','=',ptids[0]['code'])])
2834+ return res and res[0] or False
2835+
2836+ _columns = {
2837+ 'cparent_id':fields.many2one('account.account', 'Parent target', help="Creates an account with the selected template under this existing parent.", required=True),
2838+ }
2839+ _defaults = {
2840+ 'cparent_id': _get_def_cparent,
2841+ }
2842+
2843+ def action_create(self,cr,uid,ids,context=None):
2844+ if context is None:
2845+ context = {}
2846+ acc_obj = self.pool.get('account.account')
2847+ tmpl_obj = self.pool.get('account.account.template')
2848+ data = self.read(cr, uid, ids)[0]
2849+ company_id = acc_obj.read(cr, uid, [data['cparent_id'][0]], ['company_id'])[0]['company_id'][0]
2850+ account_template = tmpl_obj.browse(cr, uid, context['tmpl_ids'])
2851+ vals = {
2852+ 'name': account_template.name,
2853+ 'currency_id': account_template.currency_id and account_template.currency_id.id or False,
2854+ 'code': account_template.code,
2855+ 'type': account_template.type,
2856+ 'user_type': account_template.user_type and account_template.user_type.id or False,
2857+ 'reconcile': account_template.reconcile,
2858+ 'shortcut': account_template.shortcut,
2859+ 'note': account_template.note,
2860+ 'parent_id': data['cparent_id'][0],
2861+ 'company_id': company_id,
2862+ }
2863+ acc_obj.create(cr, uid, vals)
2864+ return {'type':'state', 'state': 'end' }
2865+
2866+ def action_cancel(self, cr, uid, ids, context=None):
2867+ return { 'type': 'state', 'state': 'end' }
2868+
2869+account_add_tmpl_wizard()
2870+
2871+class account_tax_code_template(osv.osv):
2872+
2873+ _name = 'account.tax.code.template'
2874+ _description = 'Tax Code Template'
2875+ _order = 'code'
2876+ _rec_name = 'code'
2877+ _columns = {
2878+ 'name': fields.char('Tax Case Name', size=64, required=True),
2879+ 'code': fields.char('Case Code', size=64),
2880+ 'info': fields.text('Description'),
2881+ 'parent_id': fields.many2one('account.tax.code.template', 'Parent Code', select=True),
2882+ 'child_ids': fields.one2many('account.tax.code.template', 'parent_id', 'Child Codes'),
2883+ 'sign': fields.float('Sign For Parent', required=True),
2884+ 'notprintable':fields.boolean("Not Printable in Invoice", help="Check this box if you don't want any VAT related to this Tax Code to appear on invoices"),
2885+ }
2886+
2887+ _defaults = {
2888+ 'sign': 1.0,
2889+ 'notprintable': False,
2890+ }
2891+
2892+ def generate_tax_code(self, cr, uid, tax_code_root_id, company_id, context=None):
2893+ '''
2894+ This function generates the tax codes from the templates of tax code that are children of the given one passed
2895+ in argument. Then it returns a dictionary with the mappping between the templates and the real objects.
2896+
2897+ :param tax_code_root_id: id of the root of all the tax code templates to process
2898+ :param company_id: id of the company the wizard is running for
2899+ :returns: dictionary with the mappping between the templates and the real objects.
2900+ :rtype: dict
2901+ '''
2902+ obj_tax_code_template = self.pool.get('account.tax.code.template')
2903+ obj_tax_code = self.pool.get('account.tax.code')
2904+ tax_code_template_ref = {}
2905+ company = self.pool.get('res.company').browse(cr, uid, company_id, context=context)
2906+
2907+ #find all the children of the tax_code_root_id
2908+ children_tax_code_template = tax_code_root_id and obj_tax_code_template.search(cr, uid, [('parent_id','child_of',[tax_code_root_id])], order='id') or []
2909+ for tax_code_template in obj_tax_code_template.browse(cr, uid, children_tax_code_template, context=context):
2910+ vals = {
2911+ 'name': (tax_code_root_id == tax_code_template.id) and company.name or tax_code_template.name,
2912+ 'code': tax_code_template.code,
2913+ 'info': tax_code_template.info,
2914+ 'parent_id': tax_code_template.parent_id and ((tax_code_template.parent_id.id in tax_code_template_ref) and tax_code_template_ref[tax_code_template.parent_id.id]) or False,
2915+ 'company_id': company_id,
2916+ 'sign': tax_code_template.sign,
2917+ }
2918+ #check if this tax code already exists
2919+ rec_list = obj_tax_code.search(cr, uid, [('name', '=', vals['name']),('code', '=', vals['code']),('company_id', '=', vals['company_id'])], context=context)
2920+ if not rec_list:
2921+ #if not yet, create it
2922+ new_tax_code = obj_tax_code.create(cr, uid, vals)
2923+ #recording the new tax code to do the mapping
2924+ tax_code_template_ref[tax_code_template.id] = new_tax_code
2925+ return tax_code_template_ref
2926+
2927+ def name_get(self, cr, uid, ids, context=None):
2928+ if not ids:
2929+ return []
2930+ if isinstance(ids, (int, long)):
2931+ ids = [ids]
2932+ reads = self.read(cr, uid, ids, ['name','code'], context, load='_classic_write')
2933+ return [(x['id'], (x['code'] and x['code'] + ' - ' or '') + x['name']) \
2934+ for x in reads]
2935+
2936+ _check_recursion = check_cycle
2937+ _constraints = [
2938+ (_check_recursion, 'Error ! You can not create recursive Tax Codes.', ['parent_id'])
2939+ ]
2940+ _order = 'code,name'
2941+account_tax_code_template()
2942+
2943+
2944+class account_chart_template(osv.osv):
2945+ _name="account.chart.template"
2946+ _description= "Templates for Account Chart"
2947+
2948+ _columns={
2949+ 'name': fields.char('Name', size=64, required=True),
2950+ 'parent_id': fields.many2one('account.chart.template', 'Parent Chart Template'),
2951+ 'code_digits': fields.integer('# of Digits', required=True, help="No. of Digits to use for account code"),
2952+ 'visible': fields.boolean('Can be Visible?', help="Set this to False if you don't want this template to be used actively in the wizard that generate Chart of Accounts from templates, this is useful when you want to generate accounts of this template only when loading its child template."),
2953+ 'complete_tax_set': fields.boolean('Complete Set of Taxes', help='This boolean helps you to choose if you want to propose to the user to encode the sale and purchase rates or choose from list of taxes. This last choice assumes that the set of tax defined on this template is complete'),
2954+ 'account_root_id': fields.many2one('account.account.template', 'Root Account', domain=[('parent_id','=',False)]),
2955+ 'tax_code_root_id': fields.many2one('account.tax.code.template', 'Root Tax Code', domain=[('parent_id','=',False)]),
2956+ 'tax_template_ids': fields.one2many('account.tax.template', 'chart_template_id', 'Tax Template List', help='List of all the taxes that have to be installed by the wizard'),
2957+ 'bank_account_view_id': fields.many2one('account.account.template', 'Bank Account'),
2958+ 'property_account_receivable': fields.many2one('account.account.template', 'Receivable Account'),
2959+ 'property_account_payable': fields.many2one('account.account.template', 'Payable Account'),
2960+ 'property_account_expense_categ': fields.many2one('account.account.template', 'Expense Category Account'),
2961+ 'property_account_income_categ': fields.many2one('account.account.template', 'Income Category Account'),
2962+ 'property_account_expense': fields.many2one('account.account.template', 'Expense Account on Product Template'),
2963+ 'property_account_income': fields.many2one('account.account.template', 'Income Account on Product Template'),
2964+ 'property_reserve_and_surplus_account': fields.many2one('account.account.template', 'Reserve and Profit/Loss Account', domain=[('type', '=', 'payable')], help='This Account is used for transferring Profit/Loss(If It is Profit: Amount will be added, Loss: Amount will be deducted.), Which is calculated from Profilt & Loss Report'),
2965+ 'property_account_income_opening': fields.many2one('account.account.template', 'Opening Entries Income Account'),
2966+ 'property_account_expense_opening': fields.many2one('account.account.template', 'Opening Entries Expense Account'),
2967+ }
2968+
2969+ _defaults = {
2970+ 'visible': True,
2971+ 'code_digits': 6,
2972+ 'complete_tax_set': True,
2973+ }
2974+
2975+account_chart_template()
2976+
2977+class account_tax_template(osv.osv):
2978+
2979+ _name = 'account.tax.template'
2980+ _description = 'Templates for Taxes'
2981+
2982+ _columns = {
2983+ 'chart_template_id': fields.many2one('account.chart.template', 'Chart Template', required=True),
2984+ 'name': fields.char('Tax Name', size=64, required=True),
2985+ 'sequence': fields.integer('Sequence', required=True, help="The sequence field is used to order the taxes lines from lower sequences to higher ones. The order is important if you have a tax that has several tax children. In this case, the evaluation order is important."),
2986+ 'amount': fields.float('Amount', required=True, digits=(14,4), help="For Tax Type percent enter % ratio between 0-1."),
2987+ 'type': fields.selection( [('percent','Percent'), ('fixed','Fixed'), ('none','None'), ('code','Python Code'), ('balance','Balance')], 'Tax Type', required=True),
2988+ 'applicable_type': fields.selection( [('true','True'), ('code','Python Code')], 'Applicable Type', required=True, help="If not applicable (computed through a Python code), the tax won't appear on the invoice."),
2989+ 'domain':fields.char('Domain', size=32, help="This field is only used if you develop your own module allowing developers to create specific taxes in a custom domain."),
2990+ 'account_collected_id':fields.many2one('account.account.template', 'Invoice Tax Account'),
2991+ 'account_paid_id':fields.many2one('account.account.template', 'Refund Tax Account'),
2992+ 'parent_id':fields.many2one('account.tax.template', 'Parent Tax Account', select=True),
2993+ 'child_depend':fields.boolean('Tax on Children', help="Set if the tax computation is based on the computation of child taxes rather than on the total amount."),
2994+ 'python_compute':fields.text('Python Code'),
2995+ 'python_compute_inv':fields.text('Python Code (reverse)'),
2996+ 'python_applicable':fields.text('Python Code'),
2997+
2998+ #
2999+ # Fields used for the VAT declaration
3000+ #
3001+ 'base_code_id': fields.many2one('account.tax.code.template', 'Base Code', help="Use this code for the VAT declaration."),
3002+ 'tax_code_id': fields.many2one('account.tax.code.template', 'Tax Code', help="Use this code for the VAT declaration."),
3003+ 'base_sign': fields.float('Base Code Sign', help="Usually 1 or -1."),
3004+ 'tax_sign': fields.float('Tax Code Sign', help="Usually 1 or -1."),
3005+
3006+ # Same fields for refund invoices
3007+
3008+ 'ref_base_code_id': fields.many2one('account.tax.code.template', 'Refund Base Code', help="Use this code for the VAT declaration."),
3009+ 'ref_tax_code_id': fields.many2one('account.tax.code.template', 'Refund Tax Code', help="Use this code for the VAT declaration."),
3010+ 'ref_base_sign': fields.float('Base Code Sign', help="Usually 1 or -1."),
3011+ 'ref_tax_sign': fields.float('Tax Code Sign', help="Usually 1 or -1."),
3012+ 'include_base_amount': fields.boolean('Include in Base Amount', help="Set if the amount of tax must be included in the base amount before computing the next taxes."),
3013+ 'description': fields.char('Internal Name', size=32),
3014+ 'type_tax_use': fields.selection([('sale','Sale'),('purchase','Purchase'),('all','All')], 'Tax Use In', required=True,),
3015+ 'price_include': fields.boolean('Tax Included in Price', help="Check this if the price you use on the product and invoices includes this tax."),
3016+ }
3017+
3018+ def name_get(self, cr, uid, ids, context=None):
3019+ if not ids:
3020+ return []
3021+ res = []
3022+ for record in self.read(cr, uid, ids, ['description','name'], context=context):
3023+ name = record['description'] and record['description'] or record['name']
3024+ res.append((record['id'],name ))
3025+ return res
3026+
3027+ def _default_company(self, cr, uid, context=None):
3028+ user = self.pool.get('res.users').browse(cr, uid, uid, context=context)
3029+ if user.company_id:
3030+ return user.company_id.id
3031+ return self.pool.get('res.company').search(cr, uid, [('parent_id', '=', False)])[0]
3032+
3033+ _defaults = {
3034+ 'python_compute': lambda *a: '''# price_unit\n# address: res.partner.address object or False\n# product: product.product object or None\n# partner: res.partner object or None\n\nresult = price_unit * 0.10''',
3035+ 'python_compute_inv': lambda *a: '''# price_unit\n# address: res.partner.address object or False\n# product: product.product object or False\n\nresult = price_unit * 0.10''',
3036+ 'applicable_type': 'true',
3037+ 'type': 'percent',
3038+ 'amount': 0,
3039+ 'sequence': 1,
3040+ 'ref_tax_sign': 1,
3041+ 'ref_base_sign': 1,
3042+ 'tax_sign': 1,
3043+ 'base_sign': 1,
3044+ 'include_base_amount': False,
3045+ 'type_tax_use': 'all',
3046+ 'price_include': 0,
3047+ }
3048+ _order = 'sequence'
3049+
3050+ def _generate_tax(self, cr, uid, tax_templates, tax_code_template_ref, company_id, context=None):
3051+ """
3052+ This method generate taxes from templates.
3053+
3054+ :param tax_templates: list of browse record of the tax templates to process
3055+ :param tax_code_template_ref: Taxcode templates reference.
3056+ :param company_id: id of the company the wizard is running for
3057+ :returns:
3058+ {
3059+ 'tax_template_to_tax': mapping between tax template and the newly generated taxes corresponding,
3060+ 'account_dict': dictionary containing a to-do list with all the accounts to assign on new taxes
3061+ }
3062+ """
3063+ if context is None:
3064+ context = {}
3065+ res = {}
3066+ todo_dict = {}
3067+ tax_template_to_tax = {}
3068+ for tax in tax_templates:
3069+ vals_tax = {
3070+ 'name':tax.name,
3071+ 'sequence': tax.sequence,
3072+ 'amount': tax.amount,
3073+ 'type': tax.type,
3074+ 'applicable_type': tax.applicable_type,
3075+ 'domain': tax.domain,
3076+ 'parent_id': tax.parent_id and ((tax.parent_id.id in tax_template_to_tax) and tax_template_to_tax[tax.parent_id.id]) or False,
3077+ 'child_depend': tax.child_depend,
3078+ 'python_compute': tax.python_compute,
3079+ 'python_compute_inv': tax.python_compute_inv,
3080+ 'python_applicable': tax.python_applicable,
3081+ 'base_code_id': tax.base_code_id and ((tax.base_code_id.id in tax_code_template_ref) and tax_code_template_ref[tax.base_code_id.id]) or False,
3082+ 'tax_code_id': tax.tax_code_id and ((tax.tax_code_id.id in tax_code_template_ref) and tax_code_template_ref[tax.tax_code_id.id]) or False,
3083+ 'base_sign': tax.base_sign,
3084+ 'tax_sign': tax.tax_sign,
3085+ 'ref_base_code_id': tax.ref_base_code_id and ((tax.ref_base_code_id.id in tax_code_template_ref) and tax_code_template_ref[tax.ref_base_code_id.id]) or False,
3086+ 'ref_tax_code_id': tax.ref_tax_code_id and ((tax.ref_tax_code_id.id in tax_code_template_ref) and tax_code_template_ref[tax.ref_tax_code_id.id]) or False,
3087+ 'ref_base_sign': tax.ref_base_sign,
3088+ 'ref_tax_sign': tax.ref_tax_sign,
3089+ 'include_base_amount': tax.include_base_amount,
3090+ 'description': tax.description,
3091+ 'company_id': company_id,
3092+ 'type_tax_use': tax.type_tax_use,
3093+ 'price_include': tax.price_include
3094+ }
3095+ new_tax = self.pool.get('account.tax').create(cr, uid, vals_tax)
3096+ tax_template_to_tax[tax.id] = new_tax
3097+ #as the accounts have not been created yet, we have to wait before filling these fields
3098+ todo_dict[new_tax] = {
3099+ 'account_collected_id': tax.account_collected_id and tax.account_collected_id.id or False,
3100+ 'account_paid_id': tax.account_paid_id and tax.account_paid_id.id or False,
3101+ }
3102+ res.update({'tax_template_to_tax': tax_template_to_tax, 'account_dict': todo_dict})
3103+ return res
3104+
3105+account_tax_template()
3106+
3107+# Fiscal Position Templates
3108+
3109+class account_fiscal_position_template(osv.osv):
3110+ _name = 'account.fiscal.position.template'
3111+ _description = 'Template for Fiscal Position'
3112+
3113+ _columns = {
3114+ 'name': fields.char('Fiscal Position Template', size=64, required=True),
3115+ 'chart_template_id': fields.many2one('account.chart.template', 'Chart Template', required=True),
3116+ 'account_ids': fields.one2many('account.fiscal.position.account.template', 'position_id', 'Account Mapping'),
3117+ 'tax_ids': fields.one2many('account.fiscal.position.tax.template', 'position_id', 'Tax Mapping'),
3118+ 'note': fields.text('Notes', translate=True),
3119+ }
3120+
3121+ def generate_fiscal_position(self, cr, uid, chart_temp_id, tax_template_ref, acc_template_ref, company_id, context=None):
3122+ """
3123+ This method generate Fiscal Position, Fiscal Position Accounts and Fiscal Position Taxes from templates.
3124+
3125+ :param chart_temp_id: Chart Template Id.
3126+ :param taxes_ids: Taxes templates reference for generating account.fiscal.position.tax.
3127+ :param acc_template_ref: Account templates reference for generating account.fiscal.position.account.
3128+ :param company_id: company_id selected from wizard.multi.charts.accounts.
3129+ :returns: True
3130+ """
3131+ if context is None:
3132+ context = {}
3133+ obj_tax_fp = self.pool.get('account.fiscal.position.tax')
3134+ obj_ac_fp = self.pool.get('account.fiscal.position.account')
3135+ obj_fiscal_position = self.pool.get('account.fiscal.position')
3136+ fp_ids = self.search(cr, uid, [('chart_template_id', '=', chart_temp_id)])
3137+ for position in self.browse(cr, uid, fp_ids, context=context):
3138+ new_fp = obj_fiscal_position.create(cr, uid, {'company_id': company_id, 'name': position.name, 'note': position.note})
3139+ for tax in position.tax_ids:
3140+ obj_tax_fp.create(cr, uid, {
3141+ 'tax_src_id': tax_template_ref[tax.tax_src_id.id],
3142+ 'tax_dest_id': tax.tax_dest_id and tax_template_ref[tax.tax_dest_id.id] or False,
3143+ 'position_id': new_fp
3144+ })
3145+ for acc in position.account_ids:
3146+ obj_ac_fp.create(cr, uid, {
3147+ 'account_src_id': acc_template_ref[acc.account_src_id.id],
3148+ 'account_dest_id': acc_template_ref[acc.account_dest_id.id],
3149+ 'position_id': new_fp
3150+ })
3151+ return True
3152+
3153+account_fiscal_position_template()
3154+
3155+class account_fiscal_position_tax_template(osv.osv):
3156+ _name = 'account.fiscal.position.tax.template'
3157+ _description = 'Template Tax Fiscal Position'
3158+ _rec_name = 'position_id'
3159+
3160+ _columns = {
3161+ 'position_id': fields.many2one('account.fiscal.position.template', 'Fiscal Position', required=True, ondelete='cascade'),
3162+ 'tax_src_id': fields.many2one('account.tax.template', 'Tax Source', required=True),
3163+ 'tax_dest_id': fields.many2one('account.tax.template', 'Replacement Tax')
3164+ }
3165+
3166+account_fiscal_position_tax_template()
3167+
3168+class account_fiscal_position_account_template(osv.osv):
3169+ _name = 'account.fiscal.position.account.template'
3170+ _description = 'Template Account Fiscal Mapping'
3171+ _rec_name = 'position_id'
3172+ _columns = {
3173+ 'position_id': fields.many2one('account.fiscal.position.template', 'Fiscal Mapping', required=True, ondelete='cascade'),
3174+ 'account_src_id': fields.many2one('account.account.template', 'Account Source', domain=[('type','<>','view')], required=True),
3175+ 'account_dest_id': fields.many2one('account.account.template', 'Account Destination', domain=[('type','<>','view')], required=True)
3176+ }
3177+
3178+account_fiscal_position_account_template()
3179+
3180+# ---------------------------------------------------------
3181+# Account generation from template wizards
3182+# ---------------------------------------------------------
3183+
3184+class wizard_multi_charts_accounts(osv.osv_memory):
3185+ """
3186+ Create a new account chart for a company.
3187+ Wizards ask for:
3188+ * a company
3189+ * an account chart template
3190+ * a number of digits for formatting code of non-view accounts
3191+ * a list of bank accounts owned by the company
3192+ Then, the wizard:
3193+ * generates all accounts from the template and assigns them to the right company
3194+ * generates all taxes and tax codes, changing account assignations
3195+ * generates all accounting properties and assigns them correctly
3196+ """
3197+ _name='wizard.multi.charts.accounts'
3198+ _inherit = 'res.config'
3199+
3200+ _columns = {
3201+ 'company_id':fields.many2one('res.company', 'Company', required=True),
3202+ 'chart_template_id': fields.many2one('account.chart.template', 'Chart Template', required=True),
3203+ 'bank_accounts_id': fields.one2many('account.bank.accounts.wizard', 'bank_account_id', 'Cash and Banks', required=True),
3204+ 'code_digits':fields.integer('# of Digits', required=True, help="No. of Digits to use for account code"),
3205+ 'seq_journal':fields.boolean('Separated Journal Sequences', help="Check this box if you want to use a different sequence for each created journal. Otherwise, all will use the same sequence."),
3206+ "sale_tax": fields.many2one("account.tax.template", "Default Sale Tax"),
3207+ "purchase_tax": fields.many2one("account.tax.template", "Default Purchase Tax"),
3208+ 'sale_tax_rate': fields.float('Sales Tax(%)'),
3209+ 'purchase_tax_rate': fields.float('Purchase Tax(%)'),
3210+ 'complete_tax_set': fields.boolean('Complete Set of Taxes', help='This boolean helps you to choose if you want to propose to the user to encode the sales and purchase rates or use the usual m2o fields. This last choice assumes that the set of tax defined for the chosen template is complete'),
3211+ }
3212+ def onchange_tax_rate(self, cr, uid, ids, rate=False, context=None):
3213+ return {'value': {'purchase_tax_rate': rate or False}}
3214+
3215+ def onchange_chart_template_id(self, cr, uid, ids, chart_template_id=False, context=None):
3216+ res = {}
3217+ tax_templ_obj = self.pool.get('account.tax.template')
3218+ res['value'] = {'complete_tax_set': False, 'sale_tax': False, 'purchase_tax': False}
3219+ if chart_template_id:
3220+ data = self.pool.get('account.chart.template').browse(cr, uid, chart_template_id, context=context)
3221+ res['value'].update({'complete_tax_set': data.complete_tax_set})
3222+ if data.complete_tax_set:
3223+ # default tax is given by the lowest sequence. For same sequence we will take the latest created as it will be the case for tax created while isntalling the generic chart of account
3224+ sale_tax_ids = tax_templ_obj.search(cr, uid, [("chart_template_id"
3225+ , "=", chart_template_id), ('type_tax_use', 'in', ('sale','all'))], order="sequence, id desc")
3226+ purchase_tax_ids = tax_templ_obj.search(cr, uid, [("chart_template_id"
3227+ , "=", chart_template_id), ('type_tax_use', 'in', ('purchase','all'))], order="sequence, id desc")
3228+ res['value'].update({'sale_tax': sale_tax_ids and sale_tax_ids[0] or False, 'purchase_tax': purchase_tax_ids and purchase_tax_ids[0] or False})
3229+
3230+ if data.code_digits:
3231+ res['value'].update({'code_digits': data.code_digits})
3232+ return res
3233+
3234+ def default_get(self, cr, uid, fields, context=None):
3235+ res = super(wizard_multi_charts_accounts, self).default_get(cr, uid, fields, context=context)
3236+ tax_templ_obj = self.pool.get('account.tax.template')
3237+
3238+ if 'bank_accounts_id' in fields:
3239+ res.update({'bank_accounts_id': [{'acc_name': _('Cash'), 'account_type': 'cash'},{'acc_name': _('Bank'), 'account_type': 'bank'}]})
3240+ if 'company_id' in fields:
3241+ res.update({'company_id': self.pool.get('res.users').browse(cr, uid, [uid], context=context)[0].company_id.id})
3242+ if 'seq_journal' in fields:
3243+ res.update({'seq_journal': True})
3244+
3245+ ids = self.pool.get('account.chart.template').search(cr, uid, [('visible', '=', True)], context=context)
3246+ if ids:
3247+ if 'chart_template_id' in fields:
3248+ res.update({'chart_template_id': ids[0]})
3249+ if 'sale_tax' in fields:
3250+ sale_tax_ids = tax_templ_obj.search(cr, uid, [("chart_template_id"
3251+ , "=", ids[0]), ('type_tax_use', 'in', ('sale','all'))], order="sequence")
3252+ res.update({'sale_tax': sale_tax_ids and sale_tax_ids[0] or False})
3253+ if 'purchase_tax' in fields:
3254+ purchase_tax_ids = tax_templ_obj.search(cr, uid, [("chart_template_id"
3255+ , "=", ids[0]), ('type_tax_use', 'in', ('purchase','all'))], order="sequence")
3256+ res.update({'purchase_tax': purchase_tax_ids and purchase_tax_ids[0] or False})
3257+ res.update({
3258+ 'purchase_tax_rate': 15.0,
3259+ 'sale_tax_rate': 15.0,
3260+ })
3261+ return res
3262+
3263+ def fields_view_get(self, cr, uid, view_id=None, view_type='form', context=None, toolbar=False, submenu=False):
3264+ res = super(wizard_multi_charts_accounts, self).fields_view_get(cr, uid, view_id=view_id, view_type=view_type, context=context, toolbar=toolbar,submenu=False)
3265+ cmp_select = []
3266+ acc_template_obj = self.pool.get('account.chart.template')
3267+ company_obj = self.pool.get('res.company')
3268+
3269+ company_ids = company_obj.search(cr, uid, [], context=context)
3270+ #display in the widget selection of companies, only the companies that haven't been configured yet (but don't care about the demo chart of accounts)
3271+ cr.execute("SELECT company_id FROM account_account WHERE active = 't' AND account_account.parent_id IS NULL AND name != %s", ("Chart For Automated Tests",))
3272+ configured_cmp = [r[0] for r in cr.fetchall()]
3273+ unconfigured_cmp = list(set(company_ids)-set(configured_cmp))
3274+ for field in res['fields']:
3275+ if field == 'company_id':
3276+ res['fields'][field]['domain'] = [('id','in',unconfigured_cmp)]
3277+ res['fields'][field]['selection'] = [('', '')]
3278+ if unconfigured_cmp:
3279+ cmp_select = [(line.id, line.name) for line in company_obj.browse(cr, uid, unconfigured_cmp)]
3280+ res['fields'][field]['selection'] = cmp_select
3281+ return res
3282+
3283+ def check_created_journals(self, cr, uid, vals_journal, company_id, context=None):
3284+ """
3285+ This method used for checking journals already created or not. If not then create new journal.
3286+ """
3287+ obj_journal = self.pool.get('account.journal')
3288+ rec_list = obj_journal.search(cr, uid, [('name','=', vals_journal['name']),('company_id', '=', company_id)], context=context)
3289+ if not rec_list:
3290+ obj_journal.create(cr, uid, vals_journal, context=context)
3291+ return True
3292+
3293+ def generate_journals(self, cr, uid, chart_template_id, acc_template_ref, company_id, context=None):
3294+ """
3295+ This method is used for creating journals.
3296+
3297+ :param chart_temp_id: Chart Template Id.
3298+ :param acc_template_ref: Account templates reference.
3299+ :param company_id: company_id selected from wizard.multi.charts.accounts.
3300+ :returns: True
3301+ """
3302+ journal_data = self._prepare_all_journals(cr, uid, chart_template_id, acc_template_ref, company_id, context=context)
3303+ for vals_journal in journal_data:
3304+ self.check_created_journals(cr, uid, vals_journal, company_id, context=context)
3305+ return True
3306+
3307+ def _prepare_all_journals(self, cr, uid, chart_template_id, acc_template_ref, company_id, context=None):
3308+ def _get_analytic_journal(journal_type):
3309+ # Get the analytic journal
3310+ data = False
3311+ if journal_type in ('sale', 'sale_refund'):
3312+ data = obj_data.get_object_reference(cr, uid, 'account', 'analytic_journal_sale')
3313+ elif journal_type in ('purchase', 'purchase_refund'):
3314+ pass
3315+ elif journal_type == 'general':
3316+ pass
3317+ return data and data[1] or False
3318+
3319+ def _get_default_account(journal_type, type='debit'):
3320+ # Get the default accounts
3321+ default_account = False
3322+ if journal_type in ('sale', 'sale_refund'):
3323+ default_account = acc_template_ref.get(template.property_account_income_categ.id)
3324+ elif journal_type in ('purchase', 'purchase_refund'):
3325+ default_account = acc_template_ref.get(template.property_account_expense_categ.id)
3326+ elif journal_type == 'situation':
3327+ if type == 'debit':
3328+ default_account = acc_template_ref.get(template.property_account_expense_opening.id)
3329+ else:
3330+ default_account = acc_template_ref.get(template.property_account_income_opening.id)
3331+ return default_account
3332+
3333+ def _get_view_id(journal_type):
3334+ # Get the journal views
3335+ if journal_type in ('general', 'situation'):
3336+ data = obj_data.get_object_reference(cr, uid, 'account', 'account_journal_view')
3337+ elif journal_type in ('sale_refund', 'purchase_refund'):
3338+ data = obj_data.get_object_reference(cr, uid, 'account', 'account_sp_refund_journal_view')
3339+ else:
3340+ data = obj_data.get_object_reference(cr, uid, 'account', 'account_sp_journal_view')
3341+ return data and data[1] or False
3342+
3343+ journal_names = {
3344+ 'sale': _('Sales Journal'),
3345+ 'purchase': _('Purchase Journal'),
3346+ 'sale_refund': _('Sales Refund Journal'),
3347+ 'purchase_refund': _('Purchase Refund Journal'),
3348+ 'general': _('Miscellaneous Journal'),
3349+ 'situation': _('Opening Entries Journal'),
3350+ }
3351+ journal_codes = {
3352+ 'sale': _('SAJ'),
3353+ 'purchase': _('EXJ'),
3354+ 'sale_refund': _('SCNJ'),
3355+ 'purchase_refund': _('ECNJ'),
3356+ 'general': _('MISC'),
3357+ 'situation': _('OPEJ'),
3358+ }
3359+
3360+ obj_data = self.pool.get('ir.model.data')
3361+ analytic_journal_obj = self.pool.get('account.analytic.journal')
3362+ template = self.pool.get('account.chart.template').browse(cr, uid, chart_template_id, context=context)
3363+
3364+ journal_data = []
3365+ for journal_type in ['sale', 'purchase', 'sale_refund', 'purchase_refund', 'general', 'situation']:
3366+ vals = {
3367+ 'type': journal_type,
3368+ 'name': journal_names[journal_type],
3369+ 'code': journal_codes[journal_type],
3370+ 'company_id': company_id,
3371+ 'centralisation': journal_type == 'situation',
3372+ 'view_id': _get_view_id(journal_type),
3373+ 'analytic_journal_id': _get_analytic_journal(journal_type),
3374+ 'default_credit_account_id': _get_default_account(journal_type, 'credit'),
3375+ 'default_debit_account_id': _get_default_account(journal_type, 'debit'),
3376+ }
3377+ journal_data.append(vals)
3378+ return journal_data
3379+
3380+ def generate_properties(self, cr, uid, chart_template_id, acc_template_ref, company_id, context=None):
3381+ """
3382+ This method used for creating properties.
3383+
3384+ :param chart_template_id: id of the current chart template for which we need to create properties
3385+ :param acc_template_ref: Mapping between ids of account templates and real accounts created from them
3386+ :param company_id: company_id selected from wizard.multi.charts.accounts.
3387+ :returns: True
3388+ """
3389+ property_obj = self.pool.get('ir.property')
3390+ field_obj = self.pool.get('ir.model.fields')
3391+ todo_list = [
3392+ ('property_account_receivable','res.partner','account.account'),
3393+ ('property_account_payable','res.partner','account.account'),
3394+ ('property_account_expense_categ','product.category','account.account'),
3395+ ('property_account_income_categ','product.category','account.account'),
3396+ ('property_account_expense','product.template','account.account'),
3397+ ('property_account_income','product.template','account.account'),
3398+ ('property_reserve_and_surplus_account','res.company','account.account')
3399+ ]
3400+ template = self.pool.get('account.chart.template').browse(cr, uid, chart_template_id, context=context)
3401+ for record in todo_list:
3402+ account = getattr(template, record[0])
3403+ value = account and 'account.account,' + str(acc_template_ref[account.id]) or False
3404+ if value:
3405+ field = field_obj.search(cr, uid, [('name', '=', record[0]),('model', '=', record[1]),('relation', '=', record[2])], context=context)
3406+ vals = {
3407+ 'name': record[0],
3408+ 'company_id': company_id,
3409+ 'fields_id': field[0],
3410+ 'value': value,
3411+ }
3412+ property_ids = property_obj.search(cr, uid, [('name','=', record[0]),('company_id', '=', company_id)], context=context)
3413+ if property_ids:
3414+ #the property exist: modify it
3415+ property_obj.write(cr, uid, property_ids, vals, context=context)
3416+ else:
3417+ #create the property
3418+ property_obj.create(cr, uid, vals, context=context)
3419+ return True
3420+
3421+ def _install_template(self, cr, uid, template_id, company_id, code_digits=None, obj_wizard=None, acc_ref={}, taxes_ref={}, tax_code_ref={}, context=None):
3422+ '''
3423+ This function recursively loads the template objects and create the real objects from them.
3424+
3425+ :param template_id: id of the chart template to load
3426+ :param company_id: id of the company the wizard is running for
3427+ :param code_digits: integer that depicts the number of digits the accounts code should have in the COA
3428+ :param obj_wizard: the current wizard for generating the COA from the templates
3429+ :param acc_ref: Mapping between ids of account templates and real accounts created from them
3430+ :param taxes_ref: Mapping between ids of tax templates and real taxes created from them
3431+ :param tax_code_ref: Mapping between ids of tax code templates and real tax codes created from them
3432+ :returns: return a tuple with a dictionary containing
3433+ * the mapping between the account template ids and the ids of the real accounts that have been generated
3434+ from them, as first item,
3435+ * a similar dictionary for mapping the tax templates and taxes, as second item,
3436+ * a last identical containing the mapping of tax code templates and tax codes
3437+ :rtype: tuple(dict, dict, dict)
3438+ '''
3439+ template = self.pool.get('account.chart.template').browse(cr, uid, template_id, context=context)
3440+ if template.parent_id:
3441+ tmp1, tmp2, tmp3 = self._install_template(cr, uid, template.parent_id.id, company_id, code_digits=code_digits, acc_ref=acc_ref, taxes_ref=taxes_ref, tax_code_ref=tax_code_ref, context=context)
3442+ acc_ref.update(tmp1)
3443+ taxes_ref.update(tmp2)
3444+ tax_code_ref.update(tmp3)
3445+ tmp1, tmp2, tmp3 = self._load_template(cr, uid, template_id, company_id, code_digits=code_digits, obj_wizard=obj_wizard, account_ref=acc_ref, taxes_ref=taxes_ref, tax_code_ref=tax_code_ref, context=context)
3446+ acc_ref.update(tmp1)
3447+ taxes_ref.update(tmp2)
3448+ tax_code_ref.update(tmp3)
3449+ return acc_ref, taxes_ref, tax_code_ref
3450+
3451+ def _load_template(self, cr, uid, template_id, company_id, code_digits=None, obj_wizard=None, account_ref={}, taxes_ref={}, tax_code_ref={}, context=None):
3452+ '''
3453+ This function generates all the objects from the templates
3454+
3455+ :param template_id: id of the chart template to load
3456+ :param company_id: id of the company the wizard is running for
3457+ :param code_digits: integer that depicts the number of digits the accounts code should have in the COA
3458+ :param obj_wizard: the current wizard for generating the COA from the templates
3459+ :param acc_ref: Mapping between ids of account templates and real accounts created from them
3460+ :param taxes_ref: Mapping between ids of tax templates and real taxes created from them
3461+ :param tax_code_ref: Mapping between ids of tax code templates and real tax codes created from them
3462+ :returns: return a tuple with a dictionary containing
3463+ * the mapping between the account template ids and the ids of the real accounts that have been generated
3464+ from them, as first item,
3465+ * a similar dictionary for mapping the tax templates and taxes, as second item,
3466+ * a last identical containing the mapping of tax code templates and tax codes
3467+ :rtype: tuple(dict, dict, dict)
3468+ '''
3469+ template = self.pool.get('account.chart.template').browse(cr, uid, template_id, context=context)
3470+ obj_tax_code_template = self.pool.get('account.tax.code.template')
3471+ obj_acc_tax = self.pool.get('account.tax')
3472+ obj_tax_temp = self.pool.get('account.tax.template')
3473+ obj_acc_template = self.pool.get('account.account.template')
3474+ obj_fiscal_position_template = self.pool.get('account.fiscal.position.template')
3475+
3476+ # create all the tax code.
3477+ tax_code_ref.update(obj_tax_code_template.generate_tax_code(cr, uid, template.tax_code_root_id.id, company_id, context=context))
3478+
3479+ # Generate taxes from templates.
3480+ tax_templates = [x for x in template.tax_template_ids]
3481+ generated_tax_res = obj_tax_temp._generate_tax(cr, uid, tax_templates, tax_code_ref, company_id, context=context)
3482+ taxes_ref.update(generated_tax_res['tax_template_to_tax'])
3483+
3484+ # Generating Accounts from templates.
3485+ account_template_ref = obj_acc_template.generate_account(cr, uid, template_id, taxes_ref, account_ref, code_digits, company_id, context=context)
3486+ account_ref.update(account_template_ref)
3487+
3488+ # writing account values on tax after creation of accounts
3489+ for key,value in generated_tax_res['account_dict'].items():
3490+ if value['account_collected_id'] or value['account_paid_id']:
3491+ obj_acc_tax.write(cr, uid, [key], {
3492+ 'account_collected_id': account_ref.get(value['account_collected_id'], False),
3493+ 'account_paid_id': account_ref.get(value['account_paid_id'], False),
3494+ })
3495+
3496+ # Create Journals
3497+ self.generate_journals(cr, uid, template_id, account_ref, company_id, context=context)
3498+
3499+ # generate properties function
3500+ self.generate_properties(cr, uid, template_id, account_ref, company_id, context=context)
3501+
3502+ # Generate Fiscal Position , Fiscal Position Accounts and Fiscal Position Taxes from templates
3503+ obj_fiscal_position_template.generate_fiscal_position(cr, uid, template_id, taxes_ref, account_ref, company_id, context=context)
3504+
3505+ return account_ref, taxes_ref, tax_code_ref
3506+
3507+ def _create_tax_templates_from_rates(self, cr, uid, obj_wizard, company_id, context=None):
3508+ '''
3509+ This function checks if the chosen chart template is configured as containing a full set of taxes, and if
3510+ it's not the case, it creates the templates for account.tax.code and for account.account.tax objects accordingly
3511+ to the provided sale/purchase rates. Then it saves the new tax templates as default taxes to use for this chart
3512+ template.
3513+
3514+ :param obj_wizard: browse record of wizard to generate COA from templates
3515+ :param company_id: id of the company for wich the wizard is running
3516+ :return: True
3517+ '''
3518+ obj_tax_code_template = self.pool.get('account.tax.code.template')
3519+ obj_tax_temp = self.pool.get('account.tax.template')
3520+ chart_template = obj_wizard.chart_template_id
3521+ vals = {}
3522+ # get the ids of all the parents of the selected account chart template
3523+ current_chart_template = chart_template
3524+ all_parents = [current_chart_template.id]
3525+ while current_chart_template.parent_id:
3526+ current_chart_template = current_chart_template.parent_id
3527+ all_parents.append(current_chart_template.id)
3528+ # create tax templates and tax code templates from purchase_tax_rate and sale_tax_rate fields
3529+ if not chart_template.complete_tax_set:
3530+ value = obj_wizard.sale_tax_rate
3531+ ref_tax_ids = obj_tax_temp.search(cr, uid, [('type_tax_use','in', ('sale','all')), ('chart_template_id', 'in', all_parents)], context=context, order="sequence, id desc", limit=1)
3532+ obj_tax_temp.write(cr, uid, ref_tax_ids, {'amount': value/100.0, 'name': _('Tax %.2f%%') % value})
3533+ value = obj_wizard.purchase_tax_rate
3534+ ref_tax_ids = obj_tax_temp.search(cr, uid, [('type_tax_use','in', ('purchase','all')), ('chart_template_id', 'in', all_parents)], context=context, order="sequence, id desc", limit=1)
3535+ obj_tax_temp.write(cr, uid, ref_tax_ids, {'amount': value/100.0, 'name': _('Purchase Tax %.2f%%') % value})
3536+ return True
3537+
3538+ def execute(self, cr, uid, ids, context=None):
3539+ '''
3540+ This function is called at the confirmation of the wizard to generate the COA from the templates. It will read
3541+ all the provided information to create the accounts, the banks, the journals, the taxes, the tax codes, the
3542+ accounting properties... accordingly for the chosen company.
3543+ '''
3544+ ir_values_obj = self.pool.get('ir.values')
3545+ obj_wizard = self.browse(cr, uid, ids[0])
3546+ company_id = obj_wizard.company_id.id
3547+ # If the floats for sale/purchase rates have been filled, create templates from them
3548+ self._create_tax_templates_from_rates(cr, uid, obj_wizard, company_id, context=context)
3549+
3550+ # Install all the templates objects and generate the real objects
3551+ acc_template_ref, taxes_ref, tax_code_ref = self._install_template(cr, uid, obj_wizard.chart_template_id.id, company_id, code_digits=obj_wizard.code_digits, obj_wizard=obj_wizard, context=context)
3552+
3553+ # write values of default taxes for product
3554+ if obj_wizard.sale_tax and taxes_ref:
3555+ ir_values_obj.set(cr, 1, key='default', key2=False, name="taxes_id", company=company_id,
3556+ models =[('product.product',False)], value=[taxes_ref[obj_wizard.sale_tax.id]])
3557+ if obj_wizard.purchase_tax and taxes_ref:
3558+ ir_values_obj.set(cr, 1, key='default', key2=False, name="supplier_taxes_id", company=company_id,
3559+ models =[('product.product',False)], value=[taxes_ref[obj_wizard.purchase_tax.id]])
3560+
3561+ # Create Bank journals
3562+ self._create_bank_journals_from_o2m(cr, uid, obj_wizard, company_id, acc_template_ref, context=context)
3563+ action = {
3564+ 'type': 'ir.actions.act_window',
3565+ 'view_type': 'form',
3566+ 'view_mode': 'form',
3567+ 'res_model': 'board.board',
3568+ 'view_id': self.pool.get('ir.model.data').get_object_reference(cr, uid, 'account', 'board_account_form')[1],
3569+ 'menu_id': self.pool.get('ir.model.data').get_object_reference(cr, uid, 'account', 'menu_finance')[1]
3570+ }
3571+ return action
3572+
3573+ def _prepare_bank_journal(self, cr, uid, line, current_num, default_account_id, company_id, context=None):
3574+ '''
3575+ This function prepares the value to use for the creation of a bank journal created through the wizard of
3576+ generating COA from templates.
3577+
3578+ :param line: dictionary containing the values encoded by the user related to his bank account
3579+ :param current_num: integer corresponding to a counter of the already created bank journals through this wizard.
3580+ :param default_account_id: id of the default debit.credit account created before for this journal.
3581+ :param company_id: id of the company for which the wizard is running
3582+ :return: mapping of field names and values
3583+ :rtype: dict
3584+ '''
3585+ obj_data = self.pool.get('ir.model.data')
3586+ obj_journal = self.pool.get('account.journal')
3587+ # Get the id of journal views
3588+ tmp = obj_data.get_object_reference(cr, uid, 'account', 'account_journal_bank_view_multi')
3589+ view_id_cur = tmp and tmp[1] or False
3590+ tmp = obj_data.get_object_reference(cr, uid, 'account', 'account_journal_bank_view')
3591+ view_id_cash = tmp and tmp[1] or False
3592+
3593+ # we need to loop again to find next number for journal code
3594+ # because we can't rely on the value current_num as,
3595+ # its possible that we already have bank journals created (e.g. by the creation of res.partner.bank)
3596+ # and the next number for account code might have been already used before for journal
3597+ for num in xrange(current_num, 100):
3598+ # journal_code has a maximal size of 5, hence we can enforce the boundary num < 100
3599+ journal_code = _('BNK')[:3] + str(num)
3600+ ids = obj_journal.search(cr, uid, [('code', '=', journal_code), ('company_id', '=', company_id)], context=context)
3601+ if not ids:
3602+ break
3603+ else:
3604+ raise osv.except_osv(_('Error'), _('Cannot generate an unused journal code.'))
3605+
3606+ vals = {
3607+ 'name': line['acc_name'],
3608+ 'code': journal_code,
3609+ 'type': line['account_type'] == 'cash' and 'cash' or 'bank',
3610+ 'company_id': company_id,
3611+ 'analytic_journal_id': False,
3612+ 'currency': False,
3613+ 'default_credit_account_id': default_account_id,
3614+ 'default_debit_account_id': default_account_id,
3615+ }
3616+ if line['currency_id']:
3617+ vals['view_id'] = view_id_cur
3618+ vals['currency'] = line['currency_id']
3619+ else:
3620+ vals['view_id'] = view_id_cash
3621+ return vals
3622+
3623+ def _prepare_bank_account(self, cr, uid, line, new_code, acc_template_ref, ref_acc_bank, company_id, context=None):
3624+ '''
3625+ This function prepares the value to use for the creation of the default debit and credit accounts of a
3626+ bank journal created through the wizard of generating COA from templates.
3627+
3628+ :param line: dictionary containing the values encoded by the user related to his bank account
3629+ :param new_code: integer corresponding to the next available number to use as account code
3630+ :param acc_template_ref: the dictionary containing the mapping between the ids of account templates and the ids
3631+ of the accounts that have been generated from them.
3632+ :param ref_acc_bank: browse record of the account template set as root of all bank accounts for the chosen
3633+ template
3634+ :param company_id: id of the company for which the wizard is running
3635+ :return: mapping of field names and values
3636+ :rtype: dict
3637+ '''
3638+ obj_data = self.pool.get('ir.model.data')
3639+
3640+ # Get the id of the user types fr-or cash and bank
3641+ tmp = obj_data.get_object_reference(cr, uid, 'account', 'data_account_type_cash')
3642+ cash_type = tmp and tmp[1] or False
3643+ tmp = obj_data.get_object_reference(cr, uid, 'account', 'data_account_type_bank')
3644+ bank_type = tmp and tmp[1] or False
3645+ return {
3646+ 'name': line['acc_name'],
3647+ 'currency_id': line['currency_id'],
3648+ 'code': new_code,
3649+ 'type': 'liquidity',
3650+ 'user_type': line['account_type'] == 'cash' and cash_type or bank_type,
3651+ 'parent_id': acc_template_ref[ref_acc_bank.id] or False,
3652+ 'company_id': company_id,
3653+ }
3654+
3655+ def _create_bank_journals_from_o2m(self, cr, uid, obj_wizard, company_id, acc_template_ref, context=None):
3656+ '''
3657+ This function creates bank journals and its accounts for each line encoded in the field bank_accounts_id of the
3658+ wizard.
3659+
3660+ :param obj_wizard: the current wizard that generates the COA from the templates.
3661+ :param company_id: the id of the company for which the wizard is running.
3662+ :param acc_template_ref: the dictionary containing the mapping between the ids of account templates and the ids
3663+ of the accounts that have been generated from them.
3664+ :return: True
3665+ '''
3666+ obj_acc = self.pool.get('account.account')
3667+ obj_journal = self.pool.get('account.journal')
3668+ code_digits = obj_wizard.code_digits
3669+
3670+ # Build a list with all the data to process
3671+ journal_data = []
3672+ if obj_wizard.bank_accounts_id:
3673+ for acc in obj_wizard.bank_accounts_id:
3674+ vals = {
3675+ 'acc_name': acc.acc_name,
3676+ 'account_type': acc.account_type,
3677+ 'currency_id': acc.currency_id.id,
3678+ }
3679+ journal_data.append(vals)
3680+ ref_acc_bank = obj_wizard.chart_template_id.bank_account_view_id
3681+ if journal_data and not ref_acc_bank.code:
3682+ raise osv.except_osv(_('Configuration Error !'), _('The bank account defined on the selected chart of accounts hasn\'t a code.'))
3683+
3684+ current_num = 1
3685+ for line in journal_data:
3686+ # Seek the next available number for the account code
3687+ while True:
3688+ new_code = str(ref_acc_bank.code.ljust(code_digits-len(str(current_num)), '0')) + str(current_num)
3689+ ids = obj_acc.search(cr, uid, [('code', '=', new_code), ('company_id', '=', company_id)])
3690+ if not ids:
3691+ break
3692+ else:
3693+ current_num += 1
3694+ # Create the default debit/credit accounts for this bank journal
3695+ vals = self._prepare_bank_account(cr, uid, line, new_code, acc_template_ref, ref_acc_bank, company_id, context=context)
3696+ default_account_id = obj_acc.create(cr, uid, vals, context=context)
3697+
3698+ #create the bank journal
3699+ vals_journal = self._prepare_bank_journal(cr, uid, line, current_num, default_account_id, company_id, context=context)
3700+ obj_journal.create(cr, uid, vals_journal)
3701+ current_num += 1
3702+ return True
3703+
3704+wizard_multi_charts_accounts()
3705+
3706+class account_bank_accounts_wizard(osv.osv_memory):
3707+ _name='account.bank.accounts.wizard'
3708+
3709+ _columns = {
3710+ 'acc_name': fields.char('Account Name.', size=64, required=True),
3711+ 'bank_account_id': fields.many2one('wizard.multi.charts.accounts', 'Bank Account', required=True),
3712+ 'currency_id': fields.many2one('res.currency', 'Secondary Currency', help="Forces all moves for this account to have this secondary currency."),
3713+ 'account_type': fields.selection([('cash','Cash'), ('check','Check'), ('bank','Bank')], 'Account Type', size=32),
3714+ }
3715+
3716+account_bank_accounts_wizard()
3717+
3718+# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
3719
3720=== added file 'account/account_analytic_line.py'
3721--- account/account_analytic_line.py 1970-01-01 00:00:00 +0000
3722+++ account/account_analytic_line.py 2012-10-04 09:12:55 +0000
3723@@ -0,0 +1,160 @@
3724+# -*- coding: utf-8 -*-
3725+##############################################################################
3726+#
3727+# OpenERP, Open Source Management Solution
3728+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
3729+#
3730+# This program is free software: you can redistribute it and/or modify
3731+# it under the terms of the GNU Affero General Public License as
3732+# published by the Free Software Foundation, either version 3 of the
3733+# License, or (at your option) any later version.
3734+#
3735+# This program is distributed in the hope that it will be useful,
3736+# but WITHOUT ANY WARRANTY; without even the implied warranty of
3737+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
3738+# GNU Affero General Public License for more details.
3739+#
3740+# You should have received a copy of the GNU Affero General Public License
3741+# along with this program. If not, see <http://www.gnu.org/licenses/>.
3742+#
3743+##############################################################################
3744+
3745+from osv import fields
3746+from osv import osv
3747+from tools.translate import _
3748+
3749+class account_analytic_line(osv.osv):
3750+ _inherit = 'account.analytic.line'
3751+ _description = 'Analytic Line'
3752+ _columns = {
3753+ 'product_uom_id': fields.many2one('product.uom', 'UoM'),
3754+ 'product_id': fields.many2one('product.product', 'Product'),
3755+ 'general_account_id': fields.many2one('account.account', 'General Account', required=True, ondelete='restrict'),
3756+ 'move_id': fields.many2one('account.move.line', 'Move Line', ondelete='cascade', select=True),
3757+ 'journal_id': fields.many2one('account.analytic.journal', 'Analytic Journal', required=True, ondelete='restrict', select=True),
3758+ 'code': fields.char('Code', size=8),
3759+ 'ref': fields.char('Ref.', size=64),
3760+ 'currency_id': fields.related('move_id', 'currency_id', type='many2one', relation='res.currency', string='Account currency', store=True, help="The related account currency if not equal to the company one.", readonly=True),
3761+ 'amount_currency': fields.related('move_id', 'amount_currency', type='float', string='Amount currency', store=True, help="The amount expressed in the related account currency if not equal to the company one.", readonly=True),
3762+ }
3763+
3764+ _defaults = {
3765+ 'date': fields.date.context_today,
3766+ 'company_id': lambda self,cr,uid,c: self.pool.get('res.company')._company_default_get(cr, uid, 'account.analytic.line', context=c),
3767+ }
3768+ _order = 'date desc'
3769+
3770+ def search(self, cr, uid, args, offset=0, limit=None, order=None, context=None, count=False):
3771+ if context is None:
3772+ context = {}
3773+ if context.get('from_date',False):
3774+ args.append(['date', '>=', context['from_date']])
3775+ if context.get('to_date',False):
3776+ args.append(['date','<=', context['to_date']])
3777+ return super(account_analytic_line, self).search(cr, uid, args, offset, limit,
3778+ order, context=context, count=count)
3779+
3780+ def _check_company(self, cr, uid, ids, context=None):
3781+ lines = self.browse(cr, uid, ids, context=context)
3782+ for l in lines:
3783+ if l.move_id and not l.account_id.company_id.id == l.move_id.account_id.company_id.id:
3784+ return False
3785+ return True
3786+
3787+ # Compute the cost based on the price type define into company
3788+ # property_valuation_price_type property
3789+ def on_change_unit_amount(self, cr, uid, id, prod_id, quantity, company_id,
3790+ unit=False, journal_id=False, context=None):
3791+ if context==None:
3792+ context={}
3793+ if not journal_id:
3794+ j_ids = self.pool.get('account.analytic.journal').search(cr, uid, [('type','=','purchase')])
3795+ journal_id = j_ids and j_ids[0] or False
3796+ if not journal_id or not prod_id:
3797+ return {}
3798+ product_obj = self.pool.get('product.product')
3799+ analytic_journal_obj =self.pool.get('account.analytic.journal')
3800+ product_price_type_obj = self.pool.get('product.price.type')
3801+ j_id = analytic_journal_obj.browse(cr, uid, journal_id, context=context)
3802+ prod = product_obj.browse(cr, uid, prod_id, context=context)
3803+ result = 0.0
3804+ if prod_id:
3805+ unit = prod.uom_id.id
3806+ if j_id.type == 'purchase':
3807+ unit = prod.uom_po_id.id
3808+ if j_id.type <> 'sale':
3809+ a = prod.product_tmpl_id.property_account_expense.id
3810+ if not a:
3811+ a = prod.categ_id.property_account_expense_categ.id
3812+ if not a:
3813+ raise osv.except_osv(_('Error !'),
3814+ _('There is no expense account defined ' \
3815+ 'for this product: "%s" (id:%d)') % \
3816+ (prod.name, prod.id,))
3817+ else:
3818+ a = prod.product_tmpl_id.property_account_income.id
3819+ if not a:
3820+ a = prod.categ_id.property_account_income_categ.id
3821+ if not a:
3822+ raise osv.except_osv(_('Error !'),
3823+ _('There is no income account defined ' \
3824+ 'for this product: "%s" (id:%d)') % \
3825+ (prod.name, prod_id,))
3826+
3827+ flag = False
3828+ # Compute based on pricetype
3829+ product_price_type_ids = product_price_type_obj.search(cr, uid, [('field','=','standard_price')], context=context)
3830+ pricetype = product_price_type_obj.browse(cr, uid, product_price_type_ids, context=context)[0]
3831+ if journal_id:
3832+ journal = analytic_journal_obj.browse(cr, uid, journal_id, context=context)
3833+ if journal.type == 'sale':
3834+ product_price_type_ids = product_price_type_obj.search(cr, uid, [('field','=','list_price')], context)
3835+ if product_price_type_ids:
3836+ pricetype = product_price_type_obj.browse(cr, uid, product_price_type_ids, context=context)[0]
3837+ # Take the company currency as the reference one
3838+ if pricetype.field == 'list_price':
3839+ flag = True
3840+ ctx = context.copy()
3841+ if unit:
3842+ # price_get() will respect a 'uom' in its context, in order
3843+ # to return a default price for those units
3844+ ctx['uom'] = unit
3845+ amount_unit = prod.price_get(pricetype.field, context=ctx)[prod.id]
3846+ prec = self.pool.get('decimal.precision').precision_get(cr, uid, 'Account')
3847+ amount = amount_unit * quantity or 0.0
3848+ result = round(amount, prec)
3849+ if not flag:
3850+ result *= -1
3851+ return {'value': {
3852+ 'amount': result,
3853+ 'general_account_id': a,
3854+ 'product_uom_id': unit
3855+ }
3856+ }
3857+
3858+ def view_header_get(self, cr, user, view_id, view_type, context=None):
3859+ if context is None:
3860+ context = {}
3861+ if context.get('account_id', False):
3862+ # account_id in context may also be pointing to an account.account.id
3863+ cr.execute('select name from account_analytic_account where id=%s', (context['account_id'],))
3864+ res = cr.fetchone()
3865+ if res:
3866+ res = _('Entries: ')+ (res[0] or '')
3867+ return res
3868+ return False
3869+
3870+account_analytic_line()
3871+
3872+class res_partner(osv.osv):
3873+ """ Inherits partner and adds contract information in the partner form """
3874+ _inherit = 'res.partner'
3875+
3876+ _columns = {
3877+ 'contract_ids': fields.one2many('account.analytic.account', \
3878+ 'partner_id', 'Contracts', readonly=True),
3879+ }
3880+
3881+res_partner()
3882+
3883+# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
3884
3885=== added file 'account/account_assert_test.xml'
3886--- account/account_assert_test.xml 1970-01-01 00:00:00 +0000
3887+++ account/account_assert_test.xml 2012-10-04 09:12:55 +0000
3888@@ -0,0 +1,8 @@
3889+<?xml version="1.0" encoding="utf-8"?>
3890+<openerp>
3891+ <data>
3892+ <assert model="account.move" search="[]" string="For all Journal Items, the state is valid implies that the sum of credits equals the sum of debits">
3893+ <test expr="not len(line_id) or line_id[0].state != 'valid' or (sum([l.debit - l.credit for l in line_id]) &lt;= 0.00001)"/>
3894+ </assert>
3895+ </data>
3896+</openerp>
3897
3898=== added file 'account/account_bank.py'
3899--- account/account_bank.py 1970-01-01 00:00:00 +0000
3900+++ account/account_bank.py 2012-10-04 09:12:55 +0000
3901@@ -0,0 +1,112 @@
3902+# -*- coding: utf-8 -*-
3903+##############################################################################
3904+#
3905+# OpenERP, Open Source Management Solution
3906+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
3907+#
3908+# This program is free software: you can redistribute it and/or modify
3909+# it under the terms of the GNU Affero General Public License as
3910+# published by the Free Software Foundation, either version 3 of the
3911+# License, or (at your option) any later version.
3912+#
3913+# This program is distributed in the hope that it will be useful,
3914+# but WITHOUT ANY WARRANTY; without even the implied warranty of
3915+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
3916+# GNU Affero General Public License for more details.
3917+#
3918+# You should have received a copy of the GNU Affero General Public License
3919+# along with this program. If not, see <http://www.gnu.org/licenses/>.
3920+#
3921+##############################################################################
3922+
3923+from tools.translate import _
3924+from osv import fields, osv
3925+
3926+class bank(osv.osv):
3927+ _inherit = "res.partner.bank"
3928+ _columns = {
3929+ 'journal_id': fields.many2one('account.journal', 'Account Journal', help="This journal will be created automatically for this bank account when you save the record"),
3930+ 'currency_id': fields.related('journal_id', 'currency', type="many2one", relation='res.currency', readonly=True,
3931+ string="Currency", help="Currency of the related account journal."),
3932+ }
3933+ def create(self, cr, uid, data, context={}):
3934+ result = super(bank, self).create(cr, uid, data, context=context)
3935+ self.post_write(cr, uid, [result], context=context)
3936+ return result
3937+
3938+ def write(self, cr, uid, ids, data, context={}):
3939+ result = super(bank, self).write(cr, uid, ids, data, context=context)
3940+ self.post_write(cr, uid, ids, context=context)
3941+ return result
3942+
3943+ def _prepare_name(self, bank):
3944+ "Return the name to use when creating a bank journal"
3945+ return (bank.bank_name or '') + ' ' + bank.acc_number
3946+
3947+ def post_write(self, cr, uid, ids, context={}):
3948+ if isinstance(ids, (int, long)):
3949+ ids = [ids]
3950+
3951+ obj_acc = self.pool.get('account.account')
3952+ obj_data = self.pool.get('ir.model.data')
3953+
3954+ for bank in self.browse(cr, uid, ids, context):
3955+ if bank.company_id and not bank.journal_id:
3956+ # Find the code and parent of the bank account to create
3957+ dig = 6
3958+ current_num = 1
3959+ ids = obj_acc.search(cr, uid, [('type','=','liquidity'), ('company_id', '=', bank.company_id.id)], context=context)
3960+ # No liquidity account exists, no template available
3961+ if not ids: continue
3962+
3963+ ref_acc_bank_temp = obj_acc.browse(cr, uid, ids[0], context=context)
3964+ ref_acc_bank = ref_acc_bank_temp.parent_id
3965+ while True:
3966+ new_code = str(ref_acc_bank.code.ljust(dig-len(str(current_num)), '0')) + str(current_num)
3967+ ids = obj_acc.search(cr, uid, [('code', '=', new_code), ('company_id', '=', bank.company_id.id)])
3968+ if not ids:
3969+ break
3970+ current_num += 1
3971+ name = self._prepare_name(bank)
3972+ acc = {
3973+ 'name': name,
3974+ 'code': new_code,
3975+ 'type': 'liquidity',
3976+ 'user_type': ref_acc_bank_temp.user_type.id,
3977+ 'reconcile': False,
3978+ 'parent_id': ref_acc_bank.id,
3979+ 'company_id': bank.company_id.id,
3980+ }
3981+ acc_bank_id = obj_acc.create(cr,uid,acc,context=context)
3982+
3983+ # Get the journal view id
3984+ data_id = obj_data.search(cr, uid, [('model','=','account.journal.view'), ('name','=','account_journal_bank_view')])
3985+ data = obj_data.browse(cr, uid, data_id[0], context=context)
3986+ view_id_cash = data.res_id
3987+
3988+ jour_obj = self.pool.get('account.journal')
3989+ new_code = 1
3990+ while True:
3991+ code = _('BNK')+str(new_code)
3992+ ids = jour_obj.search(cr, uid, [('code','=',code)], context=context)
3993+ if not ids:
3994+ break
3995+ new_code += 1
3996+
3997+ #create the bank journal
3998+ vals_journal = {
3999+ 'name': name,
4000+ 'code': code,
4001+ 'type': 'bank',
4002+ 'company_id': bank.company_id.id,
4003+ 'analytic_journal_id': False,
4004+ 'default_credit_account_id': acc_bank_id,
4005+ 'default_debit_account_id': acc_bank_id,
4006+ 'view_id': view_id_cash
4007+ }
4008+ journal_id = jour_obj.create(cr, uid, vals_journal, context=context)
4009+
4010+ self.write(cr, uid, [bank.id], {'journal_id': journal_id}, context=context)
4011+ return True
4012+
4013+# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
4014
4015=== added file 'account/account_bank_statement.py'
4016--- account/account_bank_statement.py 1970-01-01 00:00:00 +0000
4017+++ account/account_bank_statement.py 2012-10-04 09:12:55 +0000
4018@@ -0,0 +1,493 @@
4019+# -*- coding: utf-8 -*-
4020+##############################################################################
4021+#
4022+# OpenERP, Open Source Management Solution
4023+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
4024+#
4025+# This program is free software: you can redistribute it and/or modify
4026+# it under the terms of the GNU Affero General Public License as
4027+# published by the Free Software Foundation, either version 3 of the
4028+# License, or (at your option) any later version.
4029+#
4030+# This program is distributed in the hope that it will be useful,
4031+# but WITHOUT ANY WARRANTY; without even the implied warranty of
4032+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
4033+# GNU Affero General Public License for more details.
4034+#
4035+# You should have received a copy of the GNU Affero General Public License
4036+# along with this program. If not, see <http://www.gnu.org/licenses/>.
4037+#
4038+##############################################################################
4039+
4040+import time
4041+
4042+from osv import fields, osv
4043+from tools.translate import _
4044+import decimal_precision as dp
4045+
4046+class account_bank_statement(osv.osv):
4047+
4048+ def create(self, cr, uid, vals, context=None):
4049+ seq = 0
4050+ if 'line_ids' in vals:
4051+ new_line_ids = []
4052+ for line in vals['line_ids']:
4053+ seq += 1
4054+ line[2]['sequence'] = seq
4055+ return super(account_bank_statement, self).create(cr, uid, vals, context=context)
4056+
4057+ def write(self, cr, uid, ids, vals, context=None):
4058+ res = super(account_bank_statement, self).write(cr, uid, ids, vals, context=context)
4059+ account_bank_statement_line_obj = self.pool.get('account.bank.statement.line')
4060+ for statement in self.browse(cr, uid, ids, context):
4061+ seq = 0
4062+ for line in statement.line_ids:
4063+ seq += 1
4064+ account_bank_statement_line_obj.write(cr, uid, [line.id], {'sequence': seq}, context=context)
4065+ return res
4066+
4067+ def _default_journal_id(self, cr, uid, context=None):
4068+ if context is None:
4069+ context = {}
4070+ journal_pool = self.pool.get('account.journal')
4071+ journal_type = context.get('journal_type', False)
4072+ journal_id = False
4073+ company_id = self.pool.get('res.company')._company_default_get(cr, uid, 'account.bank.statement',context=context)
4074+ if journal_type:
4075+ ids = journal_pool.search(cr, uid, [('type', '=', journal_type),('company_id','=',company_id)])
4076+ if ids:
4077+ journal_id = ids[0]
4078+ return journal_id
4079+
4080+ def _end_balance(self, cursor, user, ids, name, attr, context=None):
4081+ res_currency_obj = self.pool.get('res.currency')
4082+ res_users_obj = self.pool.get('res.users')
4083+ res = {}
4084+
4085+ company_currency_id = res_users_obj.browse(cursor, user, user,
4086+ context=context).company_id.currency_id.id
4087+
4088+ statements = self.browse(cursor, user, ids, context=context)
4089+ for statement in statements:
4090+ res[statement.id] = statement.balance_start
4091+ currency_id = statement.currency.id
4092+ for line in statement.move_line_ids:
4093+ if line.debit > 0:
4094+ if line.account_id.id == \
4095+ statement.journal_id.default_debit_account_id.id:
4096+ res[statement.id] += res_currency_obj.compute(cursor,
4097+ user, company_currency_id, currency_id,
4098+ line.debit, context=context)
4099+ else:
4100+ if line.account_id.id == \
4101+ statement.journal_id.default_credit_account_id.id:
4102+ res[statement.id] -= res_currency_obj.compute(cursor,
4103+ user, company_currency_id, currency_id,
4104+ line.credit, context=context)
4105+
4106+ if statement.state in ('draft', 'open'):
4107+ for line in statement.line_ids:
4108+ res[statement.id] += line.amount
4109+ for r in res:
4110+ res[r] = round(res[r], 2)
4111+ return res
4112+
4113+ def _get_period(self, cr, uid, context=None):
4114+ periods = self.pool.get('account.period').find(cr, uid)
4115+ if periods:
4116+ return periods[0]
4117+ return False
4118+
4119+ def _currency(self, cursor, user, ids, name, args, context=None):
4120+ res = {}
4121+ res_currency_obj = self.pool.get('res.currency')
4122+ res_users_obj = self.pool.get('res.users')
4123+ default_currency = res_users_obj.browse(cursor, user,
4124+ user, context=context).company_id.currency_id
4125+ for statement in self.browse(cursor, user, ids, context=context):
4126+ currency = statement.journal_id.currency
4127+ if not currency:
4128+ currency = default_currency
4129+ res[statement.id] = currency.id
4130+ currency_names = {}
4131+ for currency_id, currency_name in res_currency_obj.name_get(cursor,
4132+ user, [x for x in res.values()], context=context):
4133+ currency_names[currency_id] = currency_name
4134+ for statement_id in res.keys():
4135+ currency_id = res[statement_id]
4136+ res[statement_id] = (currency_id, currency_names[currency_id])
4137+ return res
4138+
4139+ def _get_statement(self, cr, uid, ids, context=None):
4140+ result = {}
4141+ for line in self.pool.get('account.bank.statement.line').browse(cr, uid, ids, context=context):
4142+ result[line.statement_id.id] = True
4143+ return result.keys()
4144+
4145+ _order = "date desc, id desc"
4146+ _name = "account.bank.statement"
4147+ _description = "Bank Statement"
4148+ _columns = {
4149+ 'name': fields.char('Name', size=64, required=True, states={'draft': [('readonly', False)]}, readonly=True, help='if you give the Name other then /, its created Accounting Entries Move will be with same name as statement name. This allows the statement entries to have the same references than the statement itself'), # readonly for account_cash_statement
4150+ 'date': fields.date('Date', required=True, states={'confirm': [('readonly', True)]}, select=True),
4151+ 'journal_id': fields.many2one('account.journal', 'Journal', required=True,
4152+ readonly=True, states={'draft':[('readonly',False)]}),
4153+ 'period_id': fields.many2one('account.period', 'Period', required=True,
4154+ states={'confirm':[('readonly', True)]}),
4155+ 'balance_start': fields.float('Starting Balance', digits_compute=dp.get_precision('Account'),
4156+ states={'confirm':[('readonly',True)]}),
4157+ 'balance_end_real': fields.float('Ending Balance', digits_compute=dp.get_precision('Account'),
4158+ states={'confirm': [('readonly', True)]}),
4159+ 'balance_end': fields.function(_end_balance,
4160+ store = {
4161+ 'account.bank.statement': (lambda self, cr, uid, ids, c={}: ids, ['line_ids','move_line_ids'], 10),
4162+ 'account.bank.statement.line': (_get_statement, ['amount'], 10),
4163+ },
4164+ string="Computed Balance", help='Balance as calculated based on Starting Balance and transaction lines'),
4165+ 'company_id': fields.related('journal_id', 'company_id', type='many2one', relation='res.company', string='Company', store=True, readonly=True),
4166+ 'line_ids': fields.one2many('account.bank.statement.line',
4167+ 'statement_id', 'Statement lines',
4168+ states={'confirm':[('readonly', True)]}),
4169+ 'move_line_ids': fields.one2many('account.move.line', 'statement_id',
4170+ 'Entry lines', states={'confirm':[('readonly',True)]}),
4171+ 'state': fields.selection([('draft', 'New'),
4172+ ('open','Open'), # used by cash statements
4173+ ('confirm', 'Closed')],
4174+ 'State', required=True, readonly="1",
4175+ help='When new statement is created the state will be \'Draft\'.\n'
4176+ 'And after getting confirmation from the bank it will be in \'Confirmed\' state.'),
4177+ 'currency': fields.function(_currency, string='Currency',
4178+ type='many2one', relation='res.currency'),
4179+ 'account_id': fields.related('journal_id', 'default_debit_account_id', type='many2one', relation='account.account', string='Account used in this journal', readonly=True, help='used in statement reconciliation domain, but shouldn\'t be used elswhere.'),
4180+ }
4181+
4182+ _defaults = {
4183+ 'name': "/",
4184+ 'date': fields.date.context_today,
4185+ 'state': 'draft',
4186+ 'journal_id': _default_journal_id,
4187+ 'period_id': _get_period,
4188+ 'company_id': lambda self,cr,uid,c: self.pool.get('res.company')._company_default_get(cr, uid, 'account.bank.statement',context=c),
4189+ }
4190+
4191+ def _check_company_id(self, cr, uid, ids, context=None):
4192+ for statement in self.browse(cr, uid, ids, context=context):
4193+ if statement.company_id.id != statement.period_id.company_id.id:
4194+ return False
4195+ return True
4196+
4197+ _constraints = [
4198+ (_check_company_id, 'The journal and period chosen have to belong to the same company.', ['journal_id','period_id']),
4199+ ]
4200+
4201+ def onchange_date(self, cr, uid, ids, date, company_id, context=None):
4202+ """
4203+ Find the correct period to use for the given date and company_id, return it and set it in the context
4204+ """
4205+ res = {}
4206+ period_pool = self.pool.get('account.period')
4207+
4208+ if context is None:
4209+ context = {}
4210+ ctx = context.copy()
4211+ ctx.update({'company_id': company_id})
4212+ pids = period_pool.find(cr, uid, dt=date, context=ctx)
4213+ if pids:
4214+ res.update({'period_id': pids[0]})
4215+ context.update({'period_id': pids[0]})
4216+
4217+ return {
4218+ 'value':res,
4219+ 'context':context,
4220+ }
4221+
4222+ def button_dummy(self, cr, uid, ids, context=None):
4223+ return self.write(cr, uid, ids, {}, context=context)
4224+
4225+ def create_move_from_st_line(self, cr, uid, st_line_id, company_currency_id, st_line_number, context=None):
4226+ if context is None:
4227+ context = {}
4228+ res_currency_obj = self.pool.get('res.currency')
4229+ account_move_obj = self.pool.get('account.move')
4230+ account_move_line_obj = self.pool.get('account.move.line')
4231+ account_bank_statement_line_obj = self.pool.get('account.bank.statement.line')
4232+ st_line = account_bank_statement_line_obj.browse(cr, uid, st_line_id, context=context)
4233+ st = st_line.statement_id
4234+
4235+ context.update({'date': st_line.date})
4236+
4237+ move_id = account_move_obj.create(cr, uid, {
4238+ 'journal_id': st.journal_id.id,
4239+ 'period_id': st.period_id.id,
4240+ 'date': st_line.date,
4241+ 'name': st_line_number,
4242+ 'ref': st_line.ref,
4243+ }, context=context)
4244+ account_bank_statement_line_obj.write(cr, uid, [st_line.id], {
4245+ 'move_ids': [(4, move_id, False)]
4246+ })
4247+
4248+ torec = []
4249+ if st_line.amount >= 0:
4250+ account_id = st.journal_id.default_credit_account_id.id
4251+ else:
4252+ account_id = st.journal_id.default_debit_account_id.id
4253+
4254+ acc_cur = ((st_line.amount<=0) and st.journal_id.default_debit_account_id) or st_line.account_id
4255+ context.update({
4256+ 'res.currency.compute.account': acc_cur,
4257+ })
4258+ amount = res_currency_obj.compute(cr, uid, st.currency.id,
4259+ company_currency_id, st_line.amount, context=context)
4260+
4261+ val = {
4262+ 'name': st_line.name,
4263+ 'date': st_line.date,
4264+ 'ref': st_line.ref,
4265+ 'move_id': move_id,
4266+ 'partner_id': ((st_line.partner_id) and st_line.partner_id.id) or False,
4267+ 'account_id': (st_line.account_id) and st_line.account_id.id,
4268+ 'credit': ((amount>0) and amount) or 0.0,
4269+ 'debit': ((amount<0) and -amount) or 0.0,
4270+ 'statement_id': st.id,
4271+ 'journal_id': st.journal_id.id,
4272+ 'period_id': st.period_id.id,
4273+ 'currency_id': st.currency.id,
4274+ 'analytic_account_id': st_line.analytic_account_id and st_line.analytic_account_id.id or False
4275+ }
4276+
4277+ if st.currency.id <> company_currency_id:
4278+ amount_cur = res_currency_obj.compute(cr, uid, company_currency_id,
4279+ st.currency.id, amount, context=context)
4280+ val['amount_currency'] = -amount_cur
4281+
4282+ if st_line.account_id and st_line.account_id.currency_id and st_line.account_id.currency_id.id <> company_currency_id:
4283+ val['currency_id'] = st_line.account_id.currency_id.id
4284+ amount_cur = res_currency_obj.compute(cr, uid, company_currency_id,
4285+ st_line.account_id.currency_id.id, amount, context=context)
4286+ val['amount_currency'] = -amount_cur
4287+
4288+ move_line_id = account_move_line_obj.create(cr, uid, val, context=context)
4289+ torec.append(move_line_id)
4290+
4291+ # Fill the secondary amount/currency
4292+ # if currency is not the same than the company
4293+ amount_currency = False
4294+ currency_id = False
4295+ if st.currency.id <> company_currency_id:
4296+ amount_currency = st_line.amount
4297+ currency_id = st.currency.id
4298+ account_move_line_obj.create(cr, uid, {
4299+ 'name': st_line.name,
4300+ 'date': st_line.date,
4301+ 'ref': st_line.ref,
4302+ 'move_id': move_id,
4303+ 'partner_id': ((st_line.partner_id) and st_line.partner_id.id) or False,
4304+ 'account_id': account_id,
4305+ 'credit': ((amount < 0) and -amount) or 0.0,
4306+ 'debit': ((amount > 0) and amount) or 0.0,
4307+ 'statement_id': st.id,
4308+ 'journal_id': st.journal_id.id,
4309+ 'period_id': st.period_id.id,
4310+ 'amount_currency': amount_currency,
4311+ 'currency_id': currency_id,
4312+ }, context=context)
4313+
4314+ for line in account_move_line_obj.browse(cr, uid, [x.id for x in
4315+ account_move_obj.browse(cr, uid, move_id,
4316+ context=context).line_id],
4317+ context=context):
4318+ if line.state <> 'valid':
4319+ raise osv.except_osv(_('Error !'),
4320+ _('Journal item "%s" is not valid.') % line.name)
4321+
4322+ # Bank statements will not consider boolean on journal entry_posted
4323+ account_move_obj.post(cr, uid, [move_id], context=context)
4324+ return move_id
4325+
4326+ def get_next_st_line_number(self, cr, uid, st_number, st_line, context=None):
4327+ return st_number + '/' + str(st_line.sequence)
4328+
4329+ def balance_check(self, cr, uid, st_id, journal_type='bank', context=None):
4330+ st = self.browse(cr, uid, st_id, context=context)
4331+ if not ((abs((st.balance_end or 0.0) - st.balance_end_real) < 0.0001) or (abs((st.balance_end or 0.0) - st.balance_end_cash) < 0.0001)):
4332+ raise osv.except_osv(_('Error !'),
4333+ _('The statement balance is incorrect !\nThe expected balance (%.2f) is different than the computed one. (%.2f)') % (st.balance_end_real, st.balance_end))
4334+ return True
4335+
4336+ def statement_close(self, cr, uid, ids, journal_type='bank', context=None):
4337+ return self.write(cr, uid, ids, {'state':'confirm'}, context=context)
4338+
4339+ def check_status_condition(self, cr, uid, state, journal_type='bank'):
4340+ return state in ('draft','open')
4341+
4342+ def button_confirm_bank(self, cr, uid, ids, context=None):
4343+ obj_seq = self.pool.get('ir.sequence')
4344+ if context is None:
4345+ context = {}
4346+
4347+ for st in self.browse(cr, uid, ids, context=context):
4348+ j_type = st.journal_id.type
4349+ company_currency_id = st.journal_id.company_id.currency_id.id
4350+ if not self.check_status_condition(cr, uid, st.state, journal_type=j_type):
4351+ continue
4352+
4353+ self.balance_check(cr, uid, st.id, journal_type=j_type, context=context)
4354+ if (not st.journal_id.default_credit_account_id) \
4355+ or (not st.journal_id.default_debit_account_id):
4356+ raise osv.except_osv(_('Configuration Error !'),
4357+ _('Please verify that an account is defined in the journal.'))
4358+
4359+ if not st.name == '/':
4360+ st_number = st.name
4361+ else:
4362+ c = {'fiscalyear_id': st.period_id.fiscalyear_id.id}
4363+ if st.journal_id.sequence_id:
4364+ st_number = obj_seq.next_by_id(cr, uid, st.journal_id.sequence_id.id, context=c)
4365+ else:
4366+ st_number = obj_seq.next_by_code(cr, uid, 'account.bank.statement', context=c)
4367+
4368+ for line in st.move_line_ids:
4369+ if line.state <> 'valid':
4370+ raise osv.except_osv(_('Error !'),
4371+ _('The account entries lines are not in valid state.'))
4372+ for st_line in st.line_ids:
4373+ if st_line.analytic_account_id:
4374+ if not st.journal_id.analytic_journal_id:
4375+ raise osv.except_osv(_('No Analytic Journal !'),_("You have to assign an analytic journal on the '%s' journal!") % (st.journal_id.name,))
4376+ if not st_line.amount:
4377+ continue
4378+ st_line_number = self.get_next_st_line_number(cr, uid, st_number, st_line, context)
4379+ self.create_move_from_st_line(cr, uid, st_line.id, company_currency_id, st_line_number, context)
4380+
4381+ self.write(cr, uid, [st.id], {
4382+ 'name': st_number,
4383+ 'balance_end_real': st.balance_end
4384+ }, context=context)
4385+ self.log(cr, uid, st.id, _('Statement %s is confirmed, journal items are created.') % (st_number,))
4386+ return self.write(cr, uid, ids, {'state':'confirm'}, context=context)
4387+
4388+ def button_cancel(self, cr, uid, ids, context=None):
4389+ done = []
4390+ account_move_obj = self.pool.get('account.move')
4391+ for st in self.browse(cr, uid, ids, context=context):
4392+ if st.state=='draft':
4393+ continue
4394+ move_ids = []
4395+ for line in st.line_ids:
4396+ move_ids += [x.id for x in line.move_ids]
4397+ account_move_obj.button_cancel(cr, uid, move_ids, context=context)
4398+ account_move_obj.unlink(cr, uid, move_ids, context)
4399+ done.append(st.id)
4400+ return self.write(cr, uid, done, {'state':'draft'}, context=context)
4401+
4402+ def onchange_journal_id(self, cr, uid, statement_id, journal_id, context=None):
4403+ cr.execute('SELECT balance_end_real \
4404+ FROM account_bank_statement \
4405+ WHERE journal_id = %s AND NOT state = %s \
4406+ ORDER BY date DESC,id DESC LIMIT 1', (journal_id, 'draft'))
4407+ res = cr.fetchone()
4408+ balance_start = res and res[0] or 0.0
4409+ journal_data = self.pool.get('account.journal').read(cr, uid, journal_id, ['default_debit_account_id', 'company_id'], context=context)
4410+ account_id = journal_data['default_debit_account_id']
4411+ company_id = journal_data['company_id']
4412+ return {'value': {'balance_start': balance_start, 'account_id': account_id, 'company_id': company_id}}
4413+
4414+ def unlink(self, cr, uid, ids, context=None):
4415+ stat = self.read(cr, uid, ids, ['state'], context=context)
4416+ unlink_ids = []
4417+ for t in stat:
4418+ if t['state'] in ('draft'):
4419+ unlink_ids.append(t['id'])
4420+ else:
4421+ raise osv.except_osv(_('Invalid action !'), _('In order to delete a bank statement, you must first cancel it to delete related journal items.'))
4422+ osv.osv.unlink(self, cr, uid, unlink_ids, context=context)
4423+ return True
4424+
4425+ def copy(self, cr, uid, id, default=None, context=None):
4426+ if default is None:
4427+ default = {}
4428+ if context is None:
4429+ context = {}
4430+ default = default.copy()
4431+ default['move_line_ids'] = []
4432+ return super(account_bank_statement, self).copy(cr, uid, id, default, context=context)
4433+
4434+account_bank_statement()
4435+
4436+class account_bank_statement_line(osv.osv):
4437+
4438+ def onchange_partner_id(self, cr, uid, ids, partner_id, context=None):
4439+ obj_partner = self.pool.get('res.partner')
4440+ if context is None:
4441+ context = {}
4442+ if not partner_id:
4443+ return {}
4444+ part = obj_partner.browse(cr, uid, partner_id, context=context)
4445+ if not part.supplier and not part.customer:
4446+ type = 'general'
4447+ elif part.supplier and part.customer:
4448+ type = 'general'
4449+ else:
4450+ if part.supplier == True:
4451+ type = 'supplier'
4452+ if part.customer == True:
4453+ type = 'customer'
4454+ res_type = self.onchange_type(cr, uid, ids, partner_id=partner_id, type=type, context=context)
4455+ if res_type['value'] and res_type['value'].get('account_id', False):
4456+ return {'value': {'type': type, 'account_id': res_type['value']['account_id']}}
4457+ return {'value': {'type': type}}
4458+
4459+ def onchange_type(self, cr, uid, line_id, partner_id, type, context=None):
4460+ res = {'value': {}}
4461+ obj_partner = self.pool.get('res.partner')
4462+ if context is None:
4463+ context = {}
4464+ if not partner_id:
4465+ return res
4466+ account_id = False
4467+ line = self.browse(cr, uid, line_id, context=context)
4468+ if not line or (line and not line[0].account_id):
4469+ part = obj_partner.browse(cr, uid, partner_id, context=context)
4470+ if type == 'supplier':
4471+ account_id = part.property_account_payable.id
4472+ else:
4473+ account_id = part.property_account_receivable.id
4474+ res['value']['account_id'] = account_id
4475+ return res
4476+
4477+ _order = "statement_id desc, sequence"
4478+ _name = "account.bank.statement.line"
4479+ _description = "Bank Statement Line"
4480+ _columns = {
4481+ 'name': fields.char('Communication', size=64, required=True),
4482+ 'date': fields.date('Date', required=True),
4483+ 'amount': fields.float('Amount', digits_compute=dp.get_precision('Account')),
4484+ 'type': fields.selection([
4485+ ('supplier','Supplier'),
4486+ ('customer','Customer'),
4487+ ('general','General')
4488+ ], 'Type', required=True),
4489+ 'partner_id': fields.many2one('res.partner', 'Partner'),
4490+ 'account_id': fields.many2one('account.account','Account',
4491+ required=True),
4492+ 'statement_id': fields.many2one('account.bank.statement', 'Statement',
4493+ select=True, required=True, ondelete='cascade'),
4494+ 'analytic_account_id': fields.many2one('account.analytic.account', 'Analytic Account'),
4495+ 'move_ids': fields.many2many('account.move',
4496+ 'account_bank_statement_line_move_rel', 'statement_line_id','move_id',
4497+ 'Moves'),
4498+ 'ref': fields.char('Reference', size=32),
4499+ 'note': fields.text('Notes'),
4500+ 'sequence': fields.integer('Sequence', select=True, help="Gives the sequence order when displaying a list of bank statement lines."),
4501+ 'company_id': fields.related('statement_id', 'company_id', type='many2one', relation='res.company', string='Company', store=True, readonly=True),
4502+ }
4503+ _defaults = {
4504+ 'name': lambda self,cr,uid,context={}: self.pool.get('ir.sequence').get(cr, uid, 'account.bank.statement.line'),
4505+ 'date': lambda self,cr,uid,context={}: context.get('date', fields.date.context_today(self,cr,uid,context=context)),
4506+ 'type': 'general',
4507+ }
4508+
4509+account_bank_statement_line()
4510+
4511+# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
4512
4513=== added file 'account/account_bank_view.xml'
4514--- account/account_bank_view.xml 1970-01-01 00:00:00 +0000
4515+++ account/account_bank_view.xml 2012-10-04 09:12:55 +0000
4516@@ -0,0 +1,62 @@
4517+<?xml version="1.0" encoding="utf-8"?>
4518+<openerp>
4519+<data>
4520+
4521+ <!--
4522+ Bank Accounts
4523+ -->
4524+
4525+ <record id="view_partner_bank_form_inherit" model="ir.ui.view">
4526+ <field name="name">Partner Bank Accounts - Journal</field>
4527+ <field name="model">res.partner.bank</field>
4528+ <field name="type">form</field>
4529+ <field name="inherit_id" ref="base.view_partner_bank_form"/>
4530+ <field name="arch" type="xml">
4531+ <group name="bank" position="after">
4532+ <group name="accounting" col="2" colspan="2" attrs="{'invisible': [('company_id','=', False)]}" groups="base.group_extended">
4533+ <separator string="Accounting Information" colspan="2"/>
4534+ <field name="journal_id"/>
4535+ <field name="currency_id"/>
4536+ </group>
4537+ </group>
4538+ </field>
4539+ </record>
4540+
4541+
4542+ <record id="view_partner_bank_tree_add_currency" model="ir.ui.view">
4543+ <field name="name">Partner Bank Accounts - Add currency on tree</field>
4544+ <field name="model">res.partner.bank</field>
4545+ <field name="type">tree</field>
4546+ <field name="inherit_id" ref="base.view_partner_bank_tree"/>
4547+ <field name="arch" type="xml">
4548+ <field name="acc_number" position="after">
4549+ <field name="currency_id"/>
4550+ </field>
4551+ </field>
4552+ </record>
4553+
4554+
4555+ <record id="action_bank_tree" model="ir.actions.act_window">
4556+ <field name="name">Setup your Bank Accounts</field>
4557+ <field name="res_model">res.partner.bank</field>
4558+ <field name="view_type">form</field>
4559+ <field name="view_mode">tree,form</field>
4560+ <field name="context" eval="{'default_partner_id':ref('base.main_partner'), 'company_hide':False, 'default_company_id':ref('base.main_company'), 'search_default_my_bank':1}"/>
4561+ <field name="help">Configure your company's bank account and select those that must appear on the report footer. You can reorder banks in the list view. If you use the accounting application of OpenERP, journals and accounts will be created automatically based on these data.</field>
4562+ </record>
4563+ <menuitem
4564+ sequence="0"
4565+ parent="account.account_account_menu"
4566+ id="menu_action_bank_tree"
4567+ action="action_bank_tree"/>
4568+
4569+
4570+ <record id="account_configuration_bank_todo" model="ir.actions.todo">
4571+ <field name="action_id" ref="action_bank_tree"/>
4572+ <field name="category_id" ref="category_accounting_configuration"/>
4573+ <field name="sequence">4</field>
4574+ </record>
4575+
4576+
4577+</data>
4578+</openerp>
4579
4580=== added file 'account/account_cash_statement.py'
4581--- account/account_cash_statement.py 1970-01-01 00:00:00 +0000
4582+++ account/account_cash_statement.py 2012-10-04 09:12:55 +0000
4583@@ -0,0 +1,346 @@
4584+# encoding: utf-8
4585+##############################################################################
4586+#
4587+# OpenERP, Open Source Management Solution
4588+# Copyright (C) 2004-2008 PC Solutions (<http://pcsol.be>). All Rights Reserved
4589+# $Id$
4590+#
4591+# This program is free software: you can redistribute it and/or modify
4592+# it under the terms of the GNU General Public License as published by
4593+# the Free Software Foundation, either version 3 of the License, or
4594+# (at your option) any later version.
4595+#
4596+# This program is distributed in the hope that it will be useful,
4597+# but WITHOUT ANY WARRANTY; without even the implied warranty of
4598+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
4599+# GNU General Public License for more details.
4600+#
4601+# You should have received a copy of the GNU General Public License
4602+# along with this program. If not, see <http://www.gnu.org/licenses/>.
4603+#
4604+##############################################################################
4605+
4606+import time
4607+
4608+from osv import osv, fields
4609+from tools.translate import _
4610+import decimal_precision as dp
4611+
4612+class account_cashbox_line(osv.osv):
4613+
4614+ """ Cash Box Details """
4615+
4616+ _name = 'account.cashbox.line'
4617+ _description = 'CashBox Line'
4618+ _rec_name = 'number'
4619+
4620+ def _sub_total(self, cr, uid, ids, name, arg, context=None):
4621+
4622+ """ Calculates Sub total
4623+ @param name: Names of fields.
4624+ @param arg: User defined arguments
4625+ @return: Dictionary of values.
4626+ """
4627+ res = {}
4628+ for obj in self.browse(cr, uid, ids, context=context):
4629+ res[obj.id] = obj.pieces * obj.number
4630+ return res
4631+
4632+ def on_change_sub(self, cr, uid, ids, pieces, number, *a):
4633+
4634+ """ Calculates Sub total on change of number
4635+ @param pieces: Names of fields.
4636+ @param number:
4637+ """
4638+ sub = pieces * number
4639+ return {'value': {'subtotal': sub or 0.0}}
4640+
4641+ _columns = {
4642+ 'pieces': fields.float('Values', digits_compute=dp.get_precision('Account')),
4643+ 'number': fields.integer('Number'),
4644+ 'subtotal': fields.function(_sub_total, string='Sub Total', type='float', digits_compute=dp.get_precision('Account')),
4645+ 'starting_id': fields.many2one('account.bank.statement', ondelete='cascade'),
4646+ 'ending_id': fields.many2one('account.bank.statement', ondelete='cascade'),
4647+ }
4648+
4649+account_cashbox_line()
4650+
4651+class account_cash_statement(osv.osv):
4652+
4653+ _inherit = 'account.bank.statement'
4654+
4655+ def _get_starting_balance(self, cr, uid, ids, context=None):
4656+
4657+ """ Find starting balance
4658+ @param name: Names of fields.
4659+ @param arg: User defined arguments
4660+ @return: Dictionary of values.
4661+ """
4662+ res = {}
4663+ for statement in self.browse(cr, uid, ids, context=context):
4664+ amount_total = 0.0
4665+
4666+ if statement.journal_id.type not in('cash'):
4667+ continue
4668+
4669+ for line in statement.starting_details_ids:
4670+ amount_total+= line.pieces * line.number
4671+ res[statement.id] = {
4672+ 'balance_start': amount_total
4673+ }
4674+ return res
4675+
4676+ def _balance_end_cash(self, cr, uid, ids, name, arg, context=None):
4677+ """ Find ending balance "
4678+ @param name: Names of fields.
4679+ @param arg: User defined arguments
4680+ @return: Dictionary of values.
4681+ """
4682+ res = {}
4683+ for statement in self.browse(cr, uid, ids, context=context):
4684+ amount_total = 0.0
4685+ for line in statement.ending_details_ids:
4686+ amount_total += line.pieces * line.number
4687+ res[statement.id] = amount_total
4688+ return res
4689+
4690+ def _get_sum_entry_encoding(self, cr, uid, ids, name, arg, context=None):
4691+
4692+ """ Find encoding total of statements "
4693+ @param name: Names of fields.
4694+ @param arg: User defined arguments
4695+ @return: Dictionary of values.
4696+ """
4697+ res2 = {}
4698+ for statement in self.browse(cr, uid, ids, context=context):
4699+ encoding_total=0.0
4700+ for line in statement.line_ids:
4701+ encoding_total += line.amount
4702+ res2[statement.id] = encoding_total
4703+ return res2
4704+
4705+ def _get_company(self, cr, uid, context=None):
4706+ user_pool = self.pool.get('res.users')
4707+ company_pool = self.pool.get('res.company')
4708+ user = user_pool.browse(cr, uid, uid, context=context)
4709+ company_id = user.company_id
4710+ if not company_id:
4711+ company_id = company_pool.search(cr, uid, [])
4712+ return company_id and company_id[0] or False
4713+
4714+ def _get_cash_open_box_lines(self, cr, uid, context=None):
4715+ res = []
4716+ curr = [1, 2, 5, 10, 20, 50, 100, 500]
4717+ for rs in curr:
4718+ dct = {
4719+ 'pieces': rs,
4720+ 'number': 0
4721+ }
4722+ res.append(dct)
4723+ journal_ids = self.pool.get('account.journal').search(cr, uid, [('type', '=', 'cash')], context=context)
4724+ if journal_ids:
4725+ results = self.search(cr, uid, [('journal_id', 'in', journal_ids),('state', '=', 'confirm')], context=context)
4726+ if results:
4727+ cash_st = self.browse(cr, uid, results, context=context)[0]
4728+ for cash_line in cash_st.ending_details_ids:
4729+ for r in res:
4730+ if cash_line.pieces == r['pieces']:
4731+ r['number'] = cash_line.number
4732+ return res
4733+
4734+ def _get_default_cash_close_box_lines(self, cr, uid, context=None):
4735+ res = []
4736+ curr = [1, 2, 5, 10, 20, 50, 100, 500]
4737+ for rs in curr:
4738+ dct = {
4739+ 'pieces': rs,
4740+ 'number': 0
4741+ }
4742+ res.append(dct)
4743+ return res
4744+
4745+ def _get_cash_close_box_lines(self, cr, uid, context=None):
4746+ res = []
4747+ curr = [1, 2, 5, 10, 20, 50, 100, 500]
4748+ for rs in curr:
4749+ dct = {
4750+ 'pieces': rs,
4751+ 'number': 0
4752+ }
4753+ res.append((0, 0, dct))
4754+ return res
4755+
4756+ def _get_cash_open_close_box_lines(self, cr, uid, context=None):
4757+ res = {}
4758+ start_l = []
4759+ end_l = []
4760+ starting_details = self._get_cash_open_box_lines(cr, uid, context=context)
4761+ ending_details = self._get_default_cash_close_box_lines(cr, uid, context)
4762+ for start in starting_details:
4763+ start_l.append((0, 0, start))
4764+ for end in ending_details:
4765+ end_l.append((0, 0, end))
4766+ res['start'] = start_l
4767+ res['end'] = end_l
4768+ return res
4769+
4770+ def _get_statement(self, cr, uid, ids, context=None):
4771+ result = {}
4772+ for line in self.pool.get('account.bank.statement.line').browse(cr, uid, ids, context=context):
4773+ result[line.statement_id.id] = True
4774+ return result.keys()
4775+
4776+ _columns = {
4777+ 'total_entry_encoding': fields.function(_get_sum_entry_encoding, string="Cash Transaction", help="Total cash transactions",
4778+ store = {
4779+ 'account.bank.statement': (lambda self, cr, uid, ids, c={}: ids, ['line_ids','move_line_ids'], 10),
4780+ 'account.bank.statement.line': (_get_statement, ['amount'], 10),
4781+ }),
4782+ 'closing_date': fields.datetime("Closed On"),
4783+ 'balance_end_cash': fields.function(_balance_end_cash, store=True, string='Closing Balance', help="Closing balance based on cashBox"),
4784+ 'starting_details_ids': fields.one2many('account.cashbox.line', 'starting_id', string='Opening Cashbox'),
4785+ 'ending_details_ids': fields.one2many('account.cashbox.line', 'ending_id', string='Closing Cashbox'),
4786+ 'user_id': fields.many2one('res.users', 'Responsible', required=False),
4787+ }
4788+ _defaults = {
4789+ 'state': 'draft',
4790+ 'date': lambda self,cr,uid,context={}: context.get('date', time.strftime("%Y-%m-%d %H:%M:%S")),
4791+ 'user_id': lambda self, cr, uid, context=None: uid,
4792+ 'starting_details_ids': _get_cash_open_box_lines,
4793+ 'ending_details_ids': _get_default_cash_close_box_lines
4794+ }
4795+
4796+ def create(self, cr, uid, vals, context=None):
4797+ if self.pool.get('account.journal').browse(cr, uid, vals['journal_id'], context=context).type == 'cash':
4798+ open_close = self._get_cash_open_close_box_lines(cr, uid, context)
4799+ if vals.get('starting_details_ids', False):
4800+ starting_details_ids = vals.get('starting_details_ids')
4801+ for start in starting_details_ids:
4802+ dict_val = start[2] or {}
4803+ for end in open_close['end']:
4804+ if end[2]['pieces'] == dict_val.get('pieces', 0.0):
4805+ end[2]['number'] += dict_val.get('number', 0.0)
4806+ vals.update({
4807+# 'ending_details_ids': open_close['start'],
4808+ 'starting_details_ids': open_close['end']
4809+ })
4810+ else:
4811+ vals.update({
4812+ 'ending_details_ids': False,
4813+ 'starting_details_ids': False
4814+ })
4815+ res_id = super(account_cash_statement, self).create(cr, uid, vals, context=context)
4816+ self.write(cr, uid, [res_id], {})
4817+ return res_id
4818+
4819+ def write(self, cr, uid, ids, vals, context=None):
4820+ """
4821+ Update redord(s) comes in {ids}, with new value comes as {vals}
4822+ return True on success, False otherwise
4823+
4824+ @param cr: cursor to database
4825+ @param user: id of current user
4826+ @param ids: list of record ids to be update
4827+ @param vals: dict of new values to be set
4828+ @param context: context arguments, like lang, time zone
4829+
4830+ @return: True on success, False otherwise
4831+ """
4832+
4833+ super(account_cash_statement, self).write(cr, uid, ids, vals, context=context)
4834+ res = self._get_starting_balance(cr, uid, ids)
4835+ for rs in res:
4836+ super(account_cash_statement, self).write(cr, uid, [rs], res.get(rs))
4837+ return True
4838+
4839+ def onchange_journal_id(self, cr, uid, statement_id, journal_id, context=None):
4840+ """ Changes balance start and starting details if journal_id changes"
4841+ @param statement_id: Changed statement_id
4842+ @param journal_id: Changed journal_id
4843+ @return: Dictionary of changed values
4844+ """
4845+ res = {}
4846+ balance_start = 0.0
4847+ if not journal_id:
4848+ res.update({
4849+ 'balance_start': balance_start
4850+ })
4851+ return res
4852+ return super(account_cash_statement, self).onchange_journal_id(cr, uid, statement_id, journal_id, context=context)
4853+
4854+ def _equal_balance(self, cr, uid, cash_id, context=None):
4855+ statement = self.browse(cr, uid, cash_id, context=context)
4856+ self.write(cr, uid, [cash_id], {'balance_end_real': statement.balance_end})
4857+ statement.balance_end_real = statement.balance_end
4858+ if statement.balance_end != statement.balance_end_cash:
4859+ return False
4860+ return True
4861+
4862+ def _user_allow(self, cr, uid, statement_id, context=None):
4863+ return True
4864+
4865+ def button_open(self, cr, uid, ids, context=None):
4866+ """ Changes statement state to Running.
4867+ @return: True
4868+ """
4869+ obj_seq = self.pool.get('ir.sequence')
4870+ if context is None:
4871+ context = {}
4872+ statement_pool = self.pool.get('account.bank.statement')
4873+ for statement in statement_pool.browse(cr, uid, ids, context=context):
4874+ vals = {}
4875+ if not self._user_allow(cr, uid, statement.id, context=context):
4876+ raise osv.except_osv(_('Error !'), (_('User %s does not have rights to access %s journal !') % (statement.user_id.name, statement.journal_id.name)))
4877+
4878+ if statement.name and statement.name == '/':
4879+ c = {'fiscalyear_id': statement.period_id.fiscalyear_id.id}
4880+ if statement.journal_id.sequence_id:
4881+ st_number = obj_seq.next_by_id(cr, uid, statement.journal_id.sequence_id.id, context=c)
4882+ else:
4883+ st_number = obj_seq.next_by_code(cr, uid, 'account.cash.statement', context=c)
4884+ vals.update({
4885+ 'name': st_number
4886+ })
4887+
4888+ vals.update({
4889+ 'state': 'open',
4890+ })
4891+ self.write(cr, uid, [statement.id], vals, context=context)
4892+ return True
4893+
4894+ def balance_check(self, cr, uid, cash_id, journal_type='bank', context=None):
4895+ if journal_type == 'bank':
4896+ return super(account_cash_statement, self).balance_check(cr, uid, cash_id, journal_type, context)
4897+ if not self._equal_balance(cr, uid, cash_id, context):
4898+ raise osv.except_osv(_('Error !'), _('The closing balance should be the same than the computed balance!'))
4899+ return True
4900+
4901+ def statement_close(self, cr, uid, ids, journal_type='bank', context=None):
4902+ if journal_type == 'bank':
4903+ return super(account_cash_statement, self).statement_close(cr, uid, ids, journal_type, context)
4904+ vals = {
4905+ 'state':'confirm',
4906+ 'closing_date': time.strftime("%Y-%m-%d %H:%M:%S")
4907+ }
4908+ return self.write(cr, uid, ids, vals, context=context)
4909+
4910+ def check_status_condition(self, cr, uid, state, journal_type='bank'):
4911+ if journal_type == 'bank':
4912+ return super(account_cash_statement, self).check_status_condition(cr, uid, state, journal_type)
4913+ return state=='open'
4914+
4915+ def button_confirm_cash(self, cr, uid, ids, context=None):
4916+ super(account_cash_statement, self).button_confirm_bank(cr, uid, ids, context=context)
4917+ return self.write(cr, uid, ids, {'closing_date': time.strftime("%Y-%m-%d %H:%M:%S")}, context=context)
4918+
4919+ def button_cancel(self, cr, uid, ids, context=None):
4920+ cash_box_line_pool = self.pool.get('account.cashbox.line')
4921+ super(account_cash_statement, self).button_cancel(cr, uid, ids, context=context)
4922+ for st in self.browse(cr, uid, ids, context):
4923+ for end in st.ending_details_ids:
4924+ cash_box_line_pool.write(cr, uid, [end.id], {'number': 0})
4925+ return True
4926+
4927+account_cash_statement()
4928+
4929+# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
4930
4931=== added file 'account/account_end_fy.xml'
4932--- account/account_end_fy.xml 1970-01-01 00:00:00 +0000
4933+++ account/account_end_fy.xml 2012-10-04 09:12:55 +0000
4934@@ -0,0 +1,20 @@
4935+<?xml version="1.0" encoding="utf-8"?>
4936+<openerp>
4937+ <data>
4938+
4939+ <record id="action_account_period_tree" model="ir.actions.act_window">
4940+ <field name="name">Close a Period</field>
4941+ <field name="res_model">account.period</field>
4942+ <field name="view_type">form</field>
4943+ <field name="view_id" ref="view_account_period_tree"/>
4944+ <field name="context">{'search_default_draft': 1}</field>
4945+ <field name="help">A period is a fiscal period of time during which accounting entries should be recorded for accounting related activities. Monthly period is the norm but depending on your countries or company needs, you could also have quarterly periods. Closing a period will make it impossible to record new accounting entries, all new entries should then be made on the following open period. Close a period when you do not want to record new entries and want to lock this period for tax related calculation.</field>
4946+ </record>
4947+ <menuitem
4948+ action="action_account_period_tree"
4949+ id="menu_action_account_period_close_tree"
4950+ parent="account.menu_account_end_year_treatments"
4951+ sequence="0" groups="base.group_extended"/>
4952+
4953+ </data>
4954+</openerp>
4955
4956=== added file 'account/account_financial_report.py'
4957--- account/account_financial_report.py 1970-01-01 00:00:00 +0000
4958+++ account/account_financial_report.py 2012-10-04 09:12:55 +0000
4959@@ -0,0 +1,135 @@
4960+# -*- coding: utf-8 -*-
4961+##############################################################################
4962+#
4963+# OpenERP, Open Source Management Solution
4964+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
4965+#
4966+# This program is free software: you can redistribute it and/or modify
4967+# it under the terms of the GNU Affero General Public License as
4968+# published by the Free Software Foundation, either version 3 of the
4969+# License, or (at your option) any later version.
4970+#
4971+# This program is distributed in the hope that it will be useful,
4972+# but WITHOUT ANY WARRANTY; without even the implied warranty of
4973+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
4974+# GNU Affero General Public License for more details.
4975+#
4976+# You should have received a copy of the GNU Affero General Public License
4977+# along with this program. If not, see <http://www.gnu.org/licenses/>.
4978+#
4979+##############################################################################
4980+
4981+import time
4982+from datetime import datetime
4983+from dateutil.relativedelta import relativedelta
4984+from operator import itemgetter
4985+
4986+import netsvc
4987+import pooler
4988+from osv import fields, osv
4989+import decimal_precision as dp
4990+from tools.translate import _
4991+
4992+# ---------------------------------------------------------
4993+# Account Financial Report
4994+# ---------------------------------------------------------
4995+
4996+class account_financial_report(osv.osv):
4997+ _name = "account.financial.report"
4998+ _description = "Account Report"
4999+
5000+ def _get_level(self, cr, uid, ids, field_name, arg, context=None):
The diff has been truncated for viewing.