Browse Source
Merge pull request #1461 from acsone/12.0-mig-onchange_helper
Merge pull request #1461 from acsone/12.0-mig-onchange_helper
[12.0][MIG] onchange helperpull/1581/head
Laurent Mignon (ACSONE)
6 years ago
committed by
GitHub
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
19 changed files with 419 additions and 0 deletions
-
63onchange_helper/README.rst
-
3onchange_helper/__init__.py
-
15onchange_helper/__manifest__.py
-
25onchange_helper/i18n/cs_CZ.po
-
27onchange_helper/i18n/de.po
-
24onchange_helper/i18n/es.po
-
24onchange_helper/i18n/fr.po
-
28onchange_helper/i18n/hr.po
-
20onchange_helper/i18n/onchange_helper.pot
-
28onchange_helper/i18n/sl.po
-
3onchange_helper/models/__init__.py
-
69onchange_helper/models/base.py
-
4onchange_helper/readme/CONTRIBUTORS.rst
-
1onchange_helper/readme/DESCRIPTION.rst
-
39onchange_helper/readme/USAGE.rst
-
3onchange_helper/tests/__init__.py
-
36onchange_helper/tests/test_onchange_helper.py
-
1setup/onchange_helper/odoo/addons/onchange_helper
-
6setup/onchange_helper/setup.py
@ -0,0 +1,63 @@ |
|||||
|
.. image:: https://img.shields.io/badge/license-AGPL--3-blue.png |
||||
|
:target: https://www.gnu.org/licenses/agpl |
||||
|
:alt: License: AGPL-3 |
||||
|
|
||||
|
=============== |
||||
|
Onchange Helper |
||||
|
=============== |
||||
|
|
||||
|
This is a technical module. Its goal is to ease the play of onchange method directly called from Python code. |
||||
|
|
||||
|
Usage |
||||
|
===== |
||||
|
|
||||
|
To use this module, you need to: |
||||
|
|
||||
|
* depend on this module |
||||
|
* call `yourmodel.play_onchanges(values, ['field'])` |
||||
|
|
||||
|
Example if you want to create a sale order and you want to get the values relative to partner_id field (as if you fill the field from UI) |
||||
|
|
||||
|
`vals = {'partner_id': 1}` |
||||
|
|
||||
|
`vals = self.env['sale.order'].play_onchanges(vals, ['partner_id'])` |
||||
|
|
||||
|
Then, `vals` will be updated with partner_invoice_id, partner_shipping_id, pricelist_id, etc... |
||||
|
|
||||
|
Bug Tracker |
||||
|
=========== |
||||
|
|
||||
|
Bugs are tracked on `GitHub Issues |
||||
|
<https://github.com/OCA/server-tools/issues>`_. In case of trouble, please |
||||
|
check there if your issue has already been reported. If you spotted it first, |
||||
|
help us smashing it by providing a detailed and welcomed feedback. |
||||
|
|
||||
|
Credits |
||||
|
======= |
||||
|
|
||||
|
Images |
||||
|
------ |
||||
|
|
||||
|
* Odoo Community Association: `Icon <https://odoo-community.org/logo.png>`_. |
||||
|
|
||||
|
Contributors |
||||
|
------------ |
||||
|
|
||||
|
* Guewen Baconnier <guewen.baconnier@camptocamp.com> |
||||
|
* Florian da Costa <florian.dacosta@akretion.com> |
||||
|
* Andrea Stirpe <a.stirpe@onestein.nl> |
||||
|
|
||||
|
Maintainer |
||||
|
---------- |
||||
|
|
||||
|
.. image:: https://odoo-community.org/logo.png |
||||
|
:alt: Odoo Community Association |
||||
|
:target: https://odoo-community.org |
||||
|
|
||||
|
This module is maintained by the OCA. |
||||
|
|
||||
|
OCA, or the Odoo Community Association, is a nonprofit organization whose |
||||
|
mission is to support the collaborative development of Odoo features and |
||||
|
promote its widespread use. |
||||
|
|
||||
|
To contribute to this module, please visit https://odoo-community.org. |
@ -0,0 +1,3 @@ |
|||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from . import models |
@ -0,0 +1,15 @@ |
|||||
|
# Copyright 2016-2017 Akretion (http://www.akretion.com) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
{ |
||||
|
'name': 'Onchange Helper', |
||||
|
'version': '12.0.1.0.0', |
||||
|
'summary': 'Technical module that ease execution' |
||||
|
' of onchange in Python code', |
||||
|
'author': 'Akretion,Camptocamp,Odoo Community Association (OCA)', |
||||
|
'website': 'https://github.com/OCA/server-tools', |
||||
|
'license': 'AGPL-3', |
||||
|
'category': 'Generic Modules', |
||||
|
'depends': ['base'], |
||||
|
'installable': True, |
||||
|
} |
@ -0,0 +1,25 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * onchange_helper |
||||
|
# |
||||
|
# Translators: |
||||
|
# Lukáš Spurný <lukasspurny8@gmail.com>, 2018 |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 11.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2018-03-03 10:08+0000\n" |
||||
|
"PO-Revision-Date: 2018-03-03 10:08+0000\n" |
||||
|
"Last-Translator: Lukáš Spurný <lukasspurny8@gmail.com>, 2018\n" |
||||
|
"Language-Team: Czech (Czech Republic) (https://www.transifex.com/oca/" |
||||
|
"teams/23907/cs_CZ/)\n" |
||||
|
"Language: cs_CZ\n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: nplurals=3; plural=(n==1) ? 0 : (n>=2 && n<=4) ? 1 : 2;\n" |
||||
|
|
||||
|
#. module: onchange_helper |
||||
|
#: model:ir.model,name:onchange_helper.model_base |
||||
|
msgid "base" |
||||
|
msgstr "základny" |
@ -0,0 +1,27 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * onchange_helper |
||||
|
# |
||||
|
# Translators: |
||||
|
# Niki Waibel <niki.waibel@gmail.com>, 2017 |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 10.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2017-06-01 14:59+0000\n" |
||||
|
"PO-Revision-Date: 2017-06-01 14:59+0000\n" |
||||
|
"Last-Translator: Niki Waibel <niki.waibel@gmail.com>, 2017\n" |
||||
|
"Language-Team: German (https://www.transifex.com/oca/teams/23907/de/)\n" |
||||
|
"Language: de\n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n" |
||||
|
|
||||
|
#. module: onchange_helper |
||||
|
#: model:ir.model,name:onchange_helper.model_base |
||||
|
msgid "base" |
||||
|
msgstr "" |
||||
|
|
||||
|
#~ msgid "ir.rule" |
||||
|
#~ msgstr "ir.rule" |
@ -0,0 +1,24 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * onchange_helper |
||||
|
# |
||||
|
# Translators: |
||||
|
# enjolras <yo@miguelrevilla.com>, 2018 |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 11.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2018-03-03 10:08+0000\n" |
||||
|
"PO-Revision-Date: 2018-03-03 10:08+0000\n" |
||||
|
"Last-Translator: enjolras <yo@miguelrevilla.com>, 2018\n" |
||||
|
"Language-Team: Spanish (https://www.transifex.com/oca/teams/23907/es/)\n" |
||||
|
"Language: es\n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n" |
||||
|
|
||||
|
#. module: onchange_helper |
||||
|
#: model:ir.model,name:onchange_helper.model_base |
||||
|
msgid "base" |
||||
|
msgstr "base" |
@ -0,0 +1,24 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * onchange_helper |
||||
|
# |
||||
|
# Translators: |
||||
|
# Quentin THEURET <odoo@kerpeo.com>, 2018 |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 11.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2018-03-03 10:08+0000\n" |
||||
|
"PO-Revision-Date: 2018-03-03 10:08+0000\n" |
||||
|
"Last-Translator: Quentin THEURET <odoo@kerpeo.com>, 2018\n" |
||||
|
"Language-Team: French (https://www.transifex.com/oca/teams/23907/fr/)\n" |
||||
|
"Language: fr\n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: nplurals=2; plural=(n > 1);\n" |
||||
|
|
||||
|
#. module: onchange_helper |
||||
|
#: model:ir.model,name:onchange_helper.model_base |
||||
|
msgid "base" |
||||
|
msgstr "base" |
@ -0,0 +1,28 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * onchange_helper |
||||
|
# |
||||
|
# Translators: |
||||
|
# OCA Transbot <transbot@odoo-community.org>, 2017 |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 10.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2017-06-01 14:59+0000\n" |
||||
|
"PO-Revision-Date: 2017-06-01 14:59+0000\n" |
||||
|
"Last-Translator: OCA Transbot <transbot@odoo-community.org>, 2017\n" |
||||
|
"Language-Team: Croatian (https://www.transifex.com/oca/teams/23907/hr/)\n" |
||||
|
"Language: hr\n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n" |
||||
|
"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n" |
||||
|
|
||||
|
#. module: onchange_helper |
||||
|
#: model:ir.model,name:onchange_helper.model_base |
||||
|
msgid "base" |
||||
|
msgstr "" |
||||
|
|
||||
|
#~ msgid "ir.rule" |
||||
|
#~ msgstr "ir.rule" |
@ -0,0 +1,20 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * onchange_helper |
||||
|
# |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 11.0\n" |
||||
|
"Report-Msgid-Bugs-To: \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: onchange_helper |
||||
|
#: model:ir.model,name:onchange_helper.model_base |
||||
|
msgid "base" |
||||
|
msgstr "" |
||||
|
|
@ -0,0 +1,28 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * onchange_helper |
||||
|
# |
||||
|
# Translators: |
||||
|
# OCA Transbot <transbot@odoo-community.org>, 2017 |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 10.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2017-06-01 14:59+0000\n" |
||||
|
"PO-Revision-Date: 2017-06-01 14:59+0000\n" |
||||
|
"Last-Translator: OCA Transbot <transbot@odoo-community.org>, 2017\n" |
||||
|
"Language-Team: Slovenian (https://www.transifex.com/oca/teams/23907/sl/)\n" |
||||
|
"Language: sl\n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: nplurals=4; plural=(n%100==1 ? 0 : n%100==2 ? 1 : n%100==3 || n" |
||||
|
"%100==4 ? 2 : 3);\n" |
||||
|
|
||||
|
#. module: onchange_helper |
||||
|
#: model:ir.model,name:onchange_helper.model_base |
||||
|
msgid "base" |
||||
|
msgstr "" |
||||
|
|
||||
|
#~ msgid "ir.rule" |
||||
|
#~ msgstr "ir.rule" |
@ -0,0 +1,3 @@ |
|||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from . import base |
@ -0,0 +1,69 @@ |
|||||
|
# Copyright 2016-2017 Akretion (http://www.akretion.com) |
||||
|
# Copyright 2016-2017 Camptocamp (http://www.camptocamp.com/) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from odoo import api, models |
||||
|
|
||||
|
|
||||
|
class Base(models.AbstractModel): |
||||
|
_inherit = 'base' |
||||
|
|
||||
|
@api.model |
||||
|
def _get_new_values(self, record, on_change_result): |
||||
|
vals = on_change_result.get('value', {}) |
||||
|
new_values = {} |
||||
|
for fieldname, value in vals.items(): |
||||
|
if fieldname not in record: |
||||
|
column = self._fields[fieldname] |
||||
|
if value and column.type == 'many2one': |
||||
|
value = value[0] # many2one are tuple (id, name) |
||||
|
new_values[fieldname] = value |
||||
|
return new_values |
||||
|
|
||||
|
@api.model |
||||
|
def play_onchanges(self, values, onchange_fields): |
||||
|
""" |
||||
|
:param values: dict of input value that |
||||
|
:param onchange_fields: fields for which onchange methods will be |
||||
|
played |
||||
|
Order in onchange_fields is very important as onchanges methods will |
||||
|
be played in that order. |
||||
|
:return: changed values |
||||
|
""" |
||||
|
# _onchange_spec() will return onchange fields from the default view |
||||
|
# we need all fields in the dict even the empty ones |
||||
|
# otherwise 'onchange()' will not apply changes to them |
||||
|
onchange_specs = { |
||||
|
field_name: '1' for field_name, field in self._fields.items() |
||||
|
} |
||||
|
all_values = values.copy() |
||||
|
# If self is a record (play onchange on existing record) |
||||
|
# we take the value of the field |
||||
|
# If self is an empty record we will have an empty value |
||||
|
if self: |
||||
|
self.ensure_one() |
||||
|
record_values = self._convert_to_write( |
||||
|
{ |
||||
|
field_name: self[field_name] |
||||
|
for field_name, field in self._fields.items() |
||||
|
} |
||||
|
) |
||||
|
else: |
||||
|
# We get default values, they may be used in onchange |
||||
|
record_values = self.default_get(self._fields.keys()) |
||||
|
for field in self._fields: |
||||
|
if field not in all_values: |
||||
|
all_values[field] = record_values.get(field, False) |
||||
|
|
||||
|
new_values = {} |
||||
|
for field in onchange_fields: |
||||
|
onchange_values = self.onchange(all_values, field, onchange_specs) |
||||
|
new_values.update(self._get_new_values(values, onchange_values)) |
||||
|
all_values.update(new_values) |
||||
|
|
||||
|
return { |
||||
|
f: v |
||||
|
for f, v in all_values.items() |
||||
|
if not (self._fields[f].compute and not self._fields[f].inverse) |
||||
|
and (f in values or f in new_values or f in onchange_fields) |
||||
|
} |
@ -0,0 +1,4 @@ |
|||||
|
* Guewen Baconnier <guewen.baconnier@camptocamp.com> |
||||
|
* Florian da Costa <florian.dacosta@akretion.com> |
||||
|
* Andrea Stirpe <a.stirpe@onestein.nl> |
||||
|
* Souheil Bejaoui <souheil.bejaoui@acsone.eu> |
@ -0,0 +1 @@ |
|||||
|
This is a technical module. Its goal is to ease the play of onchange method directly called from Python code. |
@ -0,0 +1,39 @@ |
|||||
|
To use this module, you need to: |
||||
|
|
||||
|
* depend on this module |
||||
|
* call `yourmodel.play_onchanges(values, ['field'])` |
||||
|
|
||||
|
Example if you want to create a sale order and you want to get the values relative to partner_id field (as if you fill the field from UI) |
||||
|
|
||||
|
`vals = {'partner_id': 1}` |
||||
|
|
||||
|
`vals = self.env['sale.order'].play_onchanges(vals, ['partner_id'])` |
||||
|
|
||||
|
Then, `vals` will be updated with partner_invoice_id, partner_shipping_id, pricelist_id, etc... |
||||
|
|
||||
|
Default values will be used to process onchange methods, if respective fields are not set in `vals`. |
||||
|
You can get them if you pass fields name in the list of fields. |
||||
|
|
||||
|
|
||||
|
`vals = {'partner_id': 1}` |
||||
|
|
||||
|
`vals = self.env['sale.order'].play_onchanges(vals, ['partner_id', 'date_order'])` |
||||
|
|
||||
|
`vals` will contain, in addition to the changed values, the default value for `date_order` |
||||
|
|
||||
|
|
||||
|
You can also use it on existing record for example: |
||||
|
|
||||
|
`vals = {'partner_shipping_id': 1}` |
||||
|
|
||||
|
`vals = sale.play_onchanges(vals, ['partner_shipping_id'])` |
||||
|
|
||||
|
Then the onchange will be played with the vals passed and the existing vals of the sale. `vals` will be updated with partner_invoice_id, pricelist_id, etc.. |
||||
|
|
||||
|
Behind the scene, `play_onchanges` will execute **all the methods** registered for the list of changed fields, so you do not have to call manually each onchange. To avoid performance issue when the method is called on a record, the record will be transformed into a memory record before calling the registered methods to avoid to trigger SQL updates command when values are assigned to the record by the onchange |
||||
|
|
||||
|
|
||||
|
Notes: |
||||
|
|
||||
|
- Order in onchange_fields is very important as onchanges methods will be played in that order. |
||||
|
- If you use memory object in `vals`, be award that onchange method in base model call `self.invalidate_cache()` that reset it. |
@ -0,0 +1,3 @@ |
|||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from . import test_onchange_helper |
@ -0,0 +1,36 @@ |
|||||
|
# Copyright 2017 Onestein (<http://www.onestein.eu>) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from odoo.tests.common import TransactionCase |
||||
|
|
||||
|
|
||||
|
class TestOnchangeHelper(TransactionCase): |
||||
|
def test01_partner_parent(self): |
||||
|
main_partner = self.env.ref('base.main_partner') |
||||
|
input_vals = dict(parent_id=main_partner.id, type='contact') |
||||
|
updated_vals = self.env['res.partner'].play_onchanges( |
||||
|
input_vals, ['parent_id'] |
||||
|
) |
||||
|
self.assertIn('country_id', updated_vals) |
||||
|
self.assertIn('state_id', updated_vals) |
||||
|
self.assertIn('street', updated_vals) |
||||
|
self.assertIn('zip', updated_vals) |
||||
|
|
||||
|
self.assertEqual( |
||||
|
updated_vals['country_id'], main_partner.country_id.id |
||||
|
) |
||||
|
self.assertEqual(updated_vals['state_id'], main_partner.state_id.id) |
||||
|
self.assertEqual(updated_vals['street'], main_partner.street) |
||||
|
self.assertEqual(updated_vals['zip'], main_partner.zip) |
||||
|
|
||||
|
def test02_partner_country(self): |
||||
|
partner_demo = self.env.ref('base.partner_demo') |
||||
|
input_vals = {'country_id': self.env.ref('base.us').id} |
||||
|
updated_vals = partner_demo.play_onchanges(input_vals, ['country_id']) |
||||
|
self.assertIn('country_id', updated_vals) |
||||
|
|
||||
|
def test_playing_onchange_on_model(self): |
||||
|
result = self.env['res.partner'].play_onchanges( |
||||
|
{'company_type': 'company'}, ['company_type'] |
||||
|
) |
||||
|
self.assertEqual(result['is_company'], True) |
@ -0,0 +1 @@ |
|||||
|
../../../../onchange_helper |
@ -0,0 +1,6 @@ |
|||||
|
import setuptools |
||||
|
|
||||
|
setuptools.setup( |
||||
|
setup_requires=['setuptools-odoo'], |
||||
|
odoo_addon=True, |
||||
|
) |
Write
Preview
Loading…
Cancel
Save
Reference in new issue