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