commit
0f05079a0a
|
@ -0,0 +1,144 @@
|
||||||
|
===============
|
||||||
|
Onchange Helper
|
||||||
|
===============
|
||||||
|
|
||||||
|
..
|
||||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
|
!! This file is generated by oca-gen-addon-readme !!
|
||||||
|
!! changes will be overwritten. !!
|
||||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
|
!! source digest: sha256:556ec1d279174035dfcc8805dad664de965fafb926ddc617899c4a7f51756444
|
||||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
|
|
||||||
|
.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png
|
||||||
|
:target: https://odoo-community.org/page/development-status
|
||||||
|
:alt: Beta
|
||||||
|
.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png
|
||||||
|
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
|
||||||
|
:alt: License: AGPL-3
|
||||||
|
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fserver--tools-lightgray.png?logo=github
|
||||||
|
:target: https://github.com/OCA/server-tools/tree/17.0/onchange_helper
|
||||||
|
:alt: OCA/server-tools
|
||||||
|
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
|
||||||
|
:target: https://translation.odoo-community.org/projects/server-tools-17-0/server-tools-17-0-onchange_helper
|
||||||
|
:alt: Translate me on Weblate
|
||||||
|
.. |badge5| image:: https://img.shields.io/badge/runboat-Try%20me-875A7B.png
|
||||||
|
:target: https://runboat.odoo-community.org/builds?repo=OCA/server-tools&target_branch=17.0
|
||||||
|
:alt: Try me on Runboat
|
||||||
|
|
||||||
|
|badge1| |badge2| |badge3| |badge4| |badge5|
|
||||||
|
|
||||||
|
This is a technical module. Its goal is to ease the play of onchange
|
||||||
|
method directly called from Python code.
|
||||||
|
|
||||||
|
**Table of contents**
|
||||||
|
|
||||||
|
.. contents::
|
||||||
|
:local:
|
||||||
|
|
||||||
|
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...
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
Known issues / Roadmap
|
||||||
|
======================
|
||||||
|
|
||||||
|
Note that onchanges tend to disappear due to the introduction of
|
||||||
|
'computed stored readonly False fields' in most cases. When migrating,
|
||||||
|
it is best to prefer changing it to that way instead of using this
|
||||||
|
module.
|
||||||
|
|
||||||
|
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 to smash it by providing a detailed and welcomed
|
||||||
|
`feedback <https://github.com/OCA/server-tools/issues/new?body=module:%20onchange_helper%0Aversion:%2017.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
|
||||||
|
|
||||||
|
Do not contact contributors directly about support or help with technical issues.
|
||||||
|
|
||||||
|
Credits
|
||||||
|
=======
|
||||||
|
|
||||||
|
Authors
|
||||||
|
-------
|
||||||
|
|
||||||
|
* Akretion
|
||||||
|
* Camptocamp
|
||||||
|
|
||||||
|
Contributors
|
||||||
|
------------
|
||||||
|
|
||||||
|
- 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>
|
||||||
|
- Kevin Khao <kevin.khao@akretion.com>
|
||||||
|
|
||||||
|
Maintainers
|
||||||
|
-----------
|
||||||
|
|
||||||
|
This module is maintained by the OCA.
|
||||||
|
|
||||||
|
.. image:: https://odoo-community.org/logo.png
|
||||||
|
:alt: Odoo Community Association
|
||||||
|
:target: https://odoo-community.org
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
This module is part of the `OCA/server-tools <https://github.com/OCA/server-tools/tree/17.0/onchange_helper>`_ project on GitHub.
|
||||||
|
|
||||||
|
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.
|
|
@ -0,0 +1,3 @@
|
||||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from . import models
|
|
@ -0,0 +1,14 @@
|
||||||
|
# Copyright 2016-2017 Akretion (http://www.akretion.com)
|
||||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "Onchange Helper",
|
||||||
|
"version": "17.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,28 @@
|
||||||
|
# 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 ""
|
||||||
|
|
||||||
|
#~ 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,28 @@
|
||||||
|
# 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: 2023-09-03 00:15+0000\n"
|
||||||
|
"Last-Translator: Ivorra78 <informatica@totmaterial.es>\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"
|
||||||
|
"X-Generator: Weblate 4.17\n"
|
||||||
|
|
||||||
|
#. module: onchange_helper
|
||||||
|
#: model:ir.model,name:onchange_helper.model_base
|
||||||
|
msgid "Base"
|
||||||
|
msgstr "Base"
|
||||||
|
|
||||||
|
#~ msgid "base"
|
||||||
|
#~ msgstr "base"
|
|
@ -0,0 +1,22 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * onchange_helper
|
||||||
|
#
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 15.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"PO-Revision-Date: 2022-09-04 06:07+0000\n"
|
||||||
|
"Last-Translator: Ignacio Buioli <ibuioli@gmail.com>\n"
|
||||||
|
"Language-Team: none\n"
|
||||||
|
"Language: es_AR\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"
|
||||||
|
"X-Generator: Weblate 4.3.2\n"
|
||||||
|
|
||||||
|
#. module: onchange_helper
|
||||||
|
#: model:ir.model,name:onchange_helper.model_base
|
||||||
|
msgid "Base"
|
||||||
|
msgstr "Base"
|
|
@ -0,0 +1,27 @@
|
||||||
|
# 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 ""
|
||||||
|
|
||||||
|
#~ 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,22 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * onchange_helper
|
||||||
|
#
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 16.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"PO-Revision-Date: 2024-01-02 16:34+0000\n"
|
||||||
|
"Last-Translator: mymage <stefano.consolaro@mymage.it>\n"
|
||||||
|
"Language-Team: none\n"
|
||||||
|
"Language: it\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"
|
||||||
|
"X-Generator: Weblate 4.17\n"
|
||||||
|
|
||||||
|
#. module: onchange_helper
|
||||||
|
#: model:ir.model,name:onchange_helper.model_base
|
||||||
|
msgid "Base"
|
||||||
|
msgstr "Base"
|
|
@ -0,0 +1,19 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * onchange_helper
|
||||||
|
#
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 16.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,22 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * onchange_helper
|
||||||
|
#
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 12.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"PO-Revision-Date: 2019-08-31 06:18+0000\n"
|
||||||
|
"Last-Translator: 黎伟杰 <674416404@qq.com>\n"
|
||||||
|
"Language-Team: none\n"
|
||||||
|
"Language: zh_CN\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=1; plural=0;\n"
|
||||||
|
"X-Generator: Weblate 3.8\n"
|
||||||
|
|
||||||
|
#. module: onchange_helper
|
||||||
|
#: model:ir.model,name:onchange_helper.model_base
|
||||||
|
msgid "Base"
|
||||||
|
msgstr "基础"
|
|
@ -0,0 +1,3 @@
|
||||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from . import base
|
|
@ -0,0 +1,71 @@
|
||||||
|
# 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["id"]
|
||||||
|
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 = self._get_fields_spec()
|
||||||
|
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 self._fields[f].readonly
|
||||||
|
)
|
||||||
|
and (f in values or f in new_values or f in onchange_fields)
|
||||||
|
}
|
|
@ -0,0 +1,3 @@
|
||||||
|
[build-system]
|
||||||
|
requires = ["whool"]
|
||||||
|
build-backend = "whool.buildapi"
|
|
@ -0,0 +1,5 @@
|
||||||
|
- 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>\>
|
||||||
|
- Kevin Khao \<<kevin.khao@akretion.com>\>
|
|
@ -0,0 +1,2 @@
|
||||||
|
This is a technical module. Its goal is to ease the play of onchange
|
||||||
|
method directly called from Python code.
|
|
@ -0,0 +1,4 @@
|
||||||
|
Note that onchanges tend to disappear due to the introduction of
|
||||||
|
'computed stored readonly False fields' in most cases. When migrating,
|
||||||
|
it is best to prefer changing it to that way instead of using this
|
||||||
|
module.
|
|
@ -0,0 +1,50 @@
|
||||||
|
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.
|
Binary file not shown.
After Width: | Height: | Size: 9.2 KiB |
|
@ -0,0 +1,673 @@
|
||||||
|
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
|
||||||
|
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||||
|
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
|
||||||
|
<head>
|
||||||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
|
||||||
|
<meta name="generator" content="Docutils: https://docutils.sourceforge.io/"/>
|
||||||
|
<title>Onchange Helper</title>
|
||||||
|
<style type="text/css">
|
||||||
|
|
||||||
|
/*
|
||||||
|
:Author: David Goodger (goodger@python.org)
|
||||||
|
:Id: $Id: html4css1.css 8954 2022-01-20 10:10:25Z milde $
|
||||||
|
:Copyright: This stylesheet has been placed in the public domain.
|
||||||
|
|
||||||
|
Default cascading style sheet for the HTML output of Docutils.
|
||||||
|
|
||||||
|
See https://docutils.sourceforge.io/docs/howto/html-stylesheets.html for how to
|
||||||
|
customize this style sheet.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* used to remove borders from tables and images */
|
||||||
|
.borderless, table.borderless td, table.borderless th {
|
||||||
|
border: 0
|
||||||
|
}
|
||||||
|
|
||||||
|
table.borderless td, table.borderless th {
|
||||||
|
/* Override padding for "table.docutils td" with "! important".
|
||||||
|
The right padding separates the table cells. */
|
||||||
|
padding: 0 0.5em 0 0 ! important
|
||||||
|
}
|
||||||
|
|
||||||
|
.first {
|
||||||
|
/* Override more specific margin styles with "! important". */
|
||||||
|
margin-top: 0 ! important
|
||||||
|
}
|
||||||
|
|
||||||
|
.last, .with-subtitle {
|
||||||
|
margin-bottom: 0 ! important
|
||||||
|
}
|
||||||
|
|
||||||
|
.hidden {
|
||||||
|
display: none
|
||||||
|
}
|
||||||
|
|
||||||
|
.subscript {
|
||||||
|
vertical-align: sub;
|
||||||
|
font-size: smaller
|
||||||
|
}
|
||||||
|
|
||||||
|
.superscript {
|
||||||
|
vertical-align: super;
|
||||||
|
font-size: smaller
|
||||||
|
}
|
||||||
|
|
||||||
|
a.toc-backref {
|
||||||
|
text-decoration: none;
|
||||||
|
color: black
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote.epigraph {
|
||||||
|
margin: 2em 5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
dl.docutils dd {
|
||||||
|
margin-bottom: 0.5em
|
||||||
|
}
|
||||||
|
|
||||||
|
object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Uncomment (and remove this text!) to get bold-faced definition list terms
|
||||||
|
dl.docutils dt {
|
||||||
|
font-weight: bold }
|
||||||
|
*/
|
||||||
|
|
||||||
|
div.abstract {
|
||||||
|
margin: 2em 5em
|
||||||
|
}
|
||||||
|
|
||||||
|
div.abstract p.topic-title {
|
||||||
|
font-weight: bold;
|
||||||
|
text-align: center
|
||||||
|
}
|
||||||
|
|
||||||
|
div.admonition, div.attention, div.caution, div.danger, div.error,
|
||||||
|
div.hint, div.important, div.note, div.tip, div.warning {
|
||||||
|
margin: 2em;
|
||||||
|
border: medium outset;
|
||||||
|
padding: 1em
|
||||||
|
}
|
||||||
|
|
||||||
|
div.admonition p.admonition-title, div.hint p.admonition-title,
|
||||||
|
div.important p.admonition-title, div.note p.admonition-title,
|
||||||
|
div.tip p.admonition-title {
|
||||||
|
font-weight: bold;
|
||||||
|
font-family: sans-serif
|
||||||
|
}
|
||||||
|
|
||||||
|
div.attention p.admonition-title, div.caution p.admonition-title,
|
||||||
|
div.danger p.admonition-title, div.error p.admonition-title,
|
||||||
|
div.warning p.admonition-title, .code .error {
|
||||||
|
color: red;
|
||||||
|
font-weight: bold;
|
||||||
|
font-family: sans-serif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Uncomment (and remove this text!) to get reduced vertical space in
|
||||||
|
compound paragraphs.
|
||||||
|
div.compound .compound-first, div.compound .compound-middle {
|
||||||
|
margin-bottom: 0.5em }
|
||||||
|
|
||||||
|
div.compound .compound-last, div.compound .compound-middle {
|
||||||
|
margin-top: 0.5em }
|
||||||
|
*/
|
||||||
|
|
||||||
|
div.dedication {
|
||||||
|
margin: 2em 5em;
|
||||||
|
text-align: center;
|
||||||
|
font-style: italic
|
||||||
|
}
|
||||||
|
|
||||||
|
div.dedication p.topic-title {
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: normal
|
||||||
|
}
|
||||||
|
|
||||||
|
div.figure {
|
||||||
|
margin-left: 2em;
|
||||||
|
margin-right: 2em
|
||||||
|
}
|
||||||
|
|
||||||
|
div.footer, div.header {
|
||||||
|
clear: both;
|
||||||
|
font-size: smaller
|
||||||
|
}
|
||||||
|
|
||||||
|
div.line-block {
|
||||||
|
display: block;
|
||||||
|
margin-top: 1em;
|
||||||
|
margin-bottom: 1em
|
||||||
|
}
|
||||||
|
|
||||||
|
div.line-block div.line-block {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
margin-left: 1.5em
|
||||||
|
}
|
||||||
|
|
||||||
|
div.sidebar {
|
||||||
|
margin: 0 0 0.5em 1em;
|
||||||
|
border: medium outset;
|
||||||
|
padding: 1em;
|
||||||
|
background-color: #ffffee;
|
||||||
|
width: 40%;
|
||||||
|
float: right;
|
||||||
|
clear: right
|
||||||
|
}
|
||||||
|
|
||||||
|
div.sidebar p.rubric {
|
||||||
|
font-family: sans-serif;
|
||||||
|
font-size: medium
|
||||||
|
}
|
||||||
|
|
||||||
|
div.system-messages {
|
||||||
|
margin: 5em
|
||||||
|
}
|
||||||
|
|
||||||
|
div.system-messages h1 {
|
||||||
|
color: red
|
||||||
|
}
|
||||||
|
|
||||||
|
div.system-message {
|
||||||
|
border: medium outset;
|
||||||
|
padding: 1em
|
||||||
|
}
|
||||||
|
|
||||||
|
div.system-message p.system-message-title {
|
||||||
|
color: red;
|
||||||
|
font-weight: bold
|
||||||
|
}
|
||||||
|
|
||||||
|
div.topic {
|
||||||
|
margin: 2em
|
||||||
|
}
|
||||||
|
|
||||||
|
h1.section-subtitle, h2.section-subtitle, h3.section-subtitle,
|
||||||
|
h4.section-subtitle, h5.section-subtitle, h6.section-subtitle {
|
||||||
|
margin-top: 0.4em
|
||||||
|
}
|
||||||
|
|
||||||
|
h1.title {
|
||||||
|
text-align: center
|
||||||
|
}
|
||||||
|
|
||||||
|
h2.subtitle {
|
||||||
|
text-align: center
|
||||||
|
}
|
||||||
|
|
||||||
|
hr.docutils {
|
||||||
|
width: 75%
|
||||||
|
}
|
||||||
|
|
||||||
|
img.align-left, .figure.align-left, object.align-left, table.align-left {
|
||||||
|
clear: left;
|
||||||
|
float: left;
|
||||||
|
margin-right: 1em
|
||||||
|
}
|
||||||
|
|
||||||
|
img.align-right, .figure.align-right, object.align-right, table.align-right {
|
||||||
|
clear: right;
|
||||||
|
float: right;
|
||||||
|
margin-left: 1em
|
||||||
|
}
|
||||||
|
|
||||||
|
img.align-center, .figure.align-center, object.align-center {
|
||||||
|
display: block;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.align-center {
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-left {
|
||||||
|
text-align: left
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-center {
|
||||||
|
clear: both;
|
||||||
|
text-align: center
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-right {
|
||||||
|
text-align: right
|
||||||
|
}
|
||||||
|
|
||||||
|
/* reset inner alignment in figures */
|
||||||
|
div.align-right {
|
||||||
|
text-align: inherit
|
||||||
|
}
|
||||||
|
|
||||||
|
/* div.align-center * { */
|
||||||
|
/* text-align: left } */
|
||||||
|
|
||||||
|
.align-top {
|
||||||
|
vertical-align: top
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-middle {
|
||||||
|
vertical-align: middle
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-bottom {
|
||||||
|
vertical-align: bottom
|
||||||
|
}
|
||||||
|
|
||||||
|
ol.simple, ul.simple {
|
||||||
|
margin-bottom: 1em
|
||||||
|
}
|
||||||
|
|
||||||
|
ol.arabic {
|
||||||
|
list-style: decimal
|
||||||
|
}
|
||||||
|
|
||||||
|
ol.loweralpha {
|
||||||
|
list-style: lower-alpha
|
||||||
|
}
|
||||||
|
|
||||||
|
ol.upperalpha {
|
||||||
|
list-style: upper-alpha
|
||||||
|
}
|
||||||
|
|
||||||
|
ol.lowerroman {
|
||||||
|
list-style: lower-roman
|
||||||
|
}
|
||||||
|
|
||||||
|
ol.upperroman {
|
||||||
|
list-style: upper-roman
|
||||||
|
}
|
||||||
|
|
||||||
|
p.attribution {
|
||||||
|
text-align: right;
|
||||||
|
margin-left: 50%
|
||||||
|
}
|
||||||
|
|
||||||
|
p.caption {
|
||||||
|
font-style: italic
|
||||||
|
}
|
||||||
|
|
||||||
|
p.credits {
|
||||||
|
font-style: italic;
|
||||||
|
font-size: smaller
|
||||||
|
}
|
||||||
|
|
||||||
|
p.label {
|
||||||
|
white-space: nowrap
|
||||||
|
}
|
||||||
|
|
||||||
|
p.rubric {
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: larger;
|
||||||
|
color: maroon;
|
||||||
|
text-align: center
|
||||||
|
}
|
||||||
|
|
||||||
|
p.sidebar-title {
|
||||||
|
font-family: sans-serif;
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: larger
|
||||||
|
}
|
||||||
|
|
||||||
|
p.sidebar-subtitle {
|
||||||
|
font-family: sans-serif;
|
||||||
|
font-weight: bold
|
||||||
|
}
|
||||||
|
|
||||||
|
p.topic-title {
|
||||||
|
font-weight: bold
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.address {
|
||||||
|
margin-bottom: 0;
|
||||||
|
margin-top: 0;
|
||||||
|
font: inherit
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.literal-block, pre.doctest-block, pre.math, pre.code {
|
||||||
|
margin-left: 2em;
|
||||||
|
margin-right: 2em
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code .ln {
|
||||||
|
color: grey;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* line numbers */
|
||||||
|
pre.code, code {
|
||||||
|
background-color: #eeeeee
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code .comment, code .comment {
|
||||||
|
color: #5C6576
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code .keyword, code .keyword {
|
||||||
|
color: #3B0D06;
|
||||||
|
font-weight: bold
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code .literal.string, code .literal.string {
|
||||||
|
color: #0C5404
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code .name.builtin, code .name.builtin {
|
||||||
|
color: #352B84
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code .deleted, code .deleted {
|
||||||
|
background-color: #DEB0A1
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code .inserted, code .inserted {
|
||||||
|
background-color: #A3D289
|
||||||
|
}
|
||||||
|
|
||||||
|
span.classifier {
|
||||||
|
font-family: sans-serif;
|
||||||
|
font-style: oblique
|
||||||
|
}
|
||||||
|
|
||||||
|
span.classifier-delimiter {
|
||||||
|
font-family: sans-serif;
|
||||||
|
font-weight: bold
|
||||||
|
}
|
||||||
|
|
||||||
|
span.interpreted {
|
||||||
|
font-family: sans-serif
|
||||||
|
}
|
||||||
|
|
||||||
|
span.option {
|
||||||
|
white-space: nowrap
|
||||||
|
}
|
||||||
|
|
||||||
|
span.pre {
|
||||||
|
white-space: pre
|
||||||
|
}
|
||||||
|
|
||||||
|
span.problematic {
|
||||||
|
color: red
|
||||||
|
}
|
||||||
|
|
||||||
|
span.section-subtitle {
|
||||||
|
/* font-size relative to parent (h1..h6 element) */
|
||||||
|
font-size: 80%
|
||||||
|
}
|
||||||
|
|
||||||
|
table.citation {
|
||||||
|
border-left: solid 1px gray;
|
||||||
|
margin-left: 1px
|
||||||
|
}
|
||||||
|
|
||||||
|
table.docinfo {
|
||||||
|
margin: 2em 4em
|
||||||
|
}
|
||||||
|
|
||||||
|
table.docutils {
|
||||||
|
margin-top: 0.5em;
|
||||||
|
margin-bottom: 0.5em
|
||||||
|
}
|
||||||
|
|
||||||
|
table.footnote {
|
||||||
|
border-left: solid 1px black;
|
||||||
|
margin-left: 1px
|
||||||
|
}
|
||||||
|
|
||||||
|
table.docutils td, table.docutils th,
|
||||||
|
table.docinfo td, table.docinfo th {
|
||||||
|
padding-left: 0.5em;
|
||||||
|
padding-right: 0.5em;
|
||||||
|
vertical-align: top
|
||||||
|
}
|
||||||
|
|
||||||
|
table.docutils th.field-name, table.docinfo th.docinfo-name {
|
||||||
|
font-weight: bold;
|
||||||
|
text-align: left;
|
||||||
|
white-space: nowrap;
|
||||||
|
padding-left: 0
|
||||||
|
}
|
||||||
|
|
||||||
|
/* "booktabs" style (no vertical lines) */
|
||||||
|
table.docutils.booktabs {
|
||||||
|
border: 0px;
|
||||||
|
border-top: 2px solid;
|
||||||
|
border-bottom: 2px solid;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.docutils.booktabs * {
|
||||||
|
border: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.docutils.booktabs th {
|
||||||
|
border-bottom: thin solid;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 tt.docutils, h2 tt.docutils, h3 tt.docutils,
|
||||||
|
h4 tt.docutils, h5 tt.docutils, h6 tt.docutils {
|
||||||
|
font-size: 100%
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.auto-toc {
|
||||||
|
list-style-type: none
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="document" id="onchange-helper">
|
||||||
|
<h1 class="title">Onchange Helper</h1>
|
||||||
|
|
||||||
|
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
|
!! This file is generated by oca-gen-addon-readme !!
|
||||||
|
!! changes will be overwritten. !!
|
||||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
|
!! source digest: sha256:556ec1d279174035dfcc8805dad664de965fafb926ddc617899c4a7f51756444
|
||||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
|
||||||
|
<p>
|
||||||
|
<a class="reference external image-reference" href="https://odoo-community.org/page/development-status">
|
||||||
|
<img alt="Beta" src="https://img.shields.io/badge/maturity-Beta-yellow.png"/>
|
||||||
|
</a>
|
||||||
|
<a class="reference external image-reference"
|
||||||
|
href="http://www.gnu.org/licenses/agpl-3.0-standalone.html">
|
||||||
|
<img alt="License: AGPL-3" src="https://img.shields.io/badge/licence-AGPL--3-blue.png"/>
|
||||||
|
</a>
|
||||||
|
<a class="reference external image-reference"
|
||||||
|
href="https://github.com/OCA/server-tools/tree/17.0/onchange_helper">
|
||||||
|
<img alt="OCA/server-tools"
|
||||||
|
src="https://img.shields.io/badge/github-OCA%2Fserver--tools-lightgray.png?logo=github"/>
|
||||||
|
</a>
|
||||||
|
<a class="reference external image-reference"
|
||||||
|
href="https://translation.odoo-community.org/projects/server-tools-17-0/server-tools-17-0-onchange_helper">
|
||||||
|
<img alt="Translate me on Weblate"
|
||||||
|
src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png"/>
|
||||||
|
</a>
|
||||||
|
<a class="reference external image-reference"
|
||||||
|
href="https://runboat.odoo-community.org/builds?repo=OCA/server-tools&target_branch=17.0">
|
||||||
|
<img alt="Try me on Runboat" src="https://img.shields.io/badge/runboat-Try%20me-875A7B.png"/>
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
<p>This is a technical module. Its goal is to ease the play of onchange
|
||||||
|
method directly called from Python code.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<strong>Table of contents</strong>
|
||||||
|
</p>
|
||||||
|
<div class="contents local topic" id="contents">
|
||||||
|
<ul class="simple">
|
||||||
|
<li>
|
||||||
|
<a class="reference internal" href="#usage" id="toc-entry-1">Usage</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a class="reference internal" href="#known-issues-roadmap" id="toc-entry-2">Known issues /
|
||||||
|
Roadmap
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a class="reference internal" href="#bug-tracker" id="toc-entry-3">Bug Tracker</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a class="reference internal" href="#credits" id="toc-entry-4">Credits</a>
|
||||||
|
<ul>
|
||||||
|
<li>
|
||||||
|
<a class="reference internal" href="#authors" id="toc-entry-5">Authors</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a class="reference internal" href="#contributors" id="toc-entry-6">Contributors</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a class="reference internal" href="#maintainers" id="toc-entry-7">Maintainers</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="usage">
|
||||||
|
<h1>
|
||||||
|
<a class="toc-backref" href="#toc-entry-1">Usage</a>
|
||||||
|
</h1>
|
||||||
|
<p>To use this module, you need to:</p>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>depend on this module</li>
|
||||||
|
<li>call yourmodel.play_onchanges(values, [‘field’])</li>
|
||||||
|
</ul>
|
||||||
|
<p>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)
|
||||||
|
</p>
|
||||||
|
<blockquote>
|
||||||
|
<p>vals = {‘partner_id’: 1}</p>
|
||||||
|
<p>vals = self.env[‘sale.order’].play_onchanges(vals, [‘partner_id’])</p>
|
||||||
|
</blockquote>
|
||||||
|
<p>Then, vals will be updated with partner_invoice_id, partner_shipping_id,
|
||||||
|
pricelist_id, etc…
|
||||||
|
</p>
|
||||||
|
<p>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.
|
||||||
|
</p>
|
||||||
|
<blockquote>
|
||||||
|
<p>vals = {‘partner_id’: 1}</p>
|
||||||
|
<p>vals = self.env[‘sale.order’].play_onchanges(vals, [‘partner_id’,
|
||||||
|
‘date_order’])
|
||||||
|
</p>
|
||||||
|
</blockquote>
|
||||||
|
<p>vals will contain, in addition to the changed values, the default value
|
||||||
|
for date_order
|
||||||
|
</p>
|
||||||
|
<p>You can also use it on existing record for example:</p>
|
||||||
|
<blockquote>
|
||||||
|
<p>vals = {‘partner_shipping_id’: 1}</p>
|
||||||
|
<p>vals = sale.play_onchanges(vals, [‘partner_shipping_id’])</p>
|
||||||
|
</blockquote>
|
||||||
|
<p>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..
|
||||||
|
</p>
|
||||||
|
<p>Behind the scene, play_onchanges will execute
|
||||||
|
<strong>all the methods</strong>
|
||||||
|
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
|
||||||
|
</p>
|
||||||
|
<p>Notes:</p>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>Order in onchange_fields is very important as onchanges methods will
|
||||||
|
be played in that order.
|
||||||
|
</li>
|
||||||
|
<li>If you use memory object in vals, be award that onchange method in
|
||||||
|
base model call self.invalidate_cache() that reset it.
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="known-issues-roadmap">
|
||||||
|
<h1>
|
||||||
|
<a class="toc-backref" href="#toc-entry-2">Known issues / Roadmap</a>
|
||||||
|
</h1>
|
||||||
|
<p>Note that onchanges tend to disappear due to the introduction of
|
||||||
|
‘computed stored readonly False fields’ in most cases. When migrating,
|
||||||
|
it is best to prefer changing it to that way instead of using this
|
||||||
|
module.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="bug-tracker">
|
||||||
|
<h1>
|
||||||
|
<a class="toc-backref" href="#toc-entry-3">Bug Tracker</a>
|
||||||
|
</h1>
|
||||||
|
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/server-tools/issues">
|
||||||
|
GitHub Issues</a>.
|
||||||
|
In case of trouble, please check there if your issue has already been reported.
|
||||||
|
If you spotted it first, help us to smash it by providing a detailed and welcomed
|
||||||
|
<a class="reference external"
|
||||||
|
href="https://github.com/OCA/server-tools/issues/new?body=module:%20onchange_helper%0Aversion:%2017.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**">
|
||||||
|
feedback</a>.
|
||||||
|
</p>
|
||||||
|
<p>Do not contact contributors directly about support or help with technical issues.</p>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="credits">
|
||||||
|
<h1>
|
||||||
|
<a class="toc-backref" href="#toc-entry-4">Credits</a>
|
||||||
|
</h1>
|
||||||
|
<div class="section" id="authors">
|
||||||
|
<h2>
|
||||||
|
<a class="toc-backref" href="#toc-entry-5">Authors</a>
|
||||||
|
</h2>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>Akretion</li>
|
||||||
|
<li>Camptocamp</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="contributors">
|
||||||
|
<h2>
|
||||||
|
<a class="toc-backref" href="#toc-entry-6">Contributors</a>
|
||||||
|
</h2>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>Guewen Baconnier <<a class="reference external"
|
||||||
|
href="mailto:guewen.baconnier@camptocamp.com">guewen.baconnier@camptocamp.com</a>>
|
||||||
|
</li>
|
||||||
|
<li>Florian da Costa <<a class="reference external"
|
||||||
|
href="mailto:florian.dacosta@akretion.com">florian.dacosta@akretion.com</a>>
|
||||||
|
</li>
|
||||||
|
<li>Andrea Stirpe <<a class="reference external" href="mailto:a.stirpe@onestein.nl">
|
||||||
|
a.stirpe@onestein.nl</a>>
|
||||||
|
</li>
|
||||||
|
<li>Souheil Bejaoui <<a class="reference external"
|
||||||
|
href="mailto:souheil.bejaoui@acsone.eu">souheil.bejaoui@acsone.eu</a>>
|
||||||
|
</li>
|
||||||
|
<li>Kevin Khao <<a class="reference external" href="mailto:kevin.khao@akretion.com">
|
||||||
|
kevin.khao@akretion.com</a>>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="maintainers">
|
||||||
|
<h2>
|
||||||
|
<a class="toc-backref" href="#toc-entry-7">Maintainers</a>
|
||||||
|
</h2>
|
||||||
|
<p>This module is maintained by the OCA.</p>
|
||||||
|
<a class="reference external image-reference" href="https://odoo-community.org">
|
||||||
|
<img alt="Odoo Community Association" src="https://odoo-community.org/logo.png"/>
|
||||||
|
</a>
|
||||||
|
<p>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.
|
||||||
|
</p>
|
||||||
|
<p>This module is part of the
|
||||||
|
<a class="reference external"
|
||||||
|
href="https://github.com/OCA/server-tools/tree/17.0/onchange_helper">OCA/server-tools
|
||||||
|
</a>
|
||||||
|
project on GitHub.
|
||||||
|
</p>
|
||||||
|
<p>You are welcome to contribute. To learn how please visit <a class="reference external"
|
||||||
|
href="https://odoo-community.org/page/Contribute">
|
||||||
|
https://odoo-community.org/page/Contribute</a>.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -0,0 +1,3 @@
|
||||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from . import test_onchange_helper
|
|
@ -0,0 +1,32 @@
|
||||||
|
# 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)
|
Loading…
Reference in New Issue