Holger Brunn
11 years ago
17 changed files with 1785 additions and 0 deletions
-
21partner_relations/__init__.py
-
94partner_relations/__openerp__.py
-
268partner_relations/i18n/nl.po
-
277partner_relations/i18n/partner_relations.pot
-
25partner_relations/model/__init__.py
-
238partner_relations/model/res_partner.py
-
330partner_relations/model/res_partner_relation.py
-
88partner_relations/model/res_partner_relation_all.py
-
48partner_relations/model/res_partner_relation_type.py
-
167partner_relations/model/res_partner_relation_type_selection.py
-
7partner_relations/security/ir.model.access.csv
-
BINpartner_relations/static/src/img/icon.png
-
15partner_relations/view/menu.xml
-
96partner_relations/view/res_partner.xml
-
33partner_relations/view/res_partner_relation.xml
-
34partner_relations/view/res_partner_relation_all.xml
-
44partner_relations/view/res_partner_relation_type.xml
@ -0,0 +1,21 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2013 Therp BV (<http://therp.nl>). |
|||
# |
|||
# 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/>. |
|||
# |
|||
############################################################################## |
|||
import model |
@ -0,0 +1,94 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2013 Therp BV (<http://therp.nl>). |
|||
# |
|||
# 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/>. |
|||
# |
|||
############################################################################## |
|||
{ |
|||
"name": "Partner relations", |
|||
"version": "1.1", |
|||
"author": "Therp BV", |
|||
"complexity": "normal", |
|||
"description": """ |
|||
Introduction |
|||
------------ |
|||
|
|||
This addon aims to provide generic means to model relations between partners. |
|||
|
|||
Examples would be 'is sibling of' or 'is friend of', but also 'has contract X |
|||
with' or 'is assistant of'. This way, you can enode your knowledge about your |
|||
partners directly in your partner list. |
|||
|
|||
Usage |
|||
----- |
|||
|
|||
Before being able to use relations, you'll have define some first. Do that in |
|||
Sales / Configuration / Address Book / Partner relations. Here, you need to |
|||
name both sides of the relation: To have an assistant-relation, you would name |
|||
one side 'is assistant of' and the other side 'has assistant'. This relation |
|||
only makes sense between people, so you would choose 'Person' for both partner |
|||
types. For the relation 'is a competitor of', both sides would be companies, |
|||
while the relation 'has worked for' should have persons on the left side and |
|||
companies on the right side. If you leave this field empty, the relation is |
|||
applicable to all types of partners. |
|||
|
|||
If you use categories to further specify the type of partners, you could for |
|||
example enforce that the 'is member of' relation can only have companies with |
|||
label 'Organization' on the left side. |
|||
|
|||
Now open a partner and choose relations as appropriate in the 'Relations' tab. |
|||
|
|||
Searching partners with relations |
|||
--------------------------------- |
|||
|
|||
Searching for relations is integrated transparently into the partner search |
|||
form. To find all assistants in your database, fill in 'is assistant of' and |
|||
autocomplete will propose to search for partners having this relation. Now if |
|||
you want to find Anna's assistant, you fill in 'Anna' and one of the proposals |
|||
is to search for partners having a relation with Anna. This results in Anna's |
|||
assistant(s), as you searched for assistants before. |
|||
|
|||
By default, only active, not expired relations are shown. If you need to find |
|||
partners that had some relation at a certain date, fill in that date in the |
|||
search box and one of the proposals is to search for relations valid at that |
|||
date.""", |
|||
"category": "Customer Relationship Management", |
|||
"depends": [ |
|||
'base', |
|||
'web_m2x_options', |
|||
'web_tree_many2one_clickable', |
|||
], |
|||
"data": [ |
|||
"view/res_partner_relation_all.xml", |
|||
'view/res_partner.xml', |
|||
'view/res_partner_relation.xml', |
|||
'view/res_partner_relation_type.xml', |
|||
'view/menu.xml', |
|||
'security/ir.model.access.csv', |
|||
], |
|||
"js": [ |
|||
], |
|||
"css": [ |
|||
], |
|||
"qweb": [ |
|||
], |
|||
"auto_install": False, |
|||
"installable": True, |
|||
"external_dependencies": { |
|||
'python': [], |
|||
}, |
|||
} |
@ -0,0 +1,268 @@ |
|||
# Translation of OpenERP Server. |
|||
# This file contains the translation of the following modules: |
|||
# * partner_relations |
|||
# |
|||
msgid "" |
|||
msgstr "" |
|||
"Project-Id-Version: OpenERP Server 7.0\n" |
|||
"Report-Msgid-Bugs-To: \n" |
|||
"POT-Creation-Date: 2014-06-19 13:02+0000\n" |
|||
"PO-Revision-Date: 2014-06-19 13:02+0000\n" |
|||
"Last-Translator: <>\n" |
|||
"Language-Team: \n" |
|||
"MIME-Version: 1.0\n" |
|||
"Content-Type: text/plain; charset=UTF-8\n" |
|||
"Content-Transfer-Encoding: \n" |
|||
"Plural-Forms: \n" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,active:0 |
|||
#: field:res.partner.relation.all,active:0 |
|||
msgid "Active" |
|||
msgstr "Actief" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_all |
|||
msgid "All (non-inverse + inverse) relations between partners" |
|||
msgstr "Alle (non-inverse + inverse) koppelingen tussen relaties" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_type_selection |
|||
msgid "All relation types" |
|||
msgstr "Alle relaties" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,relation_all_ids:0 |
|||
msgid "All relations with current partner" |
|||
msgstr "Alle koppelingen met huidige relatie" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,this_partner_id:0 |
|||
msgid "Current partner" |
|||
msgstr "Huidige relatie" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,partner_category_this:0 |
|||
#: field:res.partner.relation.type.selection,search_partner_category_this:0 |
|||
msgid "Current record's category" |
|||
msgstr "Categorie van huidige record" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,contact_type_this:0 |
|||
msgid "Current record's partner type" |
|||
msgstr "Relatietype van huidige record" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,date_end:0 |
|||
#: field:res.partner.relation.all,date_end:0 |
|||
msgid "Ending date" |
|||
msgstr "Einddatum" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_id:0 |
|||
msgid "Has relation of type" |
|||
msgstr "Heeft koppeling" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_partner_id:0 |
|||
msgid "Has relation with" |
|||
msgstr "Heeft koppeling met" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,name_inverse:0 |
|||
msgid "Inverse name" |
|||
msgstr "Inverse naam" |
|||
|
|||
#. module: partner_relations |
|||
#: selection:res.partner.relation.all,record_type:0 |
|||
#: selection:res.partner.relation.type.selection,record_type:0 |
|||
msgid "Inverse type" |
|||
msgstr "Inverse type" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,left_partner_id:0 |
|||
msgid "Left partner" |
|||
msgstr "Linker relatie" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,partner_category_left:0 |
|||
msgid "Left partner category" |
|||
msgstr "Linker relatielabel" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,contact_type_left:0 |
|||
msgid "Left partner type" |
|||
msgstr "Linker relatietype" |
|||
|
|||
#. module: partner_relations |
|||
#: view:res.partner.relation.type:0 |
|||
msgid "Left side of relation" |
|||
msgstr "Linkerkant van de koppeling" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,name:0 |
|||
#: field:res.partner.relation.type.selection,name:0 |
|||
msgid "Name" |
|||
msgstr "Naam" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,other_partner_id:0 |
|||
msgid "Other partner" |
|||
msgstr "Andere relatie" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,partner_category_other:0 |
|||
msgid "Other record's category" |
|||
msgstr "Categorie andere record" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,contact_type_other:0 |
|||
msgid "Other record's partner type" |
|||
msgstr "Type andere record" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_type |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_type_inverse |
|||
msgid "Parter relation type" |
|||
msgstr "Type relatiekoppeling" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner |
|||
#: field:res.partner.relation,partner_id_display:0 |
|||
msgid "Partner" |
|||
msgstr "Relatie" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_category |
|||
msgid "Partner Categories" |
|||
msgstr "Relatie Categorieën" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation |
|||
#: view:res.partner.relation:0 |
|||
#: view:res.partner.relation.type:0 |
|||
msgid "Partner relation" |
|||
msgstr "Relatiekoppeling" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.actions.act_window,name:partner_relations.action_res_partner_relation_type |
|||
#: model:ir.ui.menu,name:partner_relations.menu_res_partner_relation_type |
|||
msgid "Partner relations" |
|||
msgstr "Relatiekoppelingen" |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "Partners cannot have a relation with themselves." |
|||
msgstr "Relaties kunnen niet aan zichzelf gekoppeld worden." |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,record_type:0 |
|||
#: field:res.partner.relation.type.selection,record_type:0 |
|||
msgid "Record type" |
|||
msgstr "Record type" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,relation_id:0 |
|||
msgid "Relation" |
|||
msgstr "Koppeling" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,is_relation_expired:0 |
|||
msgid "Relation is expired" |
|||
msgstr "Koppeling is afgelopen" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,is_relation_future:0 |
|||
msgid "Relation is in the future" |
|||
msgstr "Koppeling is in de toekomst" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,type_id:0 |
|||
msgid "Relation type" |
|||
msgstr "Koppelingstype" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_date:0 |
|||
msgid "Relation valid" |
|||
msgstr "Datum koppeling" |
|||
|
|||
#. module: partner_relations |
|||
#: view:res.partner:0 |
|||
#: field:res.partner,relation_ids:0 |
|||
msgid "Relations" |
|||
msgstr "Koppelingen" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,right_partner_id:0 |
|||
msgid "Right partner" |
|||
msgstr "Rechter relatie" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,partner_category_right:0 |
|||
msgid "Right partner category" |
|||
msgstr "Rechter relatielabel" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,contact_type_right:0 |
|||
msgid "Right partner type" |
|||
msgstr "Rechter relatietype" |
|||
|
|||
#. module: partner_relations |
|||
#: view:res.partner.relation.type:0 |
|||
msgid "Right side of relation" |
|||
msgstr "Rechterkant van de koppeling" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,date_start:0 |
|||
#: field:res.partner.relation.all,date_start:0 |
|||
msgid "Starting date" |
|||
msgstr "Begindatum" |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "The left partner is not applicable for this relation type." |
|||
msgstr "De linker relatie is niet geldig voor dit type koppeling." |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "The right partner is not applicable for this relation type." |
|||
msgstr "De rechter relatie is niet geldig voor dit type koppeling." |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "The starting date cannot be after the ending date." |
|||
msgstr "De begindatum mag niet na de einddatum liggen." |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,type_id:0 |
|||
#: field:res.partner.relation,type_selection_id:0 |
|||
#: selection:res.partner.relation.all,record_type:0 |
|||
#: selection:res.partner.relation.type.selection,record_type:0 |
|||
#: field:res.partner.relation.type.selection,type_id:0 |
|||
msgid "Type" |
|||
msgstr "Type" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.category,only_for_organisation:0 |
|||
msgid "Valid for organisation only" |
|||
msgstr "Alleen geldig voor organisaties" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.category,only_for_person:0 |
|||
msgid "Valid for person only" |
|||
msgstr "Alleen geldig voor personen" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_partner_category_id:0 |
|||
msgid "Has relation with a partner in category" |
|||
msgstr "Heeft koppeling met relatie van de categorie" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.actions.act_window,name:partner_relations.action_show_partner_relations |
|||
msgid "Show partner's relations" |
|||
msgstr "Toon relatiekoppelingen" |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "The same relation can't be created twice." |
|||
msgstr "Dezelfde koppeling kan niet dubbel aan worden gemaakt." |
@ -0,0 +1,277 @@ |
|||
# Translation of OpenERP Server. |
|||
# This file contains the translation of the following modules: |
|||
# * partner_relations |
|||
# |
|||
msgid "" |
|||
msgstr "" |
|||
"Project-Id-Version: OpenERP Server 7.0\n" |
|||
"Report-Msgid-Bugs-To: \n" |
|||
"POT-Creation-Date: 2014-07-01 08:52+0000\n" |
|||
"PO-Revision-Date: 2014-07-01 08:52+0000\n" |
|||
"Last-Translator: <>\n" |
|||
"Language-Team: \n" |
|||
"MIME-Version: 1.0\n" |
|||
"Content-Type: text/plain; charset=UTF-8\n" |
|||
"Content-Transfer-Encoding: \n" |
|||
"Plural-Forms: \n" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,active:0 |
|||
#: field:res.partner.relation.all,active:0 |
|||
msgid "Active" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_all |
|||
msgid "All (non-inverse + inverse) relations between partners" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_type_selection |
|||
msgid "All relation types" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,relation_all_ids:0 |
|||
msgid "All relations with current partner" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,this_partner_id:0 |
|||
msgid "Current partner" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,partner_category_this:0 |
|||
#: field:res.partner.relation.type.selection,search_partner_category_this:0 |
|||
msgid "Current record's category" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,contact_type_this:0 |
|||
msgid "Current record's partner type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,date_end:0 |
|||
#: field:res.partner.relation.all,date_end:0 |
|||
msgid "Ending date" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_id:0 |
|||
msgid "Has relation of type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_partner_id:0 |
|||
msgid "Has relation with" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_partner_category_id:0 |
|||
msgid "Has relation with a partner in category" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,name_inverse:0 |
|||
msgid "Inverse name" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: selection:res.partner.relation.all,record_type:0 |
|||
#: selection:res.partner.relation.type.selection,record_type:0 |
|||
msgid "Inverse type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,left_partner_id:0 |
|||
msgid "Left partner" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,partner_category_left:0 |
|||
msgid "Left partner category" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,contact_type_left:0 |
|||
msgid "Left partner type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: view:res.partner.relation.type:0 |
|||
msgid "Left side of relation" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,name:0 |
|||
#: field:res.partner.relation.type.selection,name:0 |
|||
msgid "Name" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,other_partner_id:0 |
|||
msgid "Other partner" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,partner_category_other:0 |
|||
msgid "Other record's category" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type.selection,contact_type_other:0 |
|||
msgid "Other record's partner type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_type |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation_type_inverse |
|||
msgid "Parter relation type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner |
|||
#: field:res.partner.relation,partner_id_display:0 |
|||
msgid "Partner" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_category |
|||
msgid "Partner Categories" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_res_partner_relation |
|||
#: view:res.partner.relation:0 |
|||
#: view:res.partner.relation.all:0 |
|||
#: view:res.partner.relation.type:0 |
|||
msgid "Partner relation" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.actions.act_window,name:partner_relations.action_res_partner_relation_type |
|||
#: model:ir.ui.menu,name:partner_relations.menu_res_partner_relation_type |
|||
#: view:res.partner.relation.all:0 |
|||
msgid "Partner relations" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "Partners cannot have a relation with themselves." |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,record_type:0 |
|||
#: field:res.partner.relation.type.selection,record_type:0 |
|||
msgid "Record type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,relation_id:0 |
|||
msgid "Relation" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,is_relation_expired:0 |
|||
msgid "Relation is expired" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,is_relation_future:0 |
|||
msgid "Relation is in the future" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.all,type_id:0 |
|||
#: field:res.partner.relation.all,type_selection_id:0 |
|||
msgid "Relation type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner,search_relation_date:0 |
|||
msgid "Relation valid" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: view:res.partner:0 |
|||
#: field:res.partner,relation_ids:0 |
|||
msgid "Relations" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,right_partner_id:0 |
|||
msgid "Right partner" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,partner_category_right:0 |
|||
msgid "Right partner category" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation.type,contact_type_right:0 |
|||
msgid "Right partner type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: view:res.partner.relation.type:0 |
|||
msgid "Right side of relation" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.actions.act_window,name:partner_relations.action_show_partner_relations |
|||
msgid "Show partner's relations" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,date_start:0 |
|||
#: field:res.partner.relation.all,date_start:0 |
|||
msgid "Starting date" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "The left partner is not applicable for this relation type." |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "The right partner is not applicable for this relation type." |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: sql_constraint:res.partner.relation:0 |
|||
msgid "The same relation can't be created twice." |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: constraint:res.partner.relation:0 |
|||
msgid "The starting date cannot be after the ending date." |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.relation,type_id:0 |
|||
#: field:res.partner.relation,type_selection_id:0 |
|||
#: selection:res.partner.relation.all,record_type:0 |
|||
#: selection:res.partner.relation.type.selection,record_type:0 |
|||
#: field:res.partner.relation.type.selection,type_id:0 |
|||
msgid "Type" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.category,only_for_organisation:0 |
|||
msgid "Valid for organisation only" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: field:res.partner.category,only_for_person:0 |
|||
msgid "Valid for person only" |
|||
msgstr "" |
|||
|
|||
#. module: partner_relations |
|||
#: model:ir.model,name:partner_relations.model_ir_translation |
|||
msgid "ir.translation" |
|||
msgstr "" |
|||
|
@ -0,0 +1,25 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2013 Therp BV (<http://therp.nl>). |
|||
# |
|||
# 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/>. |
|||
# |
|||
############################################################################## |
|||
import res_partner |
|||
import res_partner_relation |
|||
import res_partner_relation_type |
|||
import res_partner_relation_type_selection |
|||
from . import res_partner_relation_all |
@ -0,0 +1,238 @@ |
|||
# -*- coding: utf-8 -*- |
|||
'''Extend res.partner model''' |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2013 Therp BV (<http://therp.nl>). |
|||
# |
|||
# 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/>. |
|||
# |
|||
############################################################################## |
|||
import time |
|||
from openerp.osv import orm, fields |
|||
from openerp.osv.expression import is_leaf |
|||
from openerp.tools import DEFAULT_SERVER_DATE_FORMAT |
|||
|
|||
|
|||
class ResPartner(orm.Model): |
|||
_inherit = 'res.partner' |
|||
|
|||
def _get_relation_ids( |
|||
self, cr, uid, ids, dummy_name, dummy_arg, context=None): |
|||
if context is None: |
|||
context = {} |
|||
|
|||
#TODO: do a permission test on returned ids |
|||
cr.execute( |
|||
'''select id, left_partner_id, right_partner_id |
|||
from res_partner_relation |
|||
where (left_partner_id in %s or right_partner_id in %s)''' + |
|||
' order by ' + self.pool['res.partner.relation']._order, |
|||
(tuple(ids), tuple(ids)) |
|||
) |
|||
result = dict([(i, []) for i in ids]) |
|||
for row in cr.fetchall(): |
|||
if row[1] in result: |
|||
result[row[1]].append(row[0]) |
|||
if row[2] in result: |
|||
result[row[2]].append(row[0]) |
|||
return result |
|||
|
|||
def _set_relation_ids( |
|||
self, cr, uid, ids, dummy_name, field_value, dummy_arg, |
|||
context=None): |
|||
if context is None: |
|||
context = {} |
|||
relation_obj = self.pool.get('res.partner.relation') |
|||
context2 = self._update_context(context, ids) |
|||
for value in field_value: |
|||
if value[0] == 0: |
|||
relation_obj.create(cr, uid, value[2], context=context2) |
|||
if value[0] == 1: |
|||
relation_obj.write( |
|||
cr, uid, value[1], value[2], context=context2) |
|||
if value[0] == 2: |
|||
relation_obj.unlink(cr, uid, value[1], context=context2) |
|||
|
|||
def _search_relation_id( |
|||
self, cr, uid, dummy_obj, name, args, context=None): |
|||
result = [] |
|||
for arg in args: |
|||
if isinstance(arg, tuple) and arg[0] == name: |
|||
if arg[1] != '=': |
|||
continue |
|||
|
|||
type_id, is_inverse = self\ |
|||
.pool['res.partner.relation.type.selection']\ |
|||
.get_type_from_selection_id(cr, uid, arg[2]) |
|||
|
|||
result.extend([ |
|||
'&', |
|||
('relation_all_ids.type_id', '=', type_id), |
|||
('relation_all_ids.record_type', '=', |
|||
'b' if is_inverse else 'a') |
|||
]) |
|||
|
|||
return result |
|||
|
|||
def _search_relation_date(self, cr, uid, obj, name, args, context=None): |
|||
result = [] |
|||
for arg in args: |
|||
if isinstance(arg, tuple) and arg[0] == name: |
|||
#TODO: handle {<,>}{,=} |
|||
if arg[1] != '=': |
|||
continue |
|||
|
|||
result.extend([ |
|||
'&', |
|||
'|', |
|||
('relation_all_ids.date_start', '=', False), |
|||
('relation_all_ids.date_start', '<=', arg[2]), |
|||
'|', |
|||
('relation_all_ids.date_end', '=', False), |
|||
('relation_all_ids.date_end', '>=', arg[2]), |
|||
]) |
|||
|
|||
return result |
|||
|
|||
def _search_related_partner_id( |
|||
self, cr, uid, dummy_obj, name, args, context=None): |
|||
result = [] |
|||
for arg in args: |
|||
if isinstance(arg, tuple) and arg[0] == name: |
|||
result.append( |
|||
( |
|||
'relation_all_ids.other_partner_id', |
|||
arg[1], |
|||
arg[2], |
|||
)) |
|||
|
|||
return result |
|||
|
|||
def _search_related_partner_category_id( |
|||
self, cr, uid, dummy_obj, name, args, context=None): |
|||
result = [] |
|||
for arg in args: |
|||
if isinstance(arg, tuple) and arg[0] == name: |
|||
result.append( |
|||
( |
|||
'relation_all_ids.other_partner_id.category_id', |
|||
arg[1], |
|||
arg[2], |
|||
)) |
|||
|
|||
return result |
|||
|
|||
_columns = { |
|||
'relation_ids': fields.function( |
|||
lambda self, *args, **kwargs: self._get_relation_ids( |
|||
*args, **kwargs), |
|||
fnct_inv=_set_relation_ids, |
|||
type='one2many', obj='res.partner.relation', |
|||
string='Relations', |
|||
selectable=False, |
|||
), |
|||
'relation_all_ids': fields.one2many( |
|||
'res.partner.relation.all', 'this_partner_id', |
|||
string='All relations with current partner', |
|||
auto_join=True, |
|||
selectable=False, |
|||
), |
|||
'search_relation_id': fields.function( |
|||
lambda self, cr, uid, ids, *args: dict([ |
|||
(i, False) for i in ids]), |
|||
fnct_search=_search_relation_id, |
|||
string='Has relation of type', |
|||
type='many2one', obj='res.partner.relation.type.selection' |
|||
), |
|||
'search_relation_partner_id': fields.function( |
|||
lambda self, cr, uid, ids, *args: dict([ |
|||
(i, False) for i in ids]), |
|||
fnct_search=_search_related_partner_id, |
|||
string='Has relation with', |
|||
type='many2one', obj='res.partner' |
|||
), |
|||
'search_relation_date': fields.function( |
|||
lambda self, cr, uid, ids, *args: dict([ |
|||
(i, False) for i in ids]), |
|||
fnct_search=_search_relation_date, |
|||
string='Relation valid', type='date' |
|||
), |
|||
'search_relation_partner_category_id': fields.function( |
|||
lambda self, cr, uid, ids, *args: dict([ |
|||
(i, False) for i in ids]), |
|||
fnct_search=_search_related_partner_category_id, |
|||
string='Has relation with a partner in category', |
|||
type='many2one', obj='res.partner.category' |
|||
), |
|||
} |
|||
|
|||
def copy_data(self, cr, uid, id, default=None, context=None): |
|||
if default is None: |
|||
default = {} |
|||
default.setdefault('relation_ids', []) |
|||
default.setdefault('relation_all_ids', []) |
|||
return super(ResPartner, self).copy_data(cr, uid, id, default=default, |
|||
context=context) |
|||
|
|||
def search(self, cr, uid, args, offset=0, limit=None, order=None, |
|||
context=None, count=False): |
|||
if context is None: |
|||
context = {} |
|||
#inject searching for current relation date if we search for relation |
|||
#properties and no explicit date was given |
|||
date_args = [] |
|||
for arg in args: |
|||
if is_leaf(arg) and arg[0].startswith('search_relation'): |
|||
if arg[0] == 'search_relation_date': |
|||
date_args = [] |
|||
break |
|||
if not date_args: |
|||
date_args = [ |
|||
('search_relation_date', '=', time.strftime( |
|||
DEFAULT_SERVER_DATE_FORMAT))] |
|||
|
|||
#because of auto_join, we have to do the active test by hand |
|||
active_args = [] |
|||
if context.get('active_test', True): |
|||
for arg in args: |
|||
if is_leaf(arg) and\ |
|||
arg[0].startswith('search_relation'): |
|||
active_args = [('relation_all_ids.active', '=', True)] |
|||
break |
|||
|
|||
return super(ResPartner, self).search( |
|||
cr, uid, args + date_args + active_args, offset=offset, |
|||
limit=limit, order=order, context=context, count=count) |
|||
|
|||
def read( |
|||
self, cr, uid, ids, fields=None, context=None, |
|||
load='_classic_read'): |
|||
return super(ResPartner, self).read( |
|||
cr, uid, ids, fields=fields, |
|||
context=self._update_context(context, ids)) |
|||
|
|||
def write(self, cr, uid, ids, vals, context=None): |
|||
return super(ResPartner, self).write( |
|||
cr, uid, ids, vals, context=self._update_context(context, ids)) |
|||
|
|||
def _update_context(self, context, ids): |
|||
if context is None: |
|||
context = {} |
|||
ids = ids if isinstance(ids, list) else [ids] if ids else [] |
|||
result = context.copy() |
|||
result.setdefault('active_id', ids[0] if ids else None) |
|||
result.setdefault('active_ids', ids) |
|||
result.setdefault('active_model', self._name) |
|||
return result |
@ -0,0 +1,330 @@ |
|||
# -*- coding: utf-8 -*- |
|||
'''Define model res.partner.relation''' |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2013 Therp BV (<http://therp.nl>). |
|||
# |
|||
# 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.orm import Model |
|||
from openerp.osv import fields |
|||
from openerp.tools.translate import _ |
|||
|
|||
|
|||
class ResPartnerRelation(Model): |
|||
'''Model res.partner.relation is used to describe all links or relations |
|||
between partners in the database. |
|||
|
|||
In many parts of the code we have to know whether the active partner is |
|||
the left partner, or the right partner. If the active partner is the |
|||
right partner we have to show the inverse name. |
|||
|
|||
Because the active partner is crucial for the working of partner |
|||
relationships, we make sure on the res.partner model that the partner id |
|||
is set in the context where needed. |
|||
''' |
|||
_name = 'res.partner.relation' |
|||
_description = 'Partner relation' |
|||
_order = 'active desc, date_start desc, date_end desc' |
|||
|
|||
def _on_right_partner(self, cr, uid, right_partner_id, context=None): |
|||
'''Determine wether functions are called in a situation where the |
|||
active partner is the right partner. Default False! |
|||
''' |
|||
if (context and 'active_ids' in context |
|||
and right_partner_id in context.get('active_ids', [])): |
|||
return True |
|||
return False |
|||
|
|||
def _correct_vals(self, cr, uid, vals, context=None): |
|||
'''Fill type and left and right partner id, according to wether |
|||
we have a normal relation type or an inverse relation type''' |
|||
vals = vals.copy() |
|||
# If type_selection_id ends in 1, it is a reverse relation type |
|||
if 'type_selection_id' in vals: |
|||
prts_model = self.pool['res.partner.relation.type.selection'] |
|||
type_selection_id = vals['type_selection_id'] |
|||
(type_id, is_reverse) = ( |
|||
prts_model.get_type_from_selection_id( |
|||
cr, uid, type_selection_id)) |
|||
vals['type_id'] = type_id |
|||
if context.get('active_id'): |
|||
if is_reverse: |
|||
vals['right_partner_id'] = context['active_id'] |
|||
else: |
|||
vals['left_partner_id'] = context['active_id'] |
|||
if vals.get('partner_id_display'): |
|||
if is_reverse: |
|||
vals['left_partner_id'] = vals['partner_id_display'] |
|||
else: |
|||
vals['right_partner_id'] = vals['partner_id_display'] |
|||
return vals |
|||
|
|||
def _get_computed_fields( |
|||
self, cr, uid, ids, field_names, arg, context=None): |
|||
'''Return a dictionary of dictionaries, with for every partner for |
|||
ids, the computed values.''' |
|||
def get_values(this, dummy_field_names, dummy_arg, context=None): |
|||
'''Get computed values for record''' |
|||
values = {} |
|||
on_right_partner = self._on_right_partner( |
|||
cr, uid, this.right_partner_id.id, context=context) |
|||
# type_selection_id |
|||
values['type_selection_id'] = ( |
|||
((this.type_id.id) * 10) + (on_right_partner and 1 or 0)) |
|||
# partner_id_display |
|||
values['partner_id_display'] = ( |
|||
on_right_partner and this.left_partner_id.id |
|||
or this.right_partner_id.id |
|||
) |
|||
# is_relation_expired |
|||
today = fields.date.context_today(self, cr, uid, context=context) |
|||
values['is_relation_expired'] = ( |
|||
this.date_end and (this.date_end < today)) |
|||
# is_relation_future |
|||
values['is_relation_future'] = this.date_start > today |
|||
return values |
|||
|
|||
return dict([ |
|||
(this.id, get_values(this, field_names, arg, context=context)) |
|||
for this in self.browse(cr, uid, ids, context=context) |
|||
]) |
|||
|
|||
def write(self, cr, uid, ids, vals, context=None): |
|||
'''Override write to correct values, before being stored.''' |
|||
vals = self._correct_vals(cr, uid, vals, context=context) |
|||
return super(ResPartnerRelation, self).write( |
|||
cr, uid, ids, vals, context=context) |
|||
|
|||
def create(self, cr, uid, vals, context=None): |
|||
'''Override create to correct values, before being stored.''' |
|||
vals = self._correct_vals(cr, uid, vals, context=context) |
|||
return super(ResPartnerRelation, self).create( |
|||
cr, uid, vals, context=context) |
|||
|
|||
def on_change_type_selection_id( |
|||
self, cr, uid, dummy_ids, type_selection_id, context=None): |
|||
'''Set domain on partner_id_display, when selection a relation type''' |
|||
result = { |
|||
'domain': {'partner_id_display': []}, |
|||
'value': {'type_id': False} |
|||
} |
|||
if not type_selection_id: |
|||
return result |
|||
prts_model = self.pool['res.partner.relation.type.selection'] |
|||
type_model = self.pool['res.partner.relation.type'] |
|||
(type_id, is_reverse) = ( |
|||
prts_model.get_type_from_selection_id( |
|||
cr, uid, type_selection_id) |
|||
) |
|||
result['value']['type_id'] = type_id |
|||
type_obj = type_model.browse(cr, uid, type_id, context=context) |
|||
partner_domain = [] |
|||
check_contact_type = type_obj.contact_type_right |
|||
check_partner_category = ( |
|||
type_obj.partner_category_right and |
|||
type_obj.partner_category_right.id |
|||
) |
|||
if is_reverse: |
|||
# partner_id_display is left partner |
|||
check_contact_type = type_obj.contact_type_left |
|||
check_partner_category = ( |
|||
type_obj.partner_category_left and |
|||
type_obj.partner_category_left.id |
|||
) |
|||
if check_contact_type == 'c': |
|||
partner_domain.append(('is_company', '=', True)) |
|||
if check_contact_type == 'p': |
|||
partner_domain.append(('is_company', '=', False)) |
|||
if check_partner_category: |
|||
partner_domain.append( |
|||
('category_id', 'child_of', check_partner_category)) |
|||
result['domain']['partner_id_display'] = partner_domain |
|||
return result |
|||
|
|||
_columns = { |
|||
'left_partner_id': fields.many2one( |
|||
'res.partner', string='Left partner', required=True, |
|||
auto_join=True, ondelete='cascade'), |
|||
'right_partner_id': fields.many2one( |
|||
'res.partner', string='Right partner', required=True, |
|||
auto_join=True, ondelete='cascade'), |
|||
'type_id': fields.many2one( |
|||
'res.partner.relation.type', string='Type', required=True, |
|||
auto_join=True), |
|||
'date_start': fields.date('Starting date'), |
|||
'date_end': fields.date('Ending date'), |
|||
'type_selection_id': fields.function( |
|||
_get_computed_fields, |
|||
multi="computed_fields", |
|||
fnct_inv=lambda *args: None, |
|||
type='many2one', obj='res.partner.relation.type.selection', |
|||
string='Type', |
|||
), |
|||
'partner_id_display': fields.function( |
|||
_get_computed_fields, |
|||
multi="computed_fields", |
|||
fnct_inv=lambda *args: None, |
|||
type='many2one', obj='res.partner', |
|||
string='Partner' |
|||
), |
|||
'is_relation_expired': fields.function( |
|||
_get_computed_fields, |
|||
multi="computed_fields", |
|||
type='boolean', |
|||
method=True, |
|||
string='Relation is expired', |
|||
), |
|||
'is_relation_future': fields.function( |
|||
_get_computed_fields, |
|||
multi="computed_fields", |
|||
type='boolean', |
|||
method=True, |
|||
string='Relation is in the future', |
|||
), |
|||
'active': fields.boolean('Active'), |
|||
} |
|||
|
|||
_defaults = { |
|||
'active': True, |
|||
} |
|||
|
|||
def _check_dates(self, cr, uid, ids, context=None): |
|||
'''End date should not be before start date, if noth filled''' |
|||
for line in self.browse(cr, uid, ids, context=context): |
|||
if line.date_start and line.date_end: |
|||
if line.date_start > line.date_end: |
|||
return False |
|||
return True |
|||
|
|||
def _check_partner_type_left(self, cr, uid, ids, context=None): |
|||
'''Check left partner for required company or person''' |
|||
for this in self.browse(cr, uid, ids, context=context): |
|||
ptype = this.type_id.contact_type_left |
|||
company = this.left_partner_id.is_company |
|||
if (ptype == 'c' and not company) or (ptype == 'p' and company): |
|||
return False |
|||
return True |
|||
|
|||
def _check_partner_type_right(self, cr, uid, ids, context=None): |
|||
'''Check right partner for required company or person''' |
|||
for this in self.browse(cr, uid, ids, context=context): |
|||
ptype = this.type_id.contact_type_right |
|||
company = this.right_partner_id.is_company |
|||
if (ptype == 'c' and not company) or (ptype == 'p' and company): |
|||
return False |
|||
return True |
|||
|
|||
def _check_not_with_self(self, cr, uid, ids, context=None): |
|||
'''Not allowed to link partner to same partner''' |
|||
for this in self.browse(cr, uid, ids, context=context): |
|||
if this.left_partner_id == this.right_partner_id: |
|||
return False |
|||
return True |
|||
|
|||
def _check_relation_uniqueness(self, cr, uid, ids, context=None): |
|||
'''Forbid multiple active relations of the same type between the same |
|||
partners''' |
|||
for this in self.browse(cr, uid, ids, context=context): |
|||
if not this.active: |
|||
continue |
|||
if self.search( |
|||
cr, uid, |
|||
[ |
|||
('type_id', '=', this.type_id.id), |
|||
('active', '=', True), |
|||
('id', '!=', this.id), |
|||
('left_partner_id', '=', this.left_partner_id.id), |
|||
('right_partner_id', '=', this.right_partner_id.id), |
|||
], |
|||
context=context): |
|||
return False |
|||
|
|||
return True |
|||
|
|||
_constraints = [ |
|||
( |
|||
_check_dates, |
|||
'The starting date cannot be after the ending date.', |
|||
['date_start', 'date_end'] |
|||
), |
|||
( |
|||
_check_partner_type_left, |
|||
'The left partner is not applicable for this relation type.', |
|||
['left_partner_id', 'type_id'] |
|||
), |
|||
( |
|||
_check_partner_type_right, |
|||
'The right partner is not applicable for this relation type.', |
|||
['right_partner_id', 'type_id'] |
|||
), |
|||
( |
|||
_check_not_with_self, |
|||
'Partners cannot have a relation with themselves.', |
|||
['left_partner_id', 'right_partner_id'] |
|||
), |
|||
( |
|||
_check_relation_uniqueness, |
|||
"The same relation can't be created twice.", |
|||
['left_partner_id', 'right_partner_id', 'active'] |
|||
) |
|||
] |
|||
|
|||
def get_action_related_partners(self, cr, uid, ids, context=None): |
|||
'''return a window action showing a list of partners taking part in the |
|||
relations names by ids. Context key 'partner_relations_show_side' |
|||
determines if we show 'left' side, 'right' side or 'all' (default) |
|||
partners. |
|||
If active_model is res.partner.relation.all, left=this and |
|||
right=other''' |
|||
if context is None: |
|||
context = {} |
|||
|
|||
field_names = {} |
|||
|
|||
if context.get('active_model', self._name) == self._name: |
|||
field_names = { |
|||
'left': ['left'], |
|||
'right': ['right'], |
|||
'all': ['left', 'right'] |
|||
} |
|||
elif context.get('active_model') == 'res.partner.relation.all': |
|||
field_names = { |
|||
'left': ['this'], |
|||
'right': ['other'], |
|||
'all': ['this', 'other'] |
|||
} |
|||
else: |
|||
assert False, 'Unknown active_model!' |
|||
|
|||
partner_ids = [] |
|||
field_names = field_names[ |
|||
context.get('partner_relations_show_side', 'all')] |
|||
field_names = ['%s_partner_id' % n for n in field_names] |
|||
|
|||
for relation in self.pool[context.get('active_model')].read( |
|||
cr, uid, ids, context=context, load='_classic_write'): |
|||
for name in field_names: |
|||
partner_ids.append(relation[name]) |
|||
|
|||
return { |
|||
'name': _('Related partners'), |
|||
'type': 'ir.actions.act_window', |
|||
'res_model': 'res.partner', |
|||
'domain': [('id', 'in', partner_ids)], |
|||
'views': [(False, 'tree'), (False, 'form')], |
|||
'view_type': 'form' |
|||
} |
@ -0,0 +1,88 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2014 Therp BV (<http://therp.nl>). |
|||
# |
|||
# 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.orm import Model |
|||
from openerp.osv import fields |
|||
from openerp.tools import drop_view_if_exists |
|||
from res_partner_relation_type_selection import ResPartnerRelationTypeSelection |
|||
|
|||
|
|||
class ResPartnerRelationAll(Model): |
|||
_auto = False |
|||
_log_access = False |
|||
_name = 'res.partner.relation.all' |
|||
_description = 'All (non-inverse + inverse) relations between partners' |
|||
|
|||
def _auto_init(self, cr, context=None): |
|||
drop_view_if_exists(cr, self._table) |
|||
cr.execute( |
|||
'''create or replace view %s as |
|||
select |
|||
id * 10 as id, |
|||
id as relation_id, |
|||
type_id, |
|||
cast('a' as char(1)) as record_type, |
|||
left_partner_id as this_partner_id, |
|||
right_partner_id as other_partner_id, |
|||
date_start, |
|||
date_end, |
|||
active, |
|||
type_id * 10 as type_selection_id |
|||
from res_partner_relation |
|||
union select |
|||
id * 10 + 1, |
|||
id, |
|||
type_id, |
|||
cast('b' as char(1)), |
|||
right_partner_id, |
|||
left_partner_id, |
|||
date_start, |
|||
date_end, |
|||
active, |
|||
type_id * 10 + 1 |
|||
from res_partner_relation''' % self._table) |
|||
|
|||
return super(ResPartnerRelationAll, self)._auto_init( |
|||
cr, context=context) |
|||
|
|||
_columns = { |
|||
'record_type': fields.selection( |
|||
ResPartnerRelationTypeSelection._RECORD_TYPES, 'Record type'), |
|||
'relation_id': fields.many2one( |
|||
'res.partner.relation', 'Relation'), |
|||
'type_id': fields.many2one( |
|||
'res.partner.relation.type', 'Relation type'), |
|||
'type_selection_id': fields.many2one( |
|||
'res.partner.relation.type.selection', 'Relation type'), |
|||
'this_partner_id': fields.many2one('res.partner', 'Current partner'), |
|||
'other_partner_id': fields.many2one('res.partner', 'Other partner'), |
|||
'date_start': fields.date('Starting date'), |
|||
'date_end': fields.date('Ending date'), |
|||
'active': fields.boolean('Active'), |
|||
} |
|||
|
|||
def name_get(self, cr, uid, ids, context=None): |
|||
return dict([ |
|||
(this.id, '%s %s %s' % ( |
|||
this.this_partner_id.name, |
|||
this.type_selection_id.name_get()[0][1], |
|||
this.other_partner_id.name, |
|||
)) |
|||
for this in self.browse(cr, uid, ids, context=context)]) |
@ -0,0 +1,48 @@ |
|||
# -*- coding: utf-8 -*- |
|||
'''Define model res.partner.relation.type''' |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2013 Therp BV (<http://therp.nl>). |
|||
# |
|||
# 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.orm import Model |
|||
from openerp.osv import fields |
|||
|
|||
|
|||
class ResPartnerRelationType(Model): |
|||
'''Model that defines relation types that might exist between partners''' |
|||
_name = 'res.partner.relation.type' |
|||
_description = 'Parter relation type' |
|||
_order = 'name' |
|||
|
|||
def _get_partner_types(self, cr, uid, context=None): |
|||
return (('c', 'Company'), ('p', 'Person'),) |
|||
|
|||
_columns = { |
|||
'name': fields.char( |
|||
'Name', size=128, required=True, translate=True), |
|||
'name_inverse': fields.char( |
|||
'Inverse name', size=128, required=True, translate=True), |
|||
'contact_type_left': fields.selection( |
|||
_get_partner_types, 'Left partner type'), |
|||
'contact_type_right': fields.selection( |
|||
_get_partner_types, 'Right partner type'), |
|||
'partner_category_left': fields.many2one( |
|||
'res.partner.category', 'Left partner category'), |
|||
'partner_category_right': fields.many2one( |
|||
'res.partner.category', 'Right partner category'), |
|||
} |
@ -0,0 +1,167 @@ |
|||
# -*- coding: UTF-8 -*- |
|||
''' |
|||
Created on 23 may 2014 |
|||
|
|||
@author: Ronald Portier, Therp |
|||
|
|||
rportier@therp.nl |
|||
http://www.therp.nl |
|||
|
|||
For the model defined here _auto is set to False to prevent creating a |
|||
database file. All i/o operations are overridden to use a sql SELECT that |
|||
takes data from res_partner_connection_type where each type is included in the |
|||
result set twice, so it appears that the connection type and the inverse |
|||
type are separate records.. |
|||
|
|||
The original function _auto_init is still called because this function |
|||
normally (if _auto == True) not only creates the db tables, but it also takes |
|||
care of registering all fields in ir_model_fields. This is needed to make |
|||
the field labels translatable. |
|||
|
|||
example content for last lines of _statement: |
|||
select id, record_type, |
|||
customer_id, customer_name, customer_city, customer_zip, customer_street, |
|||
caller_id, caller_name, caller_phone, caller_fax, caller_email |
|||
from FULL_LIST as ResPartnerRelationTypeSelection where record_type = 'c' |
|||
ORDER BY ResPartnerRelationTypeSelection.customer_name asc, |
|||
ResPartnerRelationTypeSelection.caller_name asc; |
|||
|
|||
''' |
|||
from openerp.osv import fields |
|||
from openerp.osv import orm |
|||
from openerp.tools import drop_view_if_exists |
|||
from openerp.addons.partner_relations.model.res_partner_relation_type\ |
|||
import ResPartnerRelationType |
|||
|
|||
|
|||
class ResPartnerRelationTypeSelection(orm.Model): |
|||
'''Virtual relation types''' |
|||
|
|||
_RECORD_TYPES = [ |
|||
('a', 'Type'), |
|||
('b', 'Inverse type'), |
|||
] |
|||
|
|||
_auto = False # Do not try to create table in _auto_init(..) |
|||
_log_access = False |
|||
|
|||
def get_type_from_selection_id(self, cr, uid, selection_id): |
|||
'''Selection id ic computed from id of underlying type and the |
|||
kind of record. This function does the inverse computation to give |
|||
back the original type id, and about the record type.''' |
|||
type_id = selection_id / 10 |
|||
is_reverse = (selection_id % 10) > 0 |
|||
return (type_id, is_reverse) |
|||
|
|||
def _auto_init(self, cr, context=None): |
|||
drop_view_if_exists(cr, self._table) |
|||
#TODO: we lose field value's translations here. |
|||
#probably we need to patch ir_translation.get_source for that |
|||
#to get res_partner_relation_type's translations |
|||
cr.execute( |
|||
'''create or replace view %s as |
|||
select |
|||
id * 10 as id, |
|||
id as type_id, |
|||
cast('a' as char(1)) as record_type, |
|||
name as name, |
|||
contact_type_left as contact_type_this, |
|||
contact_type_right as contact_type_other, |
|||
partner_category_left as partner_category_this, |
|||
partner_category_right as partner_category_other |
|||
from res_partner_relation_type |
|||
union select |
|||
id * 10 + 1, |
|||
id, |
|||
cast('b' as char(1)), |
|||
name_inverse, |
|||
contact_type_right, |
|||
contact_type_left, |
|||
partner_category_right, |
|||
partner_category_left |
|||
from res_partner_relation_type''' % self._table) |
|||
|
|||
return super(ResPartnerRelationTypeSelection, self)._auto_init( |
|||
cr, context=context) |
|||
|
|||
def _search_partner_category_this(self, cr, uid, obj, field_name, args, |
|||
context=None): |
|||
category_ids = [] |
|||
|
|||
for arg in args: |
|||
if isinstance(arg, tuple) and arg[0] == field_name\ |
|||
and (arg[1] == '=' or arg[1] == 'in'): |
|||
#TODO don't we have an api function to eval that? |
|||
for delta in arg[2]: |
|||
if delta[0] == 6: |
|||
category_ids.extend(delta[2]) |
|||
|
|||
if category_ids: |
|||
return [ |
|||
'|', |
|||
('partner_category_this', '=', False), |
|||
('partner_category_this', 'in', category_ids), |
|||
] |
|||
else: |
|||
return [('partner_category_this', '=', False)] |
|||
|
|||
_name = 'res.partner.relation.type.selection' |
|||
_description = 'All relation types' |
|||
_foreign_keys = [] |
|||
_columns = { |
|||
'record_type': fields.selection(_RECORD_TYPES, 'Record type', size=16), |
|||
'type_id': fields.integer('Type'), |
|||
'name': fields.char('Name', size=64), |
|||
'contact_type_this': fields.selection( |
|||
ResPartnerRelationType._get_partner_types.im_func, |
|||
'Current record\'s partner type'), |
|||
'contact_type_other': fields.selection( |
|||
ResPartnerRelationType._get_partner_types.im_func, |
|||
'Other record\'s partner type'), |
|||
'partner_category_this': fields.many2one( |
|||
'res.partner.category', 'Current record\'s category'), |
|||
'partner_category_other': fields.many2one( |
|||
'res.partner.category', 'Other record\'s category'), |
|||
#search field to handle many2many deltas from the client |
|||
'search_partner_category_this': fields.function( |
|||
lambda self, cr, uid, ids, context=None: dict( |
|||
[(i, False) for i in ids]), |
|||
fnct_search=_search_partner_category_this, |
|||
type='many2many', obj='res.partner.category', |
|||
string='Current record\'s category'), |
|||
} |
|||
_order = 'name asc' |
|||
|
|||
def name_get(self, cr, uid, ids, context=None): |
|||
'translate name using translations from res.partner.relation.type' |
|||
result = super(ResPartnerRelationTypeSelection, self).name_get( |
|||
cr, uid, ids, context=context) |
|||
ir_translation = self.pool['ir.translation'] |
|||
return [ |
|||
(i, ir_translation._get_source( |
|||
cr, uid, |
|||
'res.partner.relation.type,name_inverse' |
|||
if self.get_type_from_selection_id(cr, uid, i)[1] |
|||
else 'res.partner.relation.type,name', |
|||
'model', context.get('lang'), name)) |
|||
for i, name in result] |
|||
|
|||
def name_search(self, cr, uid, name='', args=None, operator='ilike', |
|||
context=None, limit=100): |
|||
'search for translated names in res.partner.relation.type' |
|||
res_partner_relation_type = self.pool['res.partner.relation.type'] |
|||
relation_ids = res_partner_relation_type.search( |
|||
cr, uid, [('name', operator, name)], |
|||
context=context) |
|||
inverse_relation_ids = res_partner_relation_type.search( |
|||
cr, uid, [('name_inverse', operator, name)], |
|||
context=context) |
|||
all_ids = self.search( |
|||
cr, uid, |
|||
[ |
|||
('id', 'in', |
|||
map(lambda x: x * 10, relation_ids) + |
|||
map(lambda x: x * 10 + 1, inverse_relation_ids)), |
|||
] + (args or []), |
|||
context=context, limit=limit) |
|||
return self.name_get(cr, uid, all_ids, context=context) |
@ -0,0 +1,7 @@ |
|||
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink |
|||
read_res_partner_relation,access_res_partner_relation,model_res_partner_relation,,1,0,0,0 |
|||
read_res_partner_relation_all,access_res_partner_relation,model_res_partner_relation_all,,1,0,0,0 |
|||
read_res_partner_relation_type,access_res_partner_relation_type,model_res_partner_relation_type,,1,0,0,0 |
|||
read_res_partner_relation_type_selection,access_res_partner_relation_type,model_res_partner_relation_type_selection,,1,0,0,0 |
|||
crud_res_partner_relation,access_res_partner_relation,model_res_partner_relation,base.group_partner_manager,1,1,1,1 |
|||
crud_res_partner_relation_type,access_res_partner_relation_type,model_res_partner_relation_type,base.group_sale_manager,1,1,1,1 |
After Width: 90 | Height: 90 | Size: 13 KiB |
@ -0,0 +1,15 @@ |
|||
<openerp> |
|||
<data> |
|||
<act_window |
|||
id="action_res_partner_relation_type" |
|||
res_model="res.partner.relation.type" |
|||
view_mode="tree,form" |
|||
name="Partner relations" |
|||
/> |
|||
<menuitem |
|||
id="menu_res_partner_relation_type" |
|||
parent="base.menu_config_address_book" |
|||
action="action_res_partner_relation_type" |
|||
/> |
|||
</data> |
|||
</openerp> |
@ -0,0 +1,96 @@ |
|||
<openerp> |
|||
<data> |
|||
<record id="view_res_partner_filter" model="ir.ui.view"> |
|||
<field name="inherit_id" ref="base.view_res_partner_filter" /> |
|||
<field name="model">res.partner</field> |
|||
<field type="xml" name="arch"> |
|||
<data> |
|||
<field name="parent_id" position="after"> |
|||
<field name="search_relation_partner_id" /> |
|||
<field name="search_relation_id" /> |
|||
<field name="search_relation_date" /> |
|||
<field name="search_relation_partner_category_id" /> |
|||
</field> |
|||
</data> |
|||
</field> |
|||
</record> |
|||
<record id="view_partner_form" model="ir.ui.view"> |
|||
<field name="inherit_id" ref="base.view_partner_form" /> |
|||
<field name="model">res.partner</field> |
|||
<field type="xml" name="arch"> |
|||
<data> |
|||
<xpath expr="//sheet/notebook" position="inside"> |
|||
<page string="Relations"> |
|||
<field |
|||
name="relation_ids" |
|||
context="{ |
|||
'active_model': 'res.partner', |
|||
'active_id': id, 'active_ids': [id], |
|||
'active_test': False, |
|||
}" |
|||
> |
|||
<tree |
|||
editable="top" |
|||
colors="gray:is_relation_expired==True or not active;blue:is_relation_future==True" |
|||
> |
|||
<field |
|||
name="type_selection_id" |
|||
required="True" |
|||
context="{ |
|||
'parent_model': 'res.partner', |
|||
'parent_id': parent.id, |
|||
}" |
|||
domain="[ |
|||
'|', |
|||
('contact_type_this', '=', parent.is_company and 'c' or 'p'), |
|||
('contact_type_this', '=', False), |
|||
('search_partner_category_this', '=', parent.category_id), |
|||
|
|||
] |
|||
" |
|||
options="{'create': false, 'create_edit': false}" |
|||
on_change="on_change_type_selection_id(type_selection_id)" |
|||
/> |
|||
<field name="type_id" invisible="True" /> |
|||
<field |
|||
name="partner_id_display" |
|||
required="True" |
|||
attrs="{ |
|||
'readonly': [('type_selection_id','=',False)], |
|||
}" |
|||
options="{'create': false, 'create_edit': false}" |
|||
widget="many2one_clickable" |
|||
/> |
|||
<field |
|||
name="date_start" |
|||
/> |
|||
<field |
|||
name="date_end" |
|||
/> |
|||
<field |
|||
name="active" |
|||
/> |
|||
<field |
|||
name="is_relation_expired" |
|||
invisible="True" |
|||
/> |
|||
<field |
|||
name="is_relation_future" |
|||
invisible="True" |
|||
/> |
|||
</tree> |
|||
</field> |
|||
</page> |
|||
</xpath> |
|||
</data> |
|||
</field> |
|||
</record> |
|||
|
|||
<act_window id="action_show_partner_relations" |
|||
name="Show partner's relations" |
|||
src_model="res.partner" |
|||
res_model="res.partner.relation.all" |
|||
domain="[('this_partner_id', 'in', active_ids)]" |
|||
key2="client_action_multi" /> |
|||
</data> |
|||
</openerp> |
@ -0,0 +1,33 @@ |
|||
<openerp> |
|||
<data> |
|||
<record id="form_res_partner_relation" model="ir.ui.view"> |
|||
<field name="model">res.partner.relation</field> |
|||
<field type="xml" name="arch"> |
|||
<form version="7.0" string="Partner relation"> |
|||
<sheet> |
|||
<field name="left_partner_id" /> |
|||
<field name="type_id" /> |
|||
<field name="right_partner_id" /> |
|||
</sheet> |
|||
</form> |
|||
</field> |
|||
</record> |
|||
<record id="action_show_right_relation_partners" model="ir.actions.server"> |
|||
<field name="sequence" eval="5"/> |
|||
<field name="state">code</field> |
|||
<field name="type">ir.actions.server</field> |
|||
<field name="model_id" ref="model_res_partner_relation"/> |
|||
<field name="code">action = self.get_action_related_partners(cr, uid, context.get('active_ids', []), dict(context or {}, partner_relations_show_side='right'))</field> |
|||
<field name="condition">True</field> |
|||
<field name="name">Show partners</field> |
|||
</record> |
|||
<record id="action_show_right_relation_partners_value" model="ir.values"> |
|||
<field name="name">Show partners</field> |
|||
<field name="key">action</field> |
|||
<field name="key2">client_action_multi</field> |
|||
<field name="model">res.partner.relation.all</field> |
|||
<field name="value" eval="'ir.actions.server,%d' % ref('partner_relations.action_show_right_relation_partners')" /> |
|||
</record> |
|||
|
|||
</data> |
|||
</openerp> |
@ -0,0 +1,34 @@ |
|||
<?xml version="1.0" encoding="UTF-8"?> |
|||
<openerp> |
|||
<data> |
|||
<record id="tree_res_partner_relation_all" model="ir.ui.view"> |
|||
<field name="model">res.partner.relation.all</field> |
|||
<field name="arch" type="xml"> |
|||
<tree string="Partner relations"> |
|||
<field name="this_partner_id" /> |
|||
<field name="type_selection_id" /> |
|||
<field name="other_partner_id" /> |
|||
<field name="date_start" /> |
|||
<field name="date_end" /> |
|||
</tree> |
|||
</field> |
|||
</record> |
|||
<record id="form_res_partner_relation_all" model="ir.ui.view"> |
|||
<field name="model">res.partner.relation.all</field> |
|||
<field name="arch" type="xml"> |
|||
<form string="Partner relation"> |
|||
<group> |
|||
<field name="this_partner_id" /> |
|||
<field name="type_selection_id" /> |
|||
<field name="other_partner_id" /> |
|||
</group> |
|||
<group> |
|||
<field name="date_start" /> |
|||
<field name="date_end" /> |
|||
<field name="active" /> |
|||
</group> |
|||
</form> |
|||
</field> |
|||
</record> |
|||
</data> |
|||
</openerp> |
@ -0,0 +1,44 @@ |
|||
<openerp> |
|||
<data> |
|||
<record id="tree_res_partner_relation_type" model="ir.ui.view"> |
|||
<field name="model">res.partner.relation.type</field> |
|||
<field name="type">tree</field> |
|||
<field type="xml" name="arch"> |
|||
<tree version="7.0" string="Partner relation"> |
|||
<field name="name" /> |
|||
<field name="name_inverse" /> |
|||
<field name="contact_type_left" /> |
|||
<field name="contact_type_right" /> |
|||
</tree> |
|||
</field> |
|||
</record> |
|||
<record id="form_res_partner_relation_type" model="ir.ui.view"> |
|||
<field name="model">res.partner.relation.type</field> |
|||
<field name="type">form</field> |
|||
<field type="xml" name="arch"> |
|||
<form version="7.0" string="Partner relation"> |
|||
<sheet> |
|||
<group> |
|||
<group |
|||
colspan="2" col="2" |
|||
string="Left side of relation" |
|||
> |
|||
<field name="name" /> |
|||
<field name="contact_type_left" /> |
|||
<field name="partner_category_left" /> |
|||
</group> |
|||
<group |
|||
colspan="2" col="2" |
|||
string="Right side of relation" |
|||
> |
|||
<field name="name_inverse" /> |
|||
<field name="contact_type_right" /> |
|||
<field name="partner_category_right" /> |
|||
</group> |
|||
</group> |
|||
</sheet> |
|||
</form> |
|||
</field> |
|||
</record> |
|||
</data> |
|||
</openerp> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue