[ADD] added onchange on country_id
[odoo/odoo.git] / openerp / addons / base / res / res_company.py
1 # -*- coding: utf-8 -*-
2 ##############################################################################
3 #
4 #    OpenERP, Open Source Management Solution
5 #    Copyright (C) 2004-2009 Tiny SPRL (<http://tiny.be>).
6 #
7 #    This program is free software: you can redistribute it and/or modify
8 #    it under the terms of the GNU Affero General Public License as
9 #    published by the Free Software Foundation, either version 3 of the
10 #    License, or (at your option) any later version.
11 #
12 #    This program is distributed in the hope that it will be useful,
13 #    but WITHOUT ANY WARRANTY; without even the implied warranty of
14 #    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 #    GNU Affero General Public License for more details.
16 #
17 #    You should have received a copy of the GNU Affero General Public License
18 #    along with this program.  If not, see <http://www.gnu.org/licenses/>.
19 #
20 ##############################################################################
21
22 from osv import osv
23 from osv import fields
24 import os
25 import tools
26 import openerp
27 from tools.translate import _
28 from tools.safe_eval import safe_eval as eval
29
30 class multi_company_default(osv.osv):
31     """
32     Manage multi company default value
33     """
34     _name = 'multi_company.default'
35     _description = 'Default multi company'
36     _order = 'company_id,sequence,id'
37
38     _columns = {
39         'sequence': fields.integer('Sequence'),
40         'name': fields.char('Name', size=256, required=True, help='Name it to easily find a record'),
41         'company_id': fields.many2one('res.company', 'Main Company', required=True,
42             help='Company where the user is connected'),
43         'company_dest_id': fields.many2one('res.company', 'Default Company', required=True,
44             help='Company to store the current record'),
45         'object_id': fields.many2one('ir.model', 'Object', required=True,
46             help='Object affected by this rule'),
47         'expression': fields.char('Expression', size=256, required=True,
48             help='Expression, must be True to match\nuse context.get or user (browse)'),
49         'field_id': fields.many2one('ir.model.fields', 'Field', help='Select field property'),
50     }
51
52     _defaults = {
53         'expression': lambda *a: 'True',
54         'sequence': lambda *a: 100,
55     }
56
57     def copy(self, cr, uid, id, default=None, context=None):
58         """
59         Add (copy) in the name when duplicate record
60         """
61         if not context:
62             context = {}
63         if not default:
64             default = {}
65         company = self.browse(cr, uid, id, context=context)
66         default = default.copy()
67         default['name'] = company.name + _(' (copy)')
68         return super(multi_company_default, self).copy(cr, uid, id, default, context=context)
69
70 multi_company_default()
71
72 class res_company(osv.osv):
73     _name = "res.company"
74     _description = 'Companies'
75     _order = 'name'
76
77     def _get_address_data(self, cr, uid, ids, field_names, arg, context=None):
78         """ Read the 'address' functional fields. """
79         result = {}
80         part_obj = self.pool.get('res.partner')
81         for company in self.browse(cr, uid, ids, context=context):
82             result[company.id] = {}.fromkeys(field_names, False)
83             if company.partner_id:
84                 address_data = part_obj.address_get(cr, openerp.SUPERUSER_ID, [company.partner_id.id], adr_pref=['default'])
85                 if address_data['default']:
86                     address = part_obj.read(cr, openerp.SUPERUSER_ID, address_data['default'], field_names, context=context)
87                     for field in field_names:
88                         result[company.id][field] = address[field] or False
89         return result
90
91
92     def _set_address_data(self, cr, uid, company_id, name, value, arg, context=None):
93         """ Write the 'address' functional fields. """
94         company = self.browse(cr, uid, company_id, context=context)
95         if company.partner_id:
96             part_obj = self.pool.get('res.partner')
97             address_data = part_obj.address_get(cr, uid, [company.partner_id.id], adr_pref=['default'])
98             address = address_data['default']
99             if address:
100                 part_obj.write(cr, uid, [address], {name: value or False})
101             else:
102                 part_obj.create(cr, uid, {name: value or False, 'parent_id': company.partner_id.id}, context=context)
103         return True
104
105
106     _columns = {
107         'name': fields.related('partner_id', 'name', string='Company Name', size=128, required=True, store=True, type='char'),
108         'parent_id': fields.many2one('res.company', 'Parent Company', select=True),
109         'child_ids': fields.one2many('res.company', 'parent_id', 'Child Companies'),
110         'partner_id': fields.many2one('res.partner', 'Partner', required=True),
111         'rml_header1': fields.char('Company Slogan', size=200, help="Appears by default on the top right corner of your printed documents (report header)."),
112         'rml_footer1': fields.char('General Information Footer', size=200),
113         'rml_footer2': fields.char('Bank Accounts Footer', size=250, help="Write here your bank accounts for customer payments."),
114         'rml_header': fields.text('RML Header', required=True),
115         'rml_header2': fields.text('RML Internal Header', required=True),
116         'rml_header3': fields.text('RML Internal Header for Landscape Reports', required=True),
117         'logo': fields.related('partner_id', 'photo', string="Logo", type="binary"),
118         'currency_id': fields.many2one('res.currency', 'Currency', required=True),
119         'currency_ids': fields.one2many('res.currency', 'company_id', 'Currency'),
120         'user_ids': fields.many2many('res.users', 'res_company_users_rel', 'cid', 'user_id', 'Accepted Users'),
121         'account_no':fields.char('Account No.', size=64),
122         'street': fields.function(_get_address_data, fnct_inv=_set_address_data, size=128, type='char', string="Street", multi='address'),
123         'street2': fields.function(_get_address_data, fnct_inv=_set_address_data, size=128, type='char', string="Street2", multi='address'),
124         'zip': fields.function(_get_address_data, fnct_inv=_set_address_data, size=24, type='char', string="Zip", multi='address'),
125         'city': fields.function(_get_address_data, fnct_inv=_set_address_data, size=24, type='char', string="City", multi='address'),
126         'state_id': fields.function(_get_address_data, fnct_inv=_set_address_data, type='many2one', domain="[('country_id', '=', country_id)]", relation='res.country.state', string="Fed. State", multi='address'),
127         'bank_ids': fields.one2many('res.partner.bank','company_id', 'Bank Accounts', help='Bank accounts related to this company'),
128         'country_id': fields.function(_get_address_data, fnct_inv=_set_address_data, type='many2one', relation='res.country', string="Country", multi='address'),
129         'email': fields.function(_get_address_data, fnct_inv=_set_address_data, size=64, type='char', string="Email", multi='address'),
130         'phone': fields.function(_get_address_data, fnct_inv=_set_address_data, size=64, type='char', string="Phone", multi='address'),
131         'fax': fields.function(_get_address_data, fnct_inv=_set_address_data, size=64, type='char', string="Fax", multi='address'),
132         'website': fields.related('partner_id', 'website', string="Website", type="char", size=64),
133         'vat': fields.related('partner_id', 'vat', string="Tax ID", type="char", size=32),
134         'company_registry': fields.char('Company Registry', size=64),
135         'paper_format': fields.selection([('a4', 'A4'), ('us_letter', 'US Letter')], "Paper Format", required=True),
136     }
137     _sql_constraints = [
138         ('name_uniq', 'unique (name)', 'The company name must be unique !')
139     ]
140     def on_change_header(self, cr, uid, ids, phone, email, fax, website, vat, reg=False, context=None):
141         val = []
142         if phone: val.append(_('Phone: ')+phone)
143         if fax: val.append(_('Fax: ')+fax)
144         if website: val.append(_('Website: ')+website)
145         if vat: val.append(_('VAT: ')+vat)
146         if reg: val.append(_('Reg: ')+reg)
147         return {'value': {'rml_footer1':' | '.join(val)}}
148     
149     def on_change_country(self, cr, uid, ids, country_id, context=None):
150         res_country_obj = self.pool.get('res.country')
151         if not country_id:
152             default_currency_id = self._get_euro(cr, uid, context=context)
153             return {'value': {'currency_id': default_currency_id}}
154         currency_id = res_country_obj.browse(cr, uid, country_id, context=context).currency_id.id
155         return {'value': {'currency_id': currency_id}}
156
157     def _search(self, cr, uid, args, offset=0, limit=None, order=None,
158             context=None, count=False, access_rights_uid=None):
159
160         if context is None:
161             context = {}
162         user_preference = context.get('user_preference', False)
163         if user_preference:
164             # We browse as superuser. Otherwise, the user would be able to
165             # select only the currently visible companies (according to rules,
166             # which are probably to allow to see the child companies) even if
167             # she belongs to some other companies.
168             user = self.pool.get('res.users').browse(cr, 1, uid, context=context)
169             cmp_ids = list(set([user.company_id.id] + [cmp.id for cmp in user.company_ids]))
170             return cmp_ids
171         return super(res_company, self)._search(cr, uid, args, offset=offset, limit=limit, order=order,
172             context=context, count=count, access_rights_uid=access_rights_uid)
173
174     def _company_default_get(self, cr, uid, object=False, field=False, context=None):
175         """
176         Check if the object for this company have a default value
177         """
178         if not context:
179             context = {}
180         proxy = self.pool.get('multi_company.default')
181         args = [
182             ('object_id.model', '=', object),
183             ('field_id', '=', field),
184         ]
185
186         ids = proxy.search(cr, uid, args, context=context)
187         user = self.pool.get('res.users').browse(cr, uid, uid, context=context)
188         for rule in proxy.browse(cr, uid, ids, context):
189             if eval(rule.expression, {'context': context, 'user': user}):
190                 return rule.company_dest_id.id
191         return user.company_id.id
192
193     @tools.ormcache()
194     def _get_company_children(self, cr, uid=None, company=None):
195         if not company:
196             return []
197         ids =  self.search(cr, uid, [('parent_id','child_of',[company])])
198         return ids
199
200     def _get_partner_hierarchy(self, cr, uid, company_id, context=None):
201         if company_id:
202             parent_id = self.browse(cr, uid, company_id)['parent_id']
203             if parent_id:
204                 return self._get_partner_hierarchy(cr, uid, parent_id.id, context)
205             else:
206                 return self._get_partner_descendance(cr, uid, company_id, [], context)
207         return []
208
209     def _get_partner_descendance(self, cr, uid, company_id, descendance, context=None):
210         descendance.append(self.browse(cr, uid, company_id).partner_id.id)
211         for child_id in self._get_company_children(cr, uid, company_id):
212             if child_id != company_id:
213                 descendance = self._get_partner_descendance(cr, uid, child_id, descendance)
214         return descendance
215
216     #
217     # This function restart the cache on the _get_company_children method
218     #
219     def cache_restart(self, cr):
220         self._get_company_children.clear_cache(self)
221
222     def create(self, cr, uid, vals, context=None):
223         if not vals.get('name', False) or vals.get('partner_id', False):
224             self.cache_restart(cr)
225             return super(res_company, self).create(cr, uid, vals, context=context)
226         obj_partner = self.pool.get('res.partner')
227         partner_id = obj_partner.create(cr, uid, {'name': vals['name'], 'is_company':True}, context=context)
228         vals.update({'partner_id': partner_id})
229         self.cache_restart(cr)
230         company_id = super(res_company, self).create(cr, uid, vals, context=context)
231         obj_partner.write(cr, uid, partner_id, {'company_id': company_id}, context=context)
232         return company_id
233
234     def write(self, cr, *args, **argv):
235         self.cache_restart(cr)
236         return super(res_company, self).write(cr, *args, **argv)
237
238     def _get_euro(self, cr, uid, context=None):
239         try:
240             data_obj = self.pool.get('ir.model.data')
241             data_id = data_obj._get_id(cr, uid, 'base', 'EUR')
242             return data_obj.browse(cr, uid, data_id, context=context).res_id
243         except:
244             return False
245
246     def _get_logo(self, cr, uid, ids):
247         return open(os.path.join( tools.config['root_path'], 'addons', 'base', 'res', 'res_company_logo.png'), 'rb') .read().encode('base64')
248
249     _header = """
250 <header>
251 <pageTemplate>
252     <frame id="first" x1="28.0" y1="28.0" width="%s" height="%s"/>
253     <pageGraphics>
254         <fill color="black"/>
255         <stroke color="black"/>
256         <setFont name="DejaVu Sans" size="8"/>
257         <drawString x="%s" y="%s"> [[ formatLang(time.strftime("%%Y-%%m-%%d"), date=True) ]]  [[ time.strftime("%%H:%%M") ]]</drawString>
258         <setFont name="DejaVu Sans Bold" size="10"/>
259         <drawCentredString x="%s" y="%s">[[ company.partner_id.name ]]</drawCentredString>
260         <stroke color="#000000"/>
261         <lines>%s</lines>
262     </pageGraphics>
263 </pageTemplate>
264 </header>"""
265
266     _header2 = _header % (539, 772, "1.0cm", "28.3cm", "11.1cm", "28.3cm", "1.0cm 28.1cm 20.1cm 28.1cm")
267
268     _header3 = _header % (786, 525, 25, 555, 440, 555, "25 550 818 550")
269
270     def _get_header(self,cr,uid,ids):
271         try :
272             header_file = tools.file_open(os.path.join('base', 'report', 'corporate_rml_header.rml'))
273             try:
274                 return header_file.read()
275             finally:
276                 header_file.close()
277         except:
278             return self._header_a4
279
280     _header_main = """
281     <header>
282     <pageTemplate>
283         <frame id="first" x1="1.3cm" y1="2.5cm" height="%s" width="19.0cm"/>
284         <pageGraphics>
285             <!-- You Logo - Change X,Y,Width and Height -->
286             <image x="1.3cm" y="%s" height="40.0" >[[ company.logo or removeParentNode('image') ]]</image>
287             <setFont name="DejaVu Sans" size="8"/>
288             <fill color="black"/>
289             <stroke color="black"/>
290             <lines>1.3cm %s 20cm %s</lines>
291
292             <drawRightString x="20cm" y="%s">[[ company.rml_header1 ]]</drawRightString>
293
294
295             <drawString x="1.3cm" y="%s">[[ company.partner_id.name ]]</drawString>
296             <drawString x="1.3cm" y="%s">[[ company.partner_id.street or  '' ]]</drawString>
297             <drawString x="1.3cm" y="%s">[[ company.partner_id.city or '' ]] - [[ company.partner_id.country_id and company.partner_id.country_id.name  or '']]</drawString>
298             <drawString x="1.3cm" y="%s">Phone:</drawString>
299             <drawRightString x="7cm" y="%s">[[ company.partner_id.phone or '' ]]</drawRightString>
300             <drawString x="1.3cm" y="%s">Mail:</drawString>
301             <drawRightString x="7cm" y="%s">[[ company.partner_id.email or '' ]]</drawRightString>
302             <lines>1.3cm %s 7cm %s</lines>
303
304             <!--page bottom-->
305
306             <lines>1.2cm 2.15cm 19.9cm 2.15cm</lines>
307
308             <drawCentredString x="10.5cm" y="1.7cm">[[ company.rml_footer1 ]]</drawCentredString>
309             <drawCentredString x="10.5cm" y="1.25cm">[[ company.rml_footer2 ]]</drawCentredString>
310             <drawCentredString x="10.5cm" y="0.8cm">Contact : [[ user.name ]] - Page: <pageNumber/></drawCentredString>
311         </pageGraphics>
312     </pageTemplate>
313 </header>"""
314
315     _header_a4 = _header_main % ('23.0cm', '27.6cm', '27.7cm', '27.7cm', '27.8cm', '27.2cm', '26.8cm', '26.4cm', '26.0cm', '26.0cm', '25.6cm', '25.6cm', '25.5cm', '25.5cm')
316     _header_letter = _header_main % ('21.3cm', '25.9cm', '26.0cm', '26.0cm', '26.1cm', '25.5cm', '25.1cm', '24.7cm', '24.3cm', '24.3cm', '23.9cm', '23.9cm', '23.8cm', '23.8cm')
317
318     def onchange_paper_format(self, cr, uid, ids, paper_format, context=None):
319         if paper_format == 'us_letter':
320             return {'value': {'rml_header': self._header_letter}}
321         return {'value': {'rml_header': self._header_a4}}
322
323     _defaults = {
324         'currency_id': _get_euro,
325         'paper_format': 'a4',
326         'rml_header':_get_header,
327         'rml_header2': _header2,
328         'rml_header3': _header3,
329         'logo':_get_logo
330     }
331
332     _constraints = [
333         (osv.osv._check_recursion, 'Error! You can not create recursive companies.', ['parent_id'])
334     ]
335
336
337 res_company()
338
339 # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
340