3
0
Fork 0

[MIG] web_widget_dropdown_dynamic: Migration to 17.0

17.0
Carlos Roca 2024-08-13 12:33:49 +02:00
parent 920e5bf943
commit 1c6d9cd008
13 changed files with 164 additions and 124 deletions

View File

@ -71,6 +71,13 @@ Usage
context="{'depending_on': other_field}"
/>
**DEMO**
On User defined filters added new field to show the feature, it is
called **Dropdown Integer**. If any user selected just One option shoud
appear, but if Mitchell Admin it should be possible to select option One
and Two.
Bug Tracker
===========
@ -106,6 +113,10 @@ Contributors
- Son Ho <sonho@trobz.com>
- `Tecnativa <https://www.tecnativa.com>`__:
- Carlos Roca
Other credits
-------------

View File

@ -1 +1,2 @@
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
from . import models

View File

@ -5,7 +5,7 @@
"name": "Dynamic Dropdown Widget",
"summary": "This module adds support for dynamic dropdown widget",
"category": "Web",
"version": "16.0.1.0.0",
"version": "17.0.1.0.0",
"license": "AGPL-3",
"author": "CorporateHub, Odoo Community Association (OCA)",
"website": "https://github.com/OCA/web",
@ -13,7 +13,14 @@
"installable": True,
"assets": {
"web.assets_backend": [
"web_widget_dropdown_dynamic/**/*",
"web_widget_dropdown_dynamic/static/src/js/field_dynamic_dropdown.esm.js",
],
"web.qunit_suite_tests": [
"web_widget_dropdown_dynamic/static/tests/web_widget_dropdown_dynamic_tests.esm.js",
],
},
"demo": [
"demo/ir_model_fields.xml",
"demo/ir_filters_view.xml",
],
}

View File

@ -0,0 +1,17 @@
<?xml version="1.0" encoding="utf-8" ?>
<odoo>
<record id="ir_filters_view_form" model="ir.ui.view">
<field name="model">ir.filters</field>
<field name="inherit_id" ref="base.ir_filters_view_form" />
<field name="arch" type="xml">
<field name="user_id" position="after">
<field
name="x_dynamic_dropdown_int"
widget="dynamic_dropdown"
options="{'values': 'dynamic_dropdown_int_method_demo'}"
context="{'depending_on': user_id}"
/>
</field>
</field>
</record>
</odoo>

View File

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="utf-8" ?>
<odoo>
<record id="field_dynamic_dropdown_int" model="ir.model.fields">
<field name="field_description">Dropdown Integer</field>
<field name="model">ir.filters</field>
<field name="model_id" ref="base.model_ir_filters" />
<field name="name">x_dynamic_dropdown_int</field>
<field name="state">manual</field>
<field name="ttype">integer</field>
</record>
</odoo>

View File

@ -0,0 +1,4 @@
from odoo.tools import config
if not config.get("without_demo"):
from . import ir_filters

View File

@ -0,0 +1,19 @@
# Copyright 2024 Tecnativa - Carlos Roca
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
from odoo import api, models
class IrFilters(models.Model):
_inherit = "ir.filters"
@api.model
def dynamic_dropdown_int_method_demo(self):
values = [
("1", "One"),
]
if self.env.context.get("depending_on") == self.env.ref("base.user_admin").id:
values += [
("2", "Two"),
]
return values

View File

@ -11,3 +11,6 @@
- [Trobz](https://trobz.com):
> - Son Ho \<<sonho@trobz.com>\>
- [Tecnativa](https://www.tecnativa.com):
- Carlos Roca

View File

@ -22,3 +22,9 @@ def method_name(self):
context="{'depending_on': other_field}"
/>
```
**DEMO**
On User defined filters added new field to show the feature, it is called
**Dropdown Integer**. If any user selected just One option shoud appear, but if
Mitchell Admin it should be possible to select option One and Two.

View File

@ -420,6 +420,11 @@ but to filter selection values. For fully-dynamic set of options, use
</span><span class="na">context=</span><span class="s">&quot;{'depending_on': other_field}&quot;</span><span class="w">
</span><span class="nt">/&gt;</span>
</pre>
<p><strong>DEMO</strong></p>
<p>On User defined filters added new field to show the feature, it is
called <strong>Dropdown Integer</strong>. If any user selected just One option shoud
appear, but if Mitchell Admin it should be possible to select option One
and Two.</p>
</div>
<div class="section" id="bug-tracker">
<h1><a class="toc-backref" href="#toc-entry-2">Bug Tracker</a></h1>
@ -459,6 +464,11 @@ If you spotted it first, help us to smash it by providing a detailed and welcome
</ul>
</blockquote>
</li>
<li><p class="first"><a class="reference external" href="https://www.tecnativa.com">Tecnativa</a>:</p>
<ul class="simple">
<li>Carlos Roca</li>
</ul>
</li>
</ul>
</div>
<div class="section" id="other-credits">

View File

@ -1,50 +0,0 @@
/** @odoo-module **/
import BasicModel from "web.BasicModel";
BasicModel.include({
/**
* Fetches all the values associated to the given fieldName.
*
* @param {Object} record - an element from the localData
* @param {Object} fieldName - the name of the field
* @param {Object} fieldInfo
* @returns {Promise<any>}
* The promise is resolved with the fetched special values.
* If this data is the same as the previously fetched one
* (for the given parameters), no RPC is done and the promise
* is resolved with the undefined value.
*/
_fetchDynamicDropdownValues: function (record, fieldName, fieldInfo) {
var model = fieldInfo.options.model || record.model;
var method = fieldInfo.values || fieldInfo.options.values;
if (!method) {
return Promise.resolve();
}
var context = record.getContext({fieldName: fieldName});
// Avoid rpc if not necessary
var hasChanged = this._saveSpecialDataCache(record, fieldName, {
context: context,
});
if (!hasChanged) {
return Promise.resolve();
}
return this._rpc({
model: model,
method: method,
context: context,
}).then(function (result) {
var new_result = result.map((val_updated) => {
return val_updated.map((e) => {
if (typeof e !== "string") {
return String(e);
}
return e;
});
});
return new_result;
});
},
});

View File

@ -1,29 +1,56 @@
/** @odoo-module **/
import core from "web.core";
import {_lt} from "@web/core/l10n/translation";
import {registry} from "@web/core/registry";
import {standardFieldProps} from "@web/views/fields/standard_field_props";
import {Component} from "@odoo/owl";
var _lt = core._lt;
import {Component, onWillStart, onWillUpdateProps} from "@odoo/owl";
export class FieldDynamicDropdown extends Component {
static template = "web.SelectionField";
static props = {
...standardFieldProps,
method: {type: String},
context: {type: Object},
};
setup() {
this.type = this.props.record.fields[this.props.name].type;
onWillStart(async () => {
this.specialData = await this._fetchSpecialData(this.props);
});
onWillUpdateProps(async (nextProps) => {
if (this.props.context.depending_on !== nextProps.context.depending_on) {
this.specialData = await this._fetchSpecialData(nextProps);
}
});
}
async _fetchSpecialData(props) {
const {resModel} = props.record.model.config;
const {specialDataCaches, orm} = props.record.model;
const key = `__reference__${props.name}-${props.context.depending_on}`;
if (!specialDataCaches[key]) {
specialDataCaches[key] = await orm.call(resModel, props.method, [], {
context: props.context,
});
}
return specialDataCaches[key];
}
get options() {
var field_type = this.props.record.fields[this.props.name].type || "";
var field_type = this.type || "";
if (["char", "integer", "selection"].includes(field_type)) {
this._setValues();
return this.props.record.fields[this.props.name].selection.filter(
(option) => option[0] !== false && option[1] !== ""
);
if (
this.props.record.data[this.props.name] &&
!this.specialData
.map((val) => val[0])
.includes(String(this.props.record.data[this.props.name]))
) {
this.props.record.update({[this.props.name]: null});
}
return this.specialData;
}
return [];
}
get value() {
const rawValue = this.props.value;
this.props.setDirty(false);
return this.props.type === "many2one" && rawValue ? rawValue[0] : rawValue;
return String(this.props.record.data[this.props.name]);
}
parseInteger(value) {
return Number(value);
}
@ -31,58 +58,32 @@ export class FieldDynamicDropdown extends Component {
* @param {Event} ev
*/
onChange(ev) {
let lastSetValue = null;
let isInvalid = false;
var isDirty = ev.target.value !== lastSetValue;
const field = this.props.record.fields[this.props.name];
let value = JSON.parse(ev.target.value);
if (isDirty) {
if (value && field.type === "integer") {
value = Number(value);
if (!value) {
if (this.props.record) {
this.props.record.setInvalidField(this.props.name);
}
isInvalid = true;
var isInvalid = false;
var value = JSON.parse(ev.target.value);
if (this.type === "integer") {
value = Number(value);
if (!value) {
if (this.props.record) {
this.props.record.setInvalidField(this.props.name);
}
}
if (!isInvalid) {
Promise.resolve(this.props.update(value));
lastSetValue = ev.target.value;
isInvalid = true;
}
}
if (this.props.setDirty) {
this.props.setDirty(isDirty);
if (!isInvalid) {
this.props.record.update({[this.props.name]: value});
}
}
stringify(value) {
return JSON.stringify(value);
}
_setValues() {
if (this.props.record.preloadedData[this.props.name]) {
var sel_value = this.props.record.preloadedData[this.props.name];
// Convert string element to integer if field is integer
if (this.props.record.fields[this.props.name].type === "integer") {
sel_value = sel_value.map((val_updated) => {
return val_updated.map((e) => {
if (typeof e === "string" && !isNaN(Number(e))) {
return Number(e);
}
return e;
});
});
}
this.props.record.fields[this.props.name].selection = sel_value;
}
}
}
FieldDynamicDropdown.description = _lt("Dynamic Dropdown");
FieldDynamicDropdown.template = "web.SelectionField";
FieldDynamicDropdown.legacySpecialData = "_fetchDynamicDropdownValues";
FieldDynamicDropdown.props = {
...standardFieldProps,
export const dynamicDropdownField = {
component: FieldDynamicDropdown,
displayName: _lt("Dynamic Dropdown"),
supportedTypes: ["char", "integer", "selection"],
extractProps: (fieldInfo, dynamicInfo) => ({
method: fieldInfo.options?.values,
context: dynamicInfo.context,
}),
};
FieldDynamicDropdown.supportedTypes = ["char", "integer", "selection"];
registry.category("fields").add("dynamic_dropdown", FieldDynamicDropdown);
registry.category("fields").add("dynamic_dropdown", dynamicDropdownField);

View File

@ -30,7 +30,7 @@ QUnit.module("web_widget_dropdown_dynamic", (hooks) => {
],
methods: {
method_name() {
return Promise.resolve([["value a", "Value A"]]);
return [["value a", "Value A"]];
},
},
},
@ -55,14 +55,14 @@ QUnit.module("web_widget_dropdown_dynamic", (hooks) => {
assert.step(args.method);
if (args.method === "method_name") {
if (args.kwargs.context.depending_on === "step-1") {
return Promise.resolve([["value", "Title"]]);
return [["value", "Title"]];
} else if (args.kwargs.context.depending_on === "step-2") {
return Promise.resolve([
return [
["value", "Title"],
["value_2", "Title 2"],
]);
];
}
return Promise.resolve([]);
return [];
}
},
});
@ -84,7 +84,7 @@ QUnit.module("web_widget_dropdown_dynamic", (hooks) => {
assert.containsN(target, "option", 1);
assert.verifySteps([
"get_views",
"read",
"web_read",
"method_name",
"method_name",
"method_name",
@ -109,13 +109,13 @@ QUnit.module("web_widget_dropdown_dynamic", (hooks) => {
assert.step(args.method);
if (args.method === "method_name") {
if (args.kwargs.context.depending_on) {
return Promise.resolve([["value b", "Value B"]]);
return [["value b", "Value B"]];
}
}
},
});
const field_target = target.querySelector("div[name='content_string']");
assert.verifySteps(["get_views", "read", "method_name"]);
assert.verifySteps(["get_views", "web_read", "method_name"]);
assert.containsN(field_target, "option", 2);
assert.containsOnce(
field_target,
@ -142,15 +142,15 @@ QUnit.module("web_widget_dropdown_dynamic", (hooks) => {
assert.step(args.method);
if (args.method === "method_name") {
if (args.kwargs.context.depending_on) {
return Promise.resolve([["10", "Value B"]]);
return [["10", "Value B"]];
}
}
},
});
const field_target = target.querySelector("div[name='content_integer']");
assert.verifySteps(["get_views", "read", "method_name"]);
assert.verifySteps(["get_views", "web_read", "method_name"]);
assert.containsN(field_target, "option", 2);
assert.containsOnce(field_target, 'option[value="10"]');
assert.containsOnce(field_target, 'option[value="\\"10\\""]');
});
QUnit.test("values are fetched w/o context (selection)", async (assert) => {
@ -169,13 +169,13 @@ QUnit.module("web_widget_dropdown_dynamic", (hooks) => {
assert.step(args.method);
if (args.method === "method_name") {
if (args.kwargs.context.depending_on) {
return Promise.resolve([["choice b", "Choice B"]]);
return [["choice b", "Choice B"]];
}
}
},
});
const field_target = target.querySelector("div[name='content_selection']");
assert.verifySteps(["get_views", "read", "method_name"]);
assert.verifySteps(["get_views", "web_read", "method_name"]);
assert.containsN(field_target, "option", 2);
assert.containsOnce(field_target, "option[value='\"choice b\"']");
});