232 lines
9.4 KiB
232 lines
9.4 KiB
# -*- coding: utf-8 -*-
|
|
##############################################################################
|
|
#
|
|
# OpenERP, Open Source Management Solution
|
|
# Copyright (C) 2013-TODAY OpenERP SA (<http://www.openerp.com>).
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Affero General Public License as
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
# License, or (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU Affero General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
##############################################################################
|
|
|
|
from openerp.osv import fields, orm, expression
|
|
from openerp.tools.translate import _
|
|
|
|
|
|
class res_partner(orm.Model):
|
|
_inherit = 'res.partner'
|
|
|
|
def _type_selection(self, cr, uid, context=None):
|
|
return [
|
|
('standalone', _('Standalone Contact')),
|
|
('attached', _('Attached to existing Contact')),
|
|
]
|
|
|
|
def _get_contact_type(self, cr, uid, ids, field_name, args, context=None):
|
|
result = dict.fromkeys(ids, 'standalone')
|
|
for partner in self.browse(cr, uid, ids, context=context):
|
|
if partner.contact_id:
|
|
result[partner.id] = 'attached'
|
|
return result
|
|
|
|
_columns = {
|
|
'contact_type': fields.function(
|
|
_get_contact_type,
|
|
type='selection',
|
|
selection=lambda self, *a, **kw: self._type_selection(*a, **kw),
|
|
string='Contact Type',
|
|
required=True,
|
|
select=1,
|
|
store=True,
|
|
),
|
|
'contact_id': fields.many2one(
|
|
'res.partner',
|
|
'Main Contact',
|
|
domain=[
|
|
('is_company', '=', False),
|
|
('contact_type', '=', 'standalone'),
|
|
],
|
|
),
|
|
'other_contact_ids': fields.one2many(
|
|
'res.partner',
|
|
'contact_id',
|
|
'Others Positions',
|
|
),
|
|
}
|
|
|
|
_defaults = {
|
|
'contact_type': 'standalone',
|
|
}
|
|
|
|
def _basecontact_check_context(self, cr, user, mode, context=None):
|
|
""" Remove 'search_show_all_positions' for non-search mode.
|
|
Keeping it in context can result in unexpected behaviour (ex: reading
|
|
one2many might return wrong result - i.e with "attached contact"
|
|
removed even if it's directly linked to a company).
|
|
"""
|
|
context = dict(context or {})
|
|
if mode != 'search':
|
|
context.pop('search_show_all_positions', None)
|
|
return context
|
|
|
|
def search(
|
|
self, cr, user, args, offset=0, limit=None, order=None,
|
|
context=None, count=False):
|
|
""" Display only standalone contact matching ``args`` or having
|
|
attached contact matching ``args`` """
|
|
if context is None:
|
|
context = {}
|
|
if context.get('search_show_all_positions') is False:
|
|
args = expression.normalize_domain(args)
|
|
attached_contact_args = expression.AND(
|
|
(args, [('contact_type', '=', 'attached')])
|
|
)
|
|
attached_contact_ids = super(res_partner, self).search(
|
|
cr, user, attached_contact_args, context=context
|
|
)
|
|
args = expression.OR((
|
|
expression.AND(([('contact_type', '=', 'standalone')], args)),
|
|
[('other_contact_ids', 'in', attached_contact_ids)],
|
|
))
|
|
return super(res_partner, self).search(
|
|
cr, user, args, offset=offset, limit=limit, order=order,
|
|
context=context, count=count
|
|
)
|
|
|
|
def create(self, cr, user, vals, context=None):
|
|
context = self._basecontact_check_context(cr, user, 'create', context)
|
|
if not vals.get('name') and vals.get('contact_id'):
|
|
vals['name'] = self.browse(
|
|
cr, user, vals['contact_id'], context=context).name
|
|
return super(res_partner, self).create(cr, user, vals, context=context)
|
|
|
|
def read(
|
|
self, cr, user, ids, fields=None, context=None,
|
|
load='_classic_read'):
|
|
context = self._basecontact_check_context(cr, user, 'read', context)
|
|
return super(res_partner, self).read(
|
|
cr, user, ids, fields=fields, context=context, load=load)
|
|
|
|
def write(self, cr, user, ids, vals, context=None):
|
|
context = self._basecontact_check_context(cr, user, 'write', context)
|
|
return super(
|
|
res_partner, self).write(cr, user, ids, vals, context=context)
|
|
|
|
def unlink(self, cr, user, ids, context=None):
|
|
context = self._basecontact_check_context(cr, user, 'unlink', context)
|
|
return super(res_partner, self).unlink(cr, user, ids, context=context)
|
|
|
|
def _commercial_partner_compute(
|
|
self, cr, uid, ids, name, args, context=None):
|
|
""" Returns the partner that is considered the commercial
|
|
entity of this partner. The commercial entity holds the master data
|
|
for all commercial fields (see :py:meth:`~_commercial_fields`) """
|
|
result = super(res_partner, self)._commercial_partner_compute(
|
|
cr, uid, ids, name, args, context=context)
|
|
for partner in self.browse(cr, uid, ids, context=context):
|
|
if partner.contact_type == 'attached' and not partner.parent_id:
|
|
result[partner.id] = partner.contact_id.id
|
|
return result
|
|
|
|
def _contact_fields(self, cr, uid, context=None):
|
|
""" Returns the list of contact fields that are synced from the parent
|
|
when a partner is attached to him. """
|
|
return ['name', 'title']
|
|
|
|
def _contact_sync_from_parent(self, cr, uid, partner, context=None):
|
|
""" Handle sync of contact fields when a new parent contact entity
|
|
is set, as if they were related fields
|
|
"""
|
|
if partner.contact_id:
|
|
contact_fields = self._contact_fields(cr, uid, context=context)
|
|
sync_vals = self._update_fields_values(
|
|
cr, uid, partner.contact_id, contact_fields, context=context
|
|
)
|
|
partner.write(sync_vals)
|
|
|
|
def update_contact(self, cr, uid, ids, vals, context=None):
|
|
if context is None:
|
|
context = {}
|
|
if context.get('__update_contact_lock'):
|
|
return
|
|
contact_fields = self._contact_fields(cr, uid, context=context)
|
|
contact_vals = dict(
|
|
(field, vals[field]) for field in contact_fields if field in vals
|
|
)
|
|
if contact_vals:
|
|
ctx = dict(context, __update_contact_lock=True)
|
|
self.write(cr, uid, ids, contact_vals, context=ctx)
|
|
|
|
def _fields_sync(self, cr, uid, partner, update_values, context=None):
|
|
"""Sync commercial fields and address fields from company and to
|
|
children, contact fields from contact and to attached contact
|
|
after create/update, just as if those were all modeled as
|
|
fields.related to the parent
|
|
"""
|
|
super(res_partner, self)._fields_sync(
|
|
cr, uid, partner, update_values, context=context
|
|
)
|
|
contact_fields = self._contact_fields(cr, uid, context=context)
|
|
# 1. From UPSTREAM: sync from parent contact
|
|
if update_values.get('contact_id'):
|
|
self._contact_sync_from_parent(cr, uid, partner, context=context)
|
|
# 2. To DOWNSTREAM: sync contact fields to parent or related
|
|
elif any(field in contact_fields for field in update_values):
|
|
update_ids = [
|
|
c.id for c in partner.other_contact_ids if not c.is_company
|
|
]
|
|
if partner.contact_id:
|
|
update_ids.append(partner.contact_id.id)
|
|
self.update_contact(
|
|
cr, uid, update_ids, update_values, context=context
|
|
)
|
|
|
|
def onchange_contact_id(self, cr, uid, ids, contact_id, context=None):
|
|
values = {}
|
|
if contact_id:
|
|
values['name'] = self.browse(
|
|
cr, uid, contact_id, context=context).name
|
|
return {'value': values}
|
|
|
|
def onchange_contact_type(self, cr, uid, ids, contact_type, context=None):
|
|
values = {}
|
|
if contact_type == 'standalone':
|
|
values['contact_id'] = False
|
|
return {'value': values}
|
|
|
|
|
|
class ir_actions_window(orm.Model):
|
|
_inherit = 'ir.actions.act_window'
|
|
|
|
def read(
|
|
self, cr, user, ids, fields=None, context=None,
|
|
load='_classic_read'):
|
|
action_ids = ids
|
|
if isinstance(ids, (int, long)):
|
|
action_ids = [ids]
|
|
actions = super(ir_actions_window, self).read(
|
|
cr, user, action_ids, fields=fields, context=context, load=load
|
|
)
|
|
for action in actions:
|
|
if action.get('res_model', '') == 'res.partner':
|
|
# By default, only show standalone contact
|
|
action_context = action.get('context', '{}') or '{}'
|
|
if 'search_show_all_positions' not in action_context:
|
|
action['context'] = action_context.replace(
|
|
'{', "{'search_show_all_positions': False,", 1
|
|
)
|
|
if isinstance(ids, (int, long)):
|
|
if actions:
|
|
return actions[0]
|
|
return False
|
|
return actions
|