mirror of https://github.com/OCA/web.git
[ADD][web_widget_domain_v11] Domain widget backport
- Most code directly copied from upstream Odoo, in the commit before the JS refactor that took place in master. - Some little glue code to make it replace v10's widget.pull/672/head
parent
977a85c133
commit
b6610665b7
|
@ -0,0 +1,68 @@
|
|||
.. image:: https://img.shields.io/badge/licence-LGPL--3-blue.svg
|
||||
:target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html
|
||||
:alt: License: LGPL-3
|
||||
|
||||
===============================
|
||||
Odoo 11.0 Domain Widget Preview
|
||||
===============================
|
||||
|
||||
This module replaces the functionality of the domain widget to use a preview of
|
||||
the brand new interface that will be found in Odoo 11.0.
|
||||
|
||||
Usage
|
||||
=====
|
||||
|
||||
To use this module, you need to:
|
||||
|
||||
#. Install any addon that makes use of the domain widget (i.e.
|
||||
``mass_mailing``).
|
||||
#. You will be able to use the updated version.
|
||||
|
||||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas
|
||||
:alt: Try me on Runbot
|
||||
:target: https://runbot.odoo-community.org/runbot/162/10.0
|
||||
|
||||
Known issues / Roadmap
|
||||
======================
|
||||
|
||||
* This addon replaces the built-in ``char_domain`` widget, so it can break
|
||||
compatibility with other addons that use it.
|
||||
|
||||
Bug Tracker
|
||||
===========
|
||||
|
||||
Bugs are tracked on `GitHub Issues
|
||||
<https://github.com/OCA/web/issues>`_. In case of trouble, please
|
||||
check there if your issue has already been reported. If you spotted it first,
|
||||
help us smash it by providing detailed and welcomed feedback.
|
||||
|
||||
Credits
|
||||
=======
|
||||
|
||||
Most code copied from https://github.com/odoo/odoo/tree/68176d80ad6053f52ed1c7bcf294ab3664986c46/addons/web/static/src
|
||||
|
||||
Images
|
||||
------
|
||||
|
||||
* Odoo Community Association: `Icon <https://github.com/OCA/maintainer-tools/blob/master/template/module/static/description/icon.svg>`_.
|
||||
|
||||
Contributors
|
||||
------------
|
||||
|
||||
* Odoo SA <https://www.odoo.com>
|
||||
* Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
|
||||
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,2 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl).
|
|
@ -0,0 +1,23 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl).
|
||||
{
|
||||
"name": "Odoo 11.0 Domain Widget",
|
||||
"summary": "Updated domain widget",
|
||||
"version": "10.0.1.0.0",
|
||||
"category": "Extra Tools",
|
||||
"website": "https://www.tecnativa.com/",
|
||||
"author": "Tecnativa, Odoo S.A., Odoo Community Association (OCA)",
|
||||
"license": "LGPL-3",
|
||||
"application": False,
|
||||
"installable": True,
|
||||
"depends": [
|
||||
"web",
|
||||
],
|
||||
"data": [
|
||||
"templates/assets.xml",
|
||||
],
|
||||
"qweb": [
|
||||
"static/src/copied-xml/templates.xml",
|
||||
],
|
||||
}
|
Binary file not shown.
After Width: | Height: | Size: 9.2 KiB |
|
@ -0,0 +1,87 @@
|
|||
|
||||
.o_domain_node {
|
||||
@o-domain-selector-vspace: 8px;
|
||||
@o-domain-selector-indent: 32px;
|
||||
@o-domain-selector-panel-space: 128px;
|
||||
|
||||
.o_domain_node_control_panel {
|
||||
.o-position-absolute(@right: 0);
|
||||
}
|
||||
|
||||
&.o_domain_tree {
|
||||
.o_domain_tree_operator_caret::after {
|
||||
.o-caret-down();
|
||||
}
|
||||
|
||||
> .o_domain_node_children_container {
|
||||
padding-left: @o-domain-selector-indent;
|
||||
|
||||
> div {
|
||||
margin-top: @o-domain-selector-vspace;
|
||||
}
|
||||
}
|
||||
|
||||
&.o_domain_selector {
|
||||
&.o_edit_mode {
|
||||
position: relative;
|
||||
|
||||
> .o_domain_node_children_container {
|
||||
padding-right: @o-domain-selector-panel-space;
|
||||
}
|
||||
}
|
||||
|
||||
> .o_domain_node_children_container {
|
||||
padding-left: 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
&.o_domain_leaf {
|
||||
&.o_read_mode {
|
||||
display: inline-block;
|
||||
margin-right: 4px;
|
||||
}
|
||||
|
||||
> .o_domain_leaf_info {
|
||||
background: @odoo-brand-lightsecondary;
|
||||
border: 1px solid darken(@odoo-brand-lightsecondary, 10%);
|
||||
padding: 2px 4px;
|
||||
|
||||
.o_domain_leaf_chain, .o_domain_leaf_value {
|
||||
font-weight: 700;
|
||||
}
|
||||
.o_domain_leaf_operator {
|
||||
font-style: italic;
|
||||
}
|
||||
}
|
||||
|
||||
> .o_domain_leaf_edition {
|
||||
.o-flex-display();
|
||||
.o-align-items(flex-end);
|
||||
|
||||
> * {
|
||||
width: percentage(1/3);
|
||||
|
||||
+ * {
|
||||
margin-left: 4px;
|
||||
}
|
||||
}
|
||||
|
||||
.o_domain_leaf_value_tags {
|
||||
.o-flex-display();
|
||||
|
||||
> * {
|
||||
.o-flex(0, 0, auto);
|
||||
}
|
||||
> input {
|
||||
.o-flex(1, 1, auto);
|
||||
width: 0;
|
||||
min-width: 50px;
|
||||
}
|
||||
.o_domain_leaf_value_remove_tag_button {
|
||||
cursor: pointer;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,97 @@
|
|||
|
||||
.o_field_selector {
|
||||
position: relative;
|
||||
|
||||
.o_field_selector_controls {
|
||||
.o-position-absolute(0, 0, 1px);
|
||||
.o-flex-display();
|
||||
.o-align-items(center);
|
||||
cursor: pointer;
|
||||
|
||||
&::after {
|
||||
.o-caret-down();
|
||||
}
|
||||
}
|
||||
.o_field_selector_popover {
|
||||
@o-field-selector-arrow-height: 7px;
|
||||
.o-position-absolute(@top: 100%, @left: 0);
|
||||
z-index: 1051;
|
||||
width: 265px;
|
||||
margin-top: @o-field-selector-arrow-height;
|
||||
background: white;
|
||||
box-shadow: 0 3px 10px rgba(0,0,0,.4);
|
||||
|
||||
&:focus {
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.o_field_selector_popover_header {
|
||||
color: white;
|
||||
background: @brand-primary;
|
||||
font-weight: bold;
|
||||
padding: 5px 0 5px 0.4em;
|
||||
|
||||
.o_field_selector_title {
|
||||
width: 100%;
|
||||
.o-text-overflow();
|
||||
padding: 0px 35px;
|
||||
text-align: center;
|
||||
}
|
||||
.o_field_selector_popover_option {
|
||||
.o-position-absolute(@top: 0);
|
||||
padding: 7px 8px 8px 6px;
|
||||
|
||||
&.o_prev_page {
|
||||
left: 0;
|
||||
border-right: 1px solid darken(@brand-primary, 10%);
|
||||
}
|
||||
&.o_field_selector_close {
|
||||
right: 0;
|
||||
border-left: 1px solid darken(@brand-primary, 10%);
|
||||
}
|
||||
&:hover {
|
||||
background: darken(@brand-primary, 10%);
|
||||
}
|
||||
}
|
||||
&:before {
|
||||
.o-position-absolute(@top: -@o-field-selector-arrow-height, @left: @o-field-selector-arrow-height);
|
||||
content: "";
|
||||
border-left: @o-field-selector-arrow-height solid rgba(0, 0, 0, 0);
|
||||
border-right: @o-field-selector-arrow-height solid rgba(0, 0, 0, 0);
|
||||
border-bottom: @o-field-selector-arrow-height solid @brand-primary;
|
||||
}
|
||||
}
|
||||
.o_field_selector_popover_body {
|
||||
.o_field_selector_page {
|
||||
position: relative;
|
||||
max-height: 320px;
|
||||
overflow: auto;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
|
||||
> .o_field_selector_item {
|
||||
list-style: none;
|
||||
position: relative;
|
||||
padding: 5px 0 5px 0.4em;
|
||||
cursor: pointer;
|
||||
font-family: Arial;
|
||||
font-size: 13px;
|
||||
color: #444;
|
||||
border-bottom: 1px solid #eee;
|
||||
&.active {
|
||||
background: #f5f5f5;
|
||||
}
|
||||
.o_field_selector_item_title {
|
||||
font-size: 12px;
|
||||
}
|
||||
.o_field_selector_relation_icon {
|
||||
.o-position-absolute(@top: 0, @right: 0, @bottom: 0);
|
||||
.o-flex-display();
|
||||
.o-align-items(center);
|
||||
padding: 10px;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,585 @@
|
|||
odoo.define("web.DomainSelector", function (require) {
|
||||
"use strict";
|
||||
|
||||
var core = require("web.core");
|
||||
var datepicker = require("web.datepicker");
|
||||
var domainUtils = require("web.domainUtils");
|
||||
var formats = require ("web.formats");
|
||||
var ModelFieldSelector = require("web.ModelFieldSelector");
|
||||
var Widget = require("web.Widget");
|
||||
|
||||
var _t = core._t;
|
||||
var _lt = core._lt;
|
||||
|
||||
// "child_of", "parent_of", "like", "not like", "=like", "=ilike"
|
||||
// are only used if user entered them manually or if got from demo data
|
||||
var operator_mapping = {
|
||||
"=": _lt("is equal to"),
|
||||
"!=": _lt("is not equal to"),
|
||||
">": _lt("greater than"),
|
||||
"<": _lt("less than"),
|
||||
">=": _lt("greater than or equal to"),
|
||||
"<=": _lt("less than or equal to"),
|
||||
"ilike": _lt("contains"),
|
||||
"not ilike": _lt("not contains"),
|
||||
"in": _lt("in"),
|
||||
"not in": _lt("not in"),
|
||||
|
||||
"child_of": _lt("child of"),
|
||||
"parent_of": _lt("parent of"),
|
||||
"like": "like",
|
||||
"not like": "not like",
|
||||
"=like": "=like",
|
||||
"=ilike": "=ilike",
|
||||
|
||||
// custom
|
||||
"set": _lt("is set"),
|
||||
"not set": _lt("is not set"),
|
||||
};
|
||||
|
||||
/// The DomainNode Widget is an abstraction for widgets which can represent and allow
|
||||
/// edition of a domain (part).
|
||||
var DomainNode = Widget.extend({
|
||||
events: {
|
||||
/// If click on the node add or delete button, notify the parent and let it handle the addition/removal
|
||||
"click .o_domain_delete_node_button": function (e) {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
this.trigger_up("delete_node_clicked", {child: this});
|
||||
},
|
||||
"click .o_domain_add_node_button": function (e) {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
this.trigger_up("add_node_clicked", {newBranch: !!$(e.currentTarget).data("branch"), child: this});
|
||||
},
|
||||
},
|
||||
/// A DomainNode needs a model and domain to work. It can also receives a set of options
|
||||
/// @param model - a string with the model name
|
||||
/// @param domain - an array of the prefix representation of the domain (or a string which represents it)
|
||||
/// @param options - an object with possible values:
|
||||
/// - readonly, a boolean to indicate if the widget is readonly or not (default to true)
|
||||
/// - operators, a list of available operators (default to null, which indicates all of supported ones)
|
||||
/// - debugMode, a boolean which is true if the widget should be in debug mode (default to false)
|
||||
/// - @see ModelFieldSelector for other options
|
||||
init: function (parent, model, domain, options) {
|
||||
this._super.apply(this, arguments);
|
||||
|
||||
this.model = model;
|
||||
this.options = _.extend({
|
||||
readonly: true,
|
||||
operators: null,
|
||||
debugMode: false,
|
||||
}, options || {});
|
||||
|
||||
this.readonly = this.options.readonly;
|
||||
this.debug = this.options.debugMode;
|
||||
},
|
||||
/// The getDomain method is an abstract method which should returns the prefix domain
|
||||
/// the widget is currently representing (an array).
|
||||
getDomain: function () {},
|
||||
});
|
||||
/// The DomainTree is a DomainNode which can handle subdomains (a domain which is composed
|
||||
/// of multiple parts). It thus will be composed of other DomainTree instances and/or leaf parts
|
||||
/// of a domain (@see DomainLeaf).
|
||||
var DomainTree = DomainNode.extend({
|
||||
template: "DomainTree",
|
||||
events: _.extend({}, DomainNode.prototype.events, {
|
||||
"click .o_domain_tree_operator_selector > ul > li > a": function (e) {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
this.changeOperator($(e.target).data("operator"));
|
||||
},
|
||||
}),
|
||||
custom_events: {
|
||||
/// If a domain child sends a request to add a child or remove one, call the appropriate methods.
|
||||
/// Propagates the event until success.
|
||||
"delete_node_clicked": function (e) {
|
||||
e.stopped = this.removeChild(e.data.child);
|
||||
},
|
||||
"add_node_clicked": function (e) {
|
||||
var domain = [["id", "=", 1]];
|
||||
if (e.data.newBranch) {
|
||||
domain = [this.operator === "&" ? "|" : "&"].concat(domain).concat(domain);
|
||||
}
|
||||
e.stopped = this.addChild(domain, e.data.child);
|
||||
},
|
||||
},
|
||||
/// @see DomainNode.init
|
||||
/// The initialization of a DomainTree creates a "children" array attribute which will contain the
|
||||
/// the DomainNode children. It also deduces the operator from the domain (default to "&").
|
||||
/// @see DomainTree._addFlattenedChildren
|
||||
init: function (parent, model, domain, options) {
|
||||
this._super.apply(this, arguments);
|
||||
this._initialize(domainUtils.stringToDomain(domain));
|
||||
},
|
||||
/// @see DomainTree.init
|
||||
_initialize: function (domain) {
|
||||
this.operator = domain[0];
|
||||
this.children = [];
|
||||
|
||||
// Add flattened children by search the appropriate number of children in the rest
|
||||
// of the domain (after the operator)
|
||||
var nbLeafsToFind = 1;
|
||||
for (var i = 1 ; i < domain.length ; i++) {
|
||||
if (_.contains(["&", "|"], domain[i])) {
|
||||
nbLeafsToFind++;
|
||||
} else if (domain[i] !== "!") {
|
||||
nbLeafsToFind--;
|
||||
}
|
||||
|
||||
if (!nbLeafsToFind) {
|
||||
var partLeft = domain.slice(1, i+1);
|
||||
var partRight = domain.slice(i+1);
|
||||
if (partLeft.length) {
|
||||
this._addFlattenedChildren(partLeft);
|
||||
}
|
||||
if (partRight.length) {
|
||||
this._addFlattenedChildren(partRight);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Mark "!" tree children so that they do not allow to add other children around them
|
||||
if (this.operator === "!") {
|
||||
this.children[0].noControlPanel = true;
|
||||
}
|
||||
},
|
||||
start: function () {
|
||||
this._postRender();
|
||||
return $.when(this._super.apply(this, arguments), this._renderChildrenTo(this.$childrenContainer));
|
||||
},
|
||||
_postRender: function () {
|
||||
this.$childrenContainer = this.$("> .o_domain_node_children_container");
|
||||
},
|
||||
_renderChildrenTo: function ($to) {
|
||||
var $div = $("<div/>");
|
||||
return $.when.apply($, _.map(this.children, (function (child) {
|
||||
return child.appendTo($div);
|
||||
}).bind(this))).then((function () {
|
||||
_.each(this.children, function (child) {
|
||||
child.$el.appendTo($to); // Forced to do it this way so that the children are not misordered
|
||||
});
|
||||
}).bind(this));
|
||||
},
|
||||
getDomain: function () {
|
||||
var childDomains = [];
|
||||
var nbChildren = 0;
|
||||
_.each(this.children, function (child) {
|
||||
var childDomain = child.getDomain();
|
||||
if (childDomain.length) {
|
||||
nbChildren++;
|
||||
childDomains = childDomains.concat(child.getDomain());
|
||||
}
|
||||
});
|
||||
var nbChildRequired = this.operator === "!" ? 1 : 2;
|
||||
var operators = _.times(nbChildren - nbChildRequired + 1, _.constant(this.operator));
|
||||
return operators.concat(childDomains);
|
||||
},
|
||||
changeOperator: function (operator) {
|
||||
this.operator = operator;
|
||||
this.trigger_up("domain_changed", {child: this});
|
||||
},
|
||||
/// The addChild method adds a domain part to the widget.
|
||||
/// @param domain - an array of the prefix-like domain to build and add to the widget
|
||||
/// @param afterNode - the node after which the new domain part must be added (at the end if not given)
|
||||
/// @trigger_up domain_changed if the child is added
|
||||
/// @return true if the part was added, false otherwise (the afterNode was not found)
|
||||
addChild: function (domain, afterNode) {
|
||||
var i = afterNode ? _.indexOf(this.children, afterNode) : this.children.length;
|
||||
if (i < 0) return false;
|
||||
|
||||
this.children.splice(i+1, 0, instantiateNode(this, this.model, domain, this.options));
|
||||
this.trigger_up("domain_changed", {child: this});
|
||||
return true;
|
||||
},
|
||||
/// The removeChild method removes a given child from the widget.
|
||||
/// @param oldChild - the child instance to remove
|
||||
/// @trigger_up domain_changed if the child is removed
|
||||
/// @return true if the child was removed, false otherwise (the widget does not own the child)
|
||||
removeChild: function (oldChild) {
|
||||
var i = _.indexOf(this.children, oldChild);
|
||||
if (i < 0) return false;
|
||||
|
||||
this.children[i].destroy();
|
||||
this.children.splice(i, 1);
|
||||
this.trigger_up("domain_changed", {child: this});
|
||||
return true;
|
||||
},
|
||||
/// The private _addFlattenedChildren method adds a child which represents the given
|
||||
/// domain. If the child has children and that the child main domain operator is the
|
||||
/// same as the current widget one, the 2-children prefix hierarchy is then simplified
|
||||
/// by making the child children the widget own children.
|
||||
/// @param domain - the domain of the child to add and simplify
|
||||
_addFlattenedChildren: function (domain) {
|
||||
var node = instantiateNode(this, this.model, domain, this.options);
|
||||
if (node === null) {
|
||||
return;
|
||||
}
|
||||
if (!node.children || node.operator !== this.operator) {
|
||||
this.children.push(node);
|
||||
return;
|
||||
}
|
||||
_.each(node.children, (function (child) {
|
||||
child.setParent(this);
|
||||
this.children.push(child);
|
||||
}).bind(this));
|
||||
node.destroy();
|
||||
},
|
||||
/// This method is ugly but achieves the right behavior without flickering.
|
||||
/// It will be refactored alongside the new views/widget API.
|
||||
_redraw: function (domain) {
|
||||
var oldChildren = this.children.slice();
|
||||
this._initialize(domain || this.getDomain());
|
||||
return this._renderChildrenTo($("<div/>")).then((function () {
|
||||
this.renderElement();
|
||||
this._postRender();
|
||||
_.each(this.children, (function (child) { child.$el.appendTo(this.$childrenContainer); }).bind(this));
|
||||
_.each(oldChildren, function (child) { child.destroy(); });
|
||||
}).bind(this));
|
||||
},
|
||||
});
|
||||
/// The DomainSelector widget can be used to build prefix char domain. It is the DomainTree
|
||||
/// specialization to use to have a fully working widget.
|
||||
///
|
||||
/// Known limitations:
|
||||
/// - Some operators like "child_of", "parent_of", "like", "not like", "=like", "=ilike"
|
||||
/// will come only if you use them from demo data or debug input.
|
||||
/// - Some kind of domain can not be build right now e.g ("country_id", "in", [1,2,3,4])
|
||||
/// but you can insert from debug input.
|
||||
var DomainSelector = DomainTree.extend({
|
||||
template: "DomainSelector",
|
||||
events: _.extend({}, DomainTree.prototype.events, {
|
||||
"click .o_domain_add_first_node_button": function (e) {
|
||||
this.addChild([["id", "=", 1]]);
|
||||
},
|
||||
/// When the debug input changes, the string prefix domain is read. If it is syntax-valid
|
||||
/// the widget is re-rendered and notifies the parents. If not, a warning is shown to the
|
||||
/// user and the input is ignored.
|
||||
"change .o_domain_debug_input": function (e) {
|
||||
var domain;
|
||||
try {
|
||||
domain = domainUtils.stringToDomain($(e.currentTarget).val());
|
||||
} catch (err) {
|
||||
this.do_warn(_t("Syntax error"), _t("The domain you entered is not properly formed"));
|
||||
return;
|
||||
}
|
||||
this._redraw(domain).then((function () {
|
||||
this.trigger_up("domain_changed", {child: this, alreadyRedrawn: true});
|
||||
}).bind(this));
|
||||
},
|
||||
}),
|
||||
custom_events: _.extend({}, DomainTree.prototype.custom_events, {
|
||||
/// If a subdomain notifies that it underwent some modifications, the DomainSelector
|
||||
/// catches the message and performs a full re-rendering.
|
||||
"domain_changed": function (e) {
|
||||
e.stopped = false;
|
||||
if (!e.data.alreadyRedrawn) {
|
||||
this._redraw();
|
||||
}
|
||||
},
|
||||
}),
|
||||
_initialize: function (domain) {
|
||||
// Check if the domain starts with implicit "&" operators and make them
|
||||
// explicit. As the DomainSelector is a specialization of a DomainTree,
|
||||
// it is waiting for a tree and not a leaf. So [] and [A] will be made
|
||||
// explicit with ["&"], ["&", A] so that tree parsing is made correctly.
|
||||
// Note: the domain is considered to be a valid one
|
||||
if (domain.length <= 1) {
|
||||
return this._super(["&"].concat(domain));
|
||||
}
|
||||
var expected = 1;
|
||||
_.each(domain, function (item) {
|
||||
if (item === "&" || item === "|") {
|
||||
expected++;
|
||||
} else if (item !== "!") {
|
||||
expected--;
|
||||
}
|
||||
});
|
||||
if (expected < 0) {
|
||||
domain = _.times(Math.abs(expected), _.constant("&")).concat(domain);
|
||||
}
|
||||
return this._super(domain);
|
||||
},
|
||||
_postRender: function () {
|
||||
this._super.apply(this, arguments);
|
||||
|
||||
// Display technical domain if in debug mode
|
||||
this.$debugInput = this.$(".o_domain_debug_input");
|
||||
if (this.$debugInput.length) {
|
||||
this.$debugInput.val(domainUtils.domainToString(this.getDomain()));
|
||||
}
|
||||
},
|
||||
});
|
||||
/// The DomainLeaf widget is a DomainNode which handles a domain which cannot be split in
|
||||
/// another subdomains, i.e. composed of a field chain, an operator and a value.
|
||||
var DomainLeaf = DomainNode.extend({
|
||||
template: "DomainLeaf",
|
||||
events: _.extend({}, DomainNode.prototype.events, {
|
||||
"change .o_domain_leaf_operator_select": function (e) {
|
||||
this.onOperatorChange($(e.currentTarget).val());
|
||||
},
|
||||
"change .o_domain_leaf_value_input": function (e) {
|
||||
if (e.currentTarget !== e.target) return;
|
||||
this.onValueChange($(e.currentTarget).val());
|
||||
},
|
||||
|
||||
// Handle the tags widget part (TODO should be an independant widget)
|
||||
"click .o_domain_leaf_value_add_tag_button": "on_add_tag",
|
||||
"keyup .o_domain_leaf_value_tags input": "on_add_tag",
|
||||
"click .o_domain_leaf_value_remove_tag_button": "on_remove_tag",
|
||||
}),
|
||||
custom_events: {
|
||||
"field_chain_changed": function (e) {
|
||||
this.onChainChange(e.data.chain);
|
||||
},
|
||||
},
|
||||
/// @see DomainNode.init
|
||||
init: function (parent, model, domain, options) {
|
||||
this._super.apply(this, arguments);
|
||||
|
||||
domain = domainUtils.stringToDomain(domain);
|
||||
this.chain = domain[0][0];
|
||||
this.operator = domain[0][1];
|
||||
this.value = domain[0][2];
|
||||
|
||||
this.operator_mapping = operator_mapping;
|
||||
},
|
||||
willStart: function () {
|
||||
var defs = [this._super.apply(this, arguments)];
|
||||
|
||||
if (!this.readonly) {
|
||||
// In edit mode, instantiate a field selector. This is done here in willStart and prepared by
|
||||
// appending it to a dummy element because the DomainLeaf rendering need some information which
|
||||
// cannot be computed before the ModelFieldSelector is fully rendered (TODO).
|
||||
this.fieldSelector = new ModelFieldSelector(this, this.model, this.chain, this.options);
|
||||
defs.push(this.fieldSelector.appendTo($("<div/>")).then((function () {
|
||||
var wDefs = [];
|
||||
|
||||
// Set list of operators according to field type
|
||||
this.operators = this._getOperatorsFromType(this.fieldSelector.selectedField.type);
|
||||
if (_.contains(["child_of", "parent_of", "like", "not like", "=like", "=ilike"], this.operator)) {
|
||||
// In case user entered manually or from demo data
|
||||
this.operators[this.operator] = operator_mapping[this.operator];
|
||||
} else if (!this.operators[this.operator]) {
|
||||
this.operators[this.operator] = "?"; // In case the domain uses an unsupported operator for the field type
|
||||
}
|
||||
|
||||
// Set list of values according to field type
|
||||
this.selectionChoices = null;
|
||||
if (this.fieldSelector.selectedField.type === "boolean") {
|
||||
this.selectionChoices = [["1", "set (true)"], ["0", "not set (false)"]];
|
||||
} else if (this.fieldSelector.selectedField.type === "selection") {
|
||||
this.selectionChoices = this.fieldSelector.selectedField.selection;
|
||||
}
|
||||
|
||||
// Adapt display value and operator for rendering
|
||||
this.displayValue = this.value;
|
||||
try {
|
||||
var f = this.fieldSelector.selectedField;
|
||||
if (!f.relation) { // TODO in this case, the value should be m2o input, etc...
|
||||
this.displayValue = formats.format_value(this.value, this.fieldSelector.selectedField);
|
||||
}
|
||||
} catch (err) {/**/}
|
||||
this.displayOperator = this.operator;
|
||||
if (this.fieldSelector.selectedField.type === "boolean") {
|
||||
this.displayValue = this.value ? "1" : "0";
|
||||
} else if ((this.operator === "!=" || this.operator === "=") && this.value === false) {
|
||||
this.displayOperator = this.operator === "!=" ? "set" : "not set";
|
||||
}
|
||||
|
||||
// TODO the value could be a m2o input, etc...
|
||||
if (_.contains(["date", "datetime"], this.fieldSelector.selectedField.type)) {
|
||||
this.valueWidget = new (this.fieldSelector.selectedField.type === "datetime" ? datepicker.DateTimeWidget : datepicker.DateWidget)(this);
|
||||
wDefs.push(this.valueWidget.appendTo("<div/>").then((function () {
|
||||
this.valueWidget.$el.addClass("o_domain_leaf_value_input");
|
||||
this.valueWidget.set_value(this.value);
|
||||
this.valueWidget.on("datetime_changed", this, function () {
|
||||
this.onValueChange(this.valueWidget.get_value());
|
||||
});
|
||||
}).bind(this)));
|
||||
}
|
||||
|
||||
return $.when.apply($, wDefs);
|
||||
}).bind(this)));
|
||||
}
|
||||
|
||||
return $.when.apply($, defs);
|
||||
},
|
||||
start: function () {
|
||||
if (!this.readonly) { // In edit mode ...
|
||||
this.fieldSelector.$el.prependTo(this.$("> .o_domain_leaf_edition")); // ... place the field selector
|
||||
if (this.valueWidget) { // ... and place the value widget if any
|
||||
this.$(".o_domain_leaf_value_input").replaceWith(this.valueWidget.$el);
|
||||
}
|
||||
}
|
||||
return this._super.apply(this, arguments);
|
||||
},
|
||||
getDomain: function () {
|
||||
return [[this.chain, this.operator, this.value]];
|
||||
},
|
||||
/// The onChainChange method handles a field chain change in the domain. In that case, the operator
|
||||
/// should be adapted to a valid one for the new field and the value should also be adapted to the
|
||||
/// new field and/or operator.
|
||||
/// @param chain - the new field chain (string)
|
||||
/// @param silent - true if the method call should not trigger_up a domain_changed event
|
||||
/// @trigger_up domain_changed event to ask for a re-rendering
|
||||
onChainChange: function (chain, silent) {
|
||||
this.chain = chain;
|
||||
|
||||
var operators = this._getOperatorsFromType(this.fieldSelector.selectedField.type);
|
||||
if (operators[this.operator] === undefined) {
|
||||
this.onOperatorChange("=", true);
|
||||
}
|
||||
|
||||
this.onValueChange(this.value, true);
|
||||
|
||||
if (!silent) this.trigger_up("domain_changed", {child: this});
|
||||
},
|
||||
/// The onOperatorChange method handles an operator change in the domain. In that case, the value
|
||||
/// should be adapted to a valid one for the new operator.
|
||||
/// @param operator - the new operator
|
||||
/// @param silent - true if the method call should not trigger_up a domain_changed event
|
||||
/// @trigger_up domain_changed event to ask for a re-rendering
|
||||
onOperatorChange: function (operator, silent) {
|
||||
this.operator = operator;
|
||||
|
||||
if (_.contains(["set", "not set"], this.operator)) {
|
||||
this.operator = this.operator === "not set" ? "=" : "!=";
|
||||
this.value = false;
|
||||
} else if (_.contains(["in", "not in"], this.operator)) {
|
||||
this.value = _.isArray(this.value) ? this.value : this.value ? ("" + this.value).split(",") : [];
|
||||
} else {
|
||||
if (_.isArray(this.value)) {
|
||||
this.value = this.value.join(",");
|
||||
}
|
||||
this.onValueChange(this.value, true);
|
||||
}
|
||||
|
||||
if (!silent) this.trigger_up("domain_changed", {child: this});
|
||||
},
|
||||
/// The onValueChange method handles a formatted value change in the domain. In that case, the value
|
||||
/// should be adapted to a valid technical one.
|
||||
/// @param value - the new formatted value
|
||||
/// @param silent - true if the method call should not trigger_up a domain_changed event
|
||||
/// @trigger_up domain_changed event to ask for a re-rendering
|
||||
onValueChange: function (value, silent) {
|
||||
var couldNotParse = false;
|
||||
try {
|
||||
this.value = formats.parse_value(value, this.fieldSelector.selectedField);
|
||||
} catch (err) {
|
||||
this.value = value;
|
||||
couldNotParse = true;
|
||||
}
|
||||
|
||||
if (this.fieldSelector.selectedField.type === "boolean") {
|
||||
if (!_.isBoolean(this.value)) { // Convert boolean-like value to boolean
|
||||
this.value = !!parseFloat(this.value);
|
||||
}
|
||||
} else if (this.fieldSelector.selectedField.type === "selection") {
|
||||
if (!_.some(this.fieldSelector.selectedField.selection, (function (option) { return option[0] === this.value; }).bind(this))) {
|
||||
this.value = this.fieldSelector.selectedField.selection[0][0];
|
||||
}
|
||||
} else if (_.contains(["date", "datetime"], this.fieldSelector.selectedField.type)) {
|
||||
if (couldNotParse || _.isBoolean(this.value)) {
|
||||
this.value = formats.parse_value(formats.format_value(Date.now(), this.fieldSelector.selectedField), this.fieldSelector.selectedField);
|
||||
}
|
||||
} else {
|
||||
if (_.isBoolean(this.value)) { // Never display "true" or "false" strings from boolean value
|
||||
this.value = "";
|
||||
}
|
||||
}
|
||||
|
||||
if (!silent) this.trigger_up("domain_changed", {child: this});
|
||||
},
|
||||
/// The private _getOperatorsFromType returns the mapping of "technical operator" to "display operator value"
|
||||
/// of the operators which are available for the given field type.
|
||||
_getOperatorsFromType: function (type) {
|
||||
var operators = {};
|
||||
|
||||
switch (type) {
|
||||
case "boolean":
|
||||
operators = {
|
||||
"=": _t("is"),
|
||||
"!=": _t("is not"),
|
||||
};
|
||||
break;
|
||||
|
||||
case "char":
|
||||
case "text":
|
||||
case "html":
|
||||
operators = _.pick(operator_mapping, "=", "!=", "ilike", "not ilike", "set", "not set", "in", "not in");
|
||||
break;
|
||||
|
||||
case "many2many":
|
||||
case "one2many":
|
||||
case "many2one":
|
||||
operators = _.pick(operator_mapping, "=", "!=", "ilike", "not ilike", "set", "not set");
|
||||
break;
|
||||
|
||||
case "integer":
|
||||
case "float":
|
||||
case "monetary":
|
||||
operators = _.pick(operator_mapping, "=", "!=", ">", "<", ">=", "<=", "ilike", "not ilike", "set", "not set");
|
||||
break;
|
||||
|
||||
case "selection":
|
||||
operators = _.pick(operator_mapping, "=", "!=", "set", "not set");
|
||||
break;
|
||||
|
||||
case "date":
|
||||
case "datetime":
|
||||
operators = _.pick(operator_mapping, "=", "!=", ">", "<", ">=", "<=", "set", "not set");
|
||||
break;
|
||||
|
||||
default:
|
||||
operators = _.extend({}, operator_mapping);
|
||||
break;
|
||||
}
|
||||
|
||||
if (this.options.operators) {
|
||||
operators = _.pick.apply(_, [operators].concat(this.options.operators));
|
||||
}
|
||||
|
||||
return operators;
|
||||
},
|
||||
|
||||
on_add_tag: function (e) {
|
||||
if (e.type === "keyup" && e.which !== $.ui.keyCode.ENTER) return;
|
||||
if (!_.contains(["not in", "in"], this.operator)) return;
|
||||
|
||||
var values = _.isArray(this.value) ? this.value.slice() : [];
|
||||
|
||||
var $input = this.$(".o_domain_leaf_value_tags input");
|
||||
var val = $input.val().trim();
|
||||
if (val && values.indexOf(val) < 0) {
|
||||
values.push(val);
|
||||
_.defer(this.onValueChange.bind(this, values));
|
||||
$input.focus();
|
||||
}
|
||||
},
|
||||
on_remove_tag: function (e) {
|
||||
var values = _.isArray(this.value) ? this.value.slice() : [];
|
||||
var val = this.$(e.currentTarget).data("value");
|
||||
|
||||
var index = values.indexOf(val);
|
||||
if (index >= 0) {
|
||||
values.splice(index, 1);
|
||||
_.defer(this.onValueChange.bind(this, values));
|
||||
}
|
||||
},
|
||||
});
|
||||
|
||||
/// The instantiateNode function instantiates a DomainTree if the given domain contains
|
||||
/// several parts and a DomainLeaf if it only contains one part. Returns null otherwise.
|
||||
function instantiateNode(parent, model, domain, options) {
|
||||
if (domain.length > 1) {
|
||||
return new DomainTree(parent, model, domain, options);
|
||||
} else if (domain.length === 1) {
|
||||
return new DomainLeaf(parent, model, domain, options);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
return DomainSelector;
|
||||
});
|
|
@ -0,0 +1,47 @@
|
|||
odoo.define("web.DomainSelectorDialog", function (require) {
|
||||
"use strict";
|
||||
|
||||
var core = require("web.core");
|
||||
var Dialog = require("web.Dialog");
|
||||
var DomainSelector = require("web.DomainSelector");
|
||||
|
||||
var _t = core._t;
|
||||
|
||||
return Dialog.extend({
|
||||
init: function (parent, model, domain, options) {
|
||||
this.model = model;
|
||||
this.options = _.extend({
|
||||
readonly: true,
|
||||
debugMode: false,
|
||||
}, options || {});
|
||||
|
||||
var buttons;
|
||||
if (this.options.readonly) {
|
||||
buttons = [
|
||||
{text: _t("Close"), close: true},
|
||||
];
|
||||
} else {
|
||||
buttons = [
|
||||
{text: _t("Save"), classes: "btn-primary", close: true, click: function () {
|
||||
this.trigger_up("domain_selected", {domain: this.domainSelector.getDomain()});
|
||||
}},
|
||||
{text: _t("Discard"), close: true},
|
||||
];
|
||||
}
|
||||
|
||||
this._super(parent, _.extend({}, {
|
||||
title: _t("Domain"),
|
||||
buttons: buttons,
|
||||
}, options || {}));
|
||||
|
||||
this.domainSelector = new DomainSelector(this, model, domain, options);
|
||||
},
|
||||
start: function () {
|
||||
this.$el.css("overflow", "visible").closest(".modal-dialog").css("height", "auto"); // This restores default modal height (bootstrap) and allows field selector to overflow
|
||||
return $.when(
|
||||
this._super.apply(this, arguments),
|
||||
this.domainSelector.appendTo(this.$el)
|
||||
);
|
||||
},
|
||||
});
|
||||
});
|
|
@ -0,0 +1,19 @@
|
|||
odoo.define("web.domainUtils", function (require) {
|
||||
"use strict";
|
||||
|
||||
var pyeval = require("web.pyeval");
|
||||
|
||||
function domainToString(domain) {
|
||||
if (_.isString(domain)) return domain;
|
||||
return JSON.stringify(domain || []).replace(/false/g, "False").replace(/true/g, "True");
|
||||
}
|
||||
function stringToDomain(domain) {
|
||||
if (!_.isString(domain)) return domain;
|
||||
return pyeval.eval("domain", domain || "[]");
|
||||
}
|
||||
|
||||
return {
|
||||
domainToString: domainToString,
|
||||
stringToDomain: stringToDomain,
|
||||
};
|
||||
});
|
|
@ -0,0 +1,336 @@
|
|||
odoo.define("web.ModelFieldSelector", function (require) {
|
||||
"use strict";
|
||||
|
||||
var core = require("web.core");
|
||||
var Model = require("web.DataModel");
|
||||
var Widget = require("web.Widget");
|
||||
|
||||
var _t = core._t;
|
||||
|
||||
/// The ModelFieldSelector widget can be used to select a particular field chain from a given model.
|
||||
var ModelFieldSelector = Widget.extend({
|
||||
template: "FieldSelector",
|
||||
events: {
|
||||
// Handle popover opening and closing
|
||||
"focusin": function () {
|
||||
clearTimeout(this._hidePopoverTimeout);
|
||||
this.showPopover();
|
||||
},
|
||||
"focusout": function () {
|
||||
this._hidePopoverTimeout = _.defer(this.hidePopover.bind(this));
|
||||
},
|
||||
"click .o_field_selector_close": "hidePopover",
|
||||
|
||||
// Handle popover field navigation
|
||||
"click .o_field_selector_prev_page": "goToPrevPage",
|
||||
"click .o_field_selector_next_page": function (e) {
|
||||
e.stopPropagation();
|
||||
this.goToNextPage(this._getLastPageField($(e.currentTarget).data("name")));
|
||||
},
|
||||
"click li.o_field_selector_select_button": function (e) {
|
||||
this.selectField(this._getLastPageField($(e.currentTarget).data("name")));
|
||||
},
|
||||
|
||||
// Handle a direct change in the debug input
|
||||
"change input": function() {
|
||||
var userChain = this.$input.val();
|
||||
if (!this.options.followRelations) {
|
||||
var fields = userChain.split(".");
|
||||
if (fields.length > 1) {
|
||||
this.do_warn(_t("Relation not allowed"), _t("You cannot follow relations for this field chain construction"));
|
||||
userChain = fields[0];
|
||||
}
|
||||
}
|
||||
this.setChain(userChain);
|
||||
this.validate(true);
|
||||
this._prefill().then(this.displayPage.bind(this, ""));
|
||||
this.trigger_up("field_chain_changed", {chain: this.chain});
|
||||
},
|
||||
|
||||
// Handle keyboard and mouse navigation to build the field chain
|
||||
"mouseover li.o_field_selector_item": function (e) {
|
||||
this.$("li.o_field_selector_item").removeClass("active");
|
||||
$(e.currentTarget).addClass("active");
|
||||
},
|
||||
"keydown": function (e) {
|
||||
if (!this.$popover.is(":visible")) return;
|
||||
var inputHasFocus = this.$input.is(":focus");
|
||||
|
||||
switch (e.which) {
|
||||
case $.ui.keyCode.UP:
|
||||
case $.ui.keyCode.DOWN:
|
||||
e.preventDefault();
|
||||
var $active = this.$("li.o_field_selector_item.active");
|
||||
var $to = $active[e.which === $.ui.keyCode.DOWN ? "next" : "prev"](".o_field_selector_item");
|
||||
if ($to.length) {
|
||||
$active.removeClass("active");
|
||||
$to.addClass("active");
|
||||
this.$popover.focus();
|
||||
|
||||
var $page = $to.closest(".o_field_selector_page");
|
||||
var full_height = $page.height();
|
||||
var el_position = $to.position().top;
|
||||
var el_height = $to.outerHeight();
|
||||
var current_scroll = $page.scrollTop();
|
||||
if (el_position < 0) {
|
||||
$page.scrollTop(current_scroll - el_height);
|
||||
} else if (full_height < el_position + el_height) {
|
||||
$page.scrollTop(current_scroll + el_height);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case $.ui.keyCode.RIGHT:
|
||||
if (inputHasFocus) break;
|
||||
e.preventDefault();
|
||||
var name = this.$("li.o_field_selector_item.active").data("name");
|
||||
if (name) {
|
||||
var field = this._getLastPageField(name);
|
||||
if (field.relation) {
|
||||
this.goToNextPage(field);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case $.ui.keyCode.LEFT:
|
||||
if (inputHasFocus) break;
|
||||
e.preventDefault();
|
||||
this.goToPrevPage();
|
||||
break;
|
||||
case $.ui.keyCode.ESCAPE:
|
||||
e.stopPropagation();
|
||||
this.hidePopover();
|
||||
break;
|
||||
case $.ui.keyCode.ENTER:
|
||||
if (inputHasFocus) break;
|
||||
e.preventDefault();
|
||||
this.selectField(this._getLastPageField(this.$("li.o_field_selector_item.active").data("name")));
|
||||
break;
|
||||
}
|
||||
},
|
||||
},
|
||||
/// The ModelFieldSelector requires a model and a initial field chain to work with.
|
||||
/// @param model - a string with the model name (e.g. "res.partner")
|
||||
/// @param chain - a string with the initial field chain (e.g. "company_id.name")
|
||||
/// @param options - an object with several options:
|
||||
/// - filters: an object which contains suboptions which determine the fields which are used
|
||||
/// - searchable: a boolean which is true if only the searchable fields have to be used (true by default)
|
||||
/// - fields: the list of fields info to use when no relation has been followed (default to null,
|
||||
/// which indicates that the widget has to request the model fields itself)
|
||||
/// - followRelations: allow to follow relation when building the chain (true by default)
|
||||
/// - debugMode: a boolean which is true if the widget is in debug mode (false by default)
|
||||
init: function (parent, model, chain, options) {
|
||||
this._super.apply(this, arguments);
|
||||
|
||||
this.model = model;
|
||||
this.chain = chain;
|
||||
this.options = _.extend({
|
||||
filters: {},
|
||||
fields: null,
|
||||
followRelations: true,
|
||||
debugMode: false,
|
||||
}, options || {});
|
||||
this.options.filters = _.extend({
|
||||
searchable: true,
|
||||
}, this.options.filters);
|
||||
|
||||
this.pages = [];
|
||||
this.selectedField = false;
|
||||
this.isSelected = true;
|
||||
this.dirty = false;
|
||||
},
|
||||
willStart: function () {
|
||||
return $.when(
|
||||
this._super.apply(this, arguments),
|
||||
this._prefill()
|
||||
);
|
||||
},
|
||||
start: function () {
|
||||
this.$input = this.$("input");
|
||||
this.$popover = this.$(".o_field_selector_popover");
|
||||
this.displayPage();
|
||||
|
||||
return this._super.apply(this, arguments);
|
||||
},
|
||||
/// The setChain method saves a new field chain string and displays it in the DOM input element.
|
||||
/// @param chain - the new field chain string
|
||||
setChain: function (chain) {
|
||||
this.chain = chain;
|
||||
this.$input.val(this.chain);
|
||||
},
|
||||
/// The addChainNode method adds a field name to the current field chain.
|
||||
/// @param fieldName - the new field name to add at the end of the current field chain
|
||||
addChainNode: function (fieldName) {
|
||||
this.dirty = true;
|
||||
if (this.isSelected) {
|
||||
this.removeChainNode();
|
||||
this.isSelected = false;
|
||||
}
|
||||
if (!this.valid) {
|
||||
this.setChain("");
|
||||
this.validate(true);
|
||||
}
|
||||
this.setChain((this.chain ? (this.chain + ".") : "") + fieldName);
|
||||
},
|
||||
/// The removeChainNode method removes the last field name at the end of the current field chain.
|
||||
removeChainNode: function () {
|
||||
this.dirty = true;
|
||||
this.setChain(this.chain.substring(0, this.chain.lastIndexOf(".")));
|
||||
},
|
||||
/// The validate method toggles the valid status of the widget and display the error message if it
|
||||
/// is not valid.
|
||||
/// @param valid - a boolean which is true if the widget is valid
|
||||
validate: function (valid) {
|
||||
this.$(".o_field_selector_warning").toggleClass("hidden", valid);
|
||||
this.valid = valid;
|
||||
},
|
||||
/// The showPopover method shows the popover to select the field chain. It prepares the popover pages
|
||||
/// before actually showing it. (if already open, does nothing)
|
||||
showPopover: function () {
|
||||
if (this._isOpen) return;
|
||||
this._isOpen = true;
|
||||
this._prefill().then((function () {
|
||||
this.displayPage();
|
||||
this.$popover.removeClass("hidden");
|
||||
}).bind(this));
|
||||
},
|
||||
/// The hidePopover method closes the popover and mark the field as selected. If the field chain changed,
|
||||
/// it notifies its parents. (if not open, does nothing)
|
||||
hidePopover: function () {
|
||||
if (!this._isOpen) return;
|
||||
this._isOpen = false;
|
||||
this.$popover.addClass("hidden");
|
||||
this.isSelected = true;
|
||||
if (this.dirty) {
|
||||
this.trigger_up("field_chain_changed", {chain: this.chain});
|
||||
this.dirty = false;
|
||||
}
|
||||
},
|
||||
/// The private _prefill method prepares the popover by filling its pages according to the current field chain.
|
||||
/// @return a deferred which is resolved once the last page is shown
|
||||
_prefill: function () {
|
||||
this.pages = [];
|
||||
return this._pushPageData(this.model).then((function() {
|
||||
return (this.chain ? processChain.call(this, this.chain.split(".").reverse()) : $.when());
|
||||
}).bind(this));
|
||||
|
||||
function processChain(chain) {
|
||||
var field = this._getLastPageField(chain.pop());
|
||||
if (field && field.relation && chain.length > 0) { // Fetch next chain node if any and possible
|
||||
return this._pushPageData(field.relation).then(processChain.bind(this, chain));
|
||||
} else if (field && chain.length === 0) { // Last node fetched, save it
|
||||
this.selectedField = field;
|
||||
this.validate(true);
|
||||
} else { // Wrong node chain
|
||||
this.validate(false);
|
||||
}
|
||||
return $.when();
|
||||
}
|
||||
},
|
||||
/// The private _pushPageData method gets the field of a particular model and adds them for the new
|
||||
/// last popover page.
|
||||
/// @param model - the model name whose fields have to be fetched
|
||||
/// @return a deferred which is resolved once the fields have been added
|
||||
_pushPageData: function (model) {
|
||||
var def;
|
||||
if (this.model === model && this.options.fields) {
|
||||
def = $.when(sortFields(this.options.fields));
|
||||
} else {
|
||||
def = fieldsCache.getFields(model, this.options.filters);
|
||||
}
|
||||
return def.then((function (fields) {
|
||||
this.pages.push(fields);
|
||||
}).bind(this));
|
||||
},
|
||||
/// The displayPage method shows the last page content of the popover. It also adapts the title according
|
||||
/// to the previous page.
|
||||
/// @param animation - an optional animation class to add to the page
|
||||
displayPage: function (animation) {
|
||||
this.$(".o_field_selector_prev_page").toggleClass("hidden", this.pages.length === 1);
|
||||
|
||||
var page = _.last(this.pages);
|
||||
var title = "";
|
||||
if (this.pages.length > 1) {
|
||||
var chainParts = this.chain.split(".");
|
||||
var prevField = _.findWhere(this.pages[this.pages.length - 2], {
|
||||
name: this.isSelected ? chainParts[chainParts.length - 2] : _.last(chainParts),
|
||||
});
|
||||
if (prevField) title = prevField.string;
|
||||
}
|
||||
this.$(".o_field_selector_popover_header .o_field_selector_title").text(title);
|
||||
this.$(".o_field_selector_page").replaceWith(core.qweb.render("FieldSelector.page", {
|
||||
lines: page,
|
||||
followRelations: this.options.followRelations,
|
||||
animation: animation,
|
||||
debug: this.options.debugMode,
|
||||
}));
|
||||
},
|
||||
/// The goToPrevPage method removes the last page, adapts the field chain and displays the new last page.
|
||||
goToPrevPage: function () {
|
||||
if (this.pages.length <= 1) return;
|
||||
this.pages.pop();
|
||||
this.removeChainNode();
|
||||
this.selectedField = this._getLastPageField(_.last(this.chain.split(".")));
|
||||
this.displayPage("o_animate_slide_left");
|
||||
},
|
||||
/// The goToNextPage method adds a new page to the popover following the given field relation and adapts
|
||||
/// the chain node according to this given field.
|
||||
/// @param field - the field to add to the chain node
|
||||
goToNextPage: function (field) {
|
||||
this.addChainNode(field.name);
|
||||
this.selectedField = field;
|
||||
this._pushPageData(field.relation).then(this.displayPage.bind(this, "o_animate_slide_right"));
|
||||
},
|
||||
/// The selectField method selects the given field and adapts the chain node according to it. It also closes
|
||||
/// the popover and thus notifies the parents about the change.
|
||||
/// @param field - the field to select
|
||||
selectField: function (field) {
|
||||
this.addChainNode(field.name);
|
||||
this.selectedField = field;
|
||||
this.hidePopover();
|
||||
},
|
||||
/// The private _getLastPageField search a field in the last page by its name.
|
||||
/// @return the field data (an object) found in the last popover page thanks to its name
|
||||
_getLastPageField: function (name) {
|
||||
return _.findWhere(_.last(this.pages), {
|
||||
name: name,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
/// Field Selector Cache
|
||||
///
|
||||
/// * Stores fields per model used in field selector
|
||||
/// * Apply filters on the fly
|
||||
var fieldsCache = {
|
||||
cache: {},
|
||||
cacheDefs: {},
|
||||
getFields: function (model, filters) {
|
||||
return (this.cacheDefs[model] ? this.cacheDefs[model] : this.updateCache(model)).then((function () {
|
||||
return this.filter(model, filters);
|
||||
}).bind(this));
|
||||
},
|
||||
updateCache: function (model) {
|
||||
this.cacheDefs[model] = new Model(model).call("fields_get", [
|
||||
false,
|
||||
["store", "searchable", "type", "string", "relation", "selection", "related"],
|
||||
]).then((function (fields) {
|
||||
this.cache[model] = sortFields(fields);
|
||||
}).bind(this));
|
||||
return this.cacheDefs[model];
|
||||
},
|
||||
filter: function (model, filters) {
|
||||
return _.filter(this.cache[model], function (f) {
|
||||
return !filters.searchable || f.searchable;
|
||||
});
|
||||
},
|
||||
};
|
||||
|
||||
function sortFields(fields) {
|
||||
return _.chain(fields)
|
||||
.pairs()
|
||||
.sortBy(function (p) { return p[1].string; })
|
||||
.map(function (p) { return _.extend({name: p[0]}, p[1]); })
|
||||
.value();
|
||||
}
|
||||
|
||||
return ModelFieldSelector;
|
||||
});
|
|
@ -0,0 +1,166 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!-- Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). -->
|
||||
|
||||
<template>
|
||||
|
||||
<div t-name="FieldDomain" t-attf-class="o_form_field_domain#{widget.get('effective_readonly') ? '' : ' o_edit_mode'}#{widget.options.in_dialog ? '' : ' o_inline_mode'}">
|
||||
<!-- domain selector will be instantiated here -->
|
||||
<div class="o_form_field_domain_panel">
|
||||
<i class="fa fa-arrow-right"/>
|
||||
<button class="btn btn-xs btn-default o_domain_show_selection_button" type="button">
|
||||
<span class="o_domain_records_count"/> record(s)
|
||||
</button>
|
||||
<span class="text-warning o_domain_error_message hidden"><i class="fa fa-exclamation-triangle"/> Invalid domain</span>
|
||||
<button t-if="widget.options.in_dialog && !widget.get('effective_readonly')"
|
||||
class="btn btn-xs btn-primary o_form_field_domain_dialog_button">Edit domain</button>
|
||||
</div>
|
||||
<div class="o_domain_model_missing o_hidden">Select a model to add a filter.</div>
|
||||
</div>
|
||||
|
||||
<t t-name="DomainNode.ControlPanel">
|
||||
<div t-if="!widget.readonly && !widget.noControlPanel" class="btn-group btn-group-sm pull-right o_domain_node_control_panel">
|
||||
<button class="btn btn-default o_domain_delete_node_button"><i class="fa fa-minus"/></button>
|
||||
<button class="btn btn-default o_domain_add_node_button"><i class="fa fa-plus"/></button>
|
||||
<button class="btn btn-default o_domain_add_node_button" data-branch="1"><i class="fa fa-ellipsis-h"/></button>
|
||||
</div>
|
||||
</t>
|
||||
<t t-name="DomainTree.OperatorSelector">
|
||||
<div t-if="!widget.readonly" class="btn-group o_domain_tree_operator_selector">
|
||||
<button class="btn btn-xs btn-primary o_domain_tree_operator_caret" data-toggle="dropdown">
|
||||
<t t-if="widget.operator === '&'">All</t>
|
||||
<t t-if="widget.operator === '|'">Any</t>
|
||||
<t t-if="widget.operator === '!'">None</t>
|
||||
</button>
|
||||
<ul class="dropdown-menu">
|
||||
<li><a href="#" data-operator="&">All</a></li>
|
||||
<li><a href="#" data-operator="|">Any</a></li>
|
||||
</ul>
|
||||
</div>
|
||||
<strong t-else="">
|
||||
<t t-if="widget.operator === '&'">ALL</t>
|
||||
<t t-if="widget.operator === '|'">ANY</t>
|
||||
<t t-if="widget.operator === '!'">NONE</t>
|
||||
</strong>
|
||||
</t>
|
||||
<div t-name="DomainSelector" t-attf-class="o_domain_node o_domain_tree o_domain_selector #{widget.readonly ? 'o_read_mode' : 'o_edit_mode'}">
|
||||
<t t-if="widget.children.length === 0">
|
||||
<span>Match <strong>all records</strong></span>
|
||||
<button t-if="!widget.readonly" class="btn btn-xs btn-primary o_domain_add_first_node_button"><i class="fa fa-plus"/> Add filter</button>
|
||||
</t>
|
||||
<t t-else="">
|
||||
<div>
|
||||
<t t-if="widget.children.length === 1">Match records with the following rule:</t>
|
||||
<t t-else="">
|
||||
<span>Match records with</span>
|
||||
<t t-call="DomainTree.OperatorSelector"/>
|
||||
<span>of the following rules:</span>
|
||||
</t>
|
||||
</div>
|
||||
|
||||
<div class="o_domain_node_children_container"/>
|
||||
</t>
|
||||
|
||||
<input t-if="widget.debug && !widget.readonly" type="text" class="o_domain_debug_input mt16"/>
|
||||
</div>
|
||||
<div t-name="DomainTree" class="o_domain_node o_domain_tree">
|
||||
<t t-call="DomainNode.ControlPanel"/>
|
||||
<t t-call="DomainTree.OperatorSelector"/>
|
||||
<span>of:</span>
|
||||
|
||||
<div class="o_domain_node_children_container"/>
|
||||
</div>
|
||||
<div t-name="DomainLeaf" t-attf-class="o_domain_node o_domain_leaf #{widget.readonly ? 'o_read_mode' : 'o_edit_mode'}">
|
||||
<t t-call="DomainNode.ControlPanel"/>
|
||||
|
||||
<div t-if="!widget.readonly" class="o_domain_leaf_edition">
|
||||
<!-- field selector will be instantiated here -->
|
||||
<select class="o_domain_leaf_operator_select">
|
||||
<option t-foreach="widget.operators" t-as="key"
|
||||
t-att-value="key"
|
||||
t-att-selected="widget.displayOperator === key ? 'selected' : None">
|
||||
<t t-esc="key_value"/>
|
||||
</option>
|
||||
</select>
|
||||
<div t-attf-class="o_ds_value_cell#{_.contains(['set', 'not set'], widget.displayOperator) ? ' hidden' : ''}">
|
||||
<t t-if="widget.selectionChoices !== null">
|
||||
<select class="o_domain_leaf_value_input">
|
||||
<option t-foreach="widget.selectionChoices" t-as="val"
|
||||
t-att-value="val[0]"
|
||||
t-att-selected="_.contains(val, widget.displayValue) ? 'selected' : None">
|
||||
<t t-esc="val[1]"/>
|
||||
</option>
|
||||
</select>
|
||||
</t>
|
||||
<t t-else="">
|
||||
<t t-if="_.contains(['in', 'not in'], widget.operator)">
|
||||
<div class="o_domain_leaf_value_input">
|
||||
<span class="badge" t-foreach="widget.displayValue" t-as="val">
|
||||
<t t-esc="val"/> <i class="o_domain_leaf_value_remove_tag_button fa fa-times" t-att-data-value="val"/>
|
||||
</span>
|
||||
</div>
|
||||
<div class="o_domain_leaf_value_tags">
|
||||
<input placeholder="Add new value" type="text"/>
|
||||
<button class="btn btn-xs btn-primary fa fa-plus o_domain_leaf_value_add_tag_button"/>
|
||||
</div>
|
||||
</t>
|
||||
<t t-else="">
|
||||
<input class="o_domain_leaf_value_input" type="text" t-att-value="widget.displayValue"/>
|
||||
</t>
|
||||
</t>
|
||||
</div>
|
||||
</div>
|
||||
<div t-else="" class="o_domain_leaf_info">
|
||||
<span class="o_domain_leaf_chain"><t t-esc="widget.chain"/></span>
|
||||
<t t-if="_.isString(widget.value)">
|
||||
<span class="o_domain_leaf_operator"><t t-esc="widget.operator_mapping[widget.operator]"/></span>
|
||||
<span class="o_domain_leaf_value text-primary">"<t t-esc="widget.value"/>"</span>
|
||||
</t>
|
||||
<t t-if="_.isArray(widget.value)">
|
||||
<span class="o_domain_leaf_operator"><t t-esc="widget.operator_mapping[widget.operator]"/></span>
|
||||
<t t-foreach="widget.value" t-as="v">
|
||||
<span class="o_domain_leaf_value text-primary">"<t t-esc="v"/>"</span>
|
||||
<t t-if="!v_last"> or </t>
|
||||
</t>
|
||||
</t>
|
||||
<t t-if="_.isNumber(widget.value)">
|
||||
<span class="o_domain_leaf_operator"><t t-esc="widget.operator_mapping[widget.operator]"/></span>
|
||||
<span class="o_domain_leaf_value text-primary"><t t-esc="widget.value"></t></span>
|
||||
</t>
|
||||
<t t-if="_.isBoolean(widget.value)">
|
||||
is
|
||||
<t t-if="widget.operator === '=' && widget.value === false || widget.operator === '!=' && widget.value === true">not</t>
|
||||
set
|
||||
</t>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div t-name="FieldSelector" class="o_field_selector">
|
||||
<input type="text" t-att-value="widget.chain" t-att-readonly="!(widget.options.debugMode) and 1 or None"/>
|
||||
<div class="o_field_selector_controls" tabindex="0">
|
||||
<i class="fa fa-exclamation-triangle o_field_selector_warning hidden" title="Invalid field chain"/>
|
||||
</div>
|
||||
<div class="o_field_selector_popover hidden" tabindex="0">
|
||||
<div class="o_field_selector_popover_header text-center">
|
||||
<i class="fa fa-arrow-left o_field_selector_popover_option o_field_selector_prev_page"/>
|
||||
<div class="o_field_selector_title"/>
|
||||
<i class="fa fa-times o_field_selector_popover_option o_field_selector_close"/>
|
||||
</div>
|
||||
<div class="o_field_selector_popover_body">
|
||||
<ul class="o_field_selector_page"/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<ul t-name="FieldSelector.page" t-attf-class="o_field_selector_page #{animation}">
|
||||
<t t-foreach="lines" t-as="line">
|
||||
<t t-set="relationToFollow" t-value="followRelations && line.relation"/>
|
||||
<li t-attf-class="o_field_selector_item #{relationToFollow and 'o_field_selector_next_page' or 'o_field_selector_select_button'}#{line_index == 0 and ' active' or ''}"
|
||||
t-att-data-name="line.name">
|
||||
<t t-esc="line.string"/>
|
||||
<div t-if="debug" class="text-muted o_field_selector_item_title"><t t-esc="line.name"/> (<t t-esc="line.type"/>)</div>
|
||||
<i t-if="relationToFollow" class="fa fa-chevron-right o_field_selector_relation_icon"/>
|
||||
</li>
|
||||
</t>
|
||||
</ul>
|
||||
|
||||
</template>
|
|
@ -0,0 +1,156 @@
|
|||
/* Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
* License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). */
|
||||
|
||||
// Many code copied from Odoo, but with modifications https://github.com/odoo/odoo/blob/68176d80ad6053f52ed1c7bcf294ab3664986c46/addons/web/static/src/js/views/form_widgets.js#L396-L528
|
||||
|
||||
odoo.define('web_widget_domain_v11.field', function(require){
|
||||
"use strict";
|
||||
var core = require('web.core');
|
||||
var DomainSelector = require("web.DomainSelector");
|
||||
var DomainSelectorDialog = require("web.DomainSelectorDialog");
|
||||
var common = require('web.form_common');
|
||||
var Model = require('web.DataModel');
|
||||
var pyeval = require('web.pyeval');
|
||||
var session = require('web.session');
|
||||
var _t = core._t;
|
||||
|
||||
/// The "Domain" field allows the user to construct a technical-prefix domain thanks to
|
||||
/// a tree-like interface and see the selected records in real time.
|
||||
/// In debug mode, an input is also there to be able to enter the prefix char domain
|
||||
/// directly (or to build advanced domains the tree-like interface does not allow to).
|
||||
var FieldDomain = common.AbstractField.extend(common.ReinitializeFieldMixin).extend({
|
||||
template: "FieldDomain",
|
||||
events: {
|
||||
"click .o_domain_show_selection_button": function (e) {
|
||||
e.preventDefault();
|
||||
this._showSelection();
|
||||
},
|
||||
"click .o_form_field_domain_dialog_button": function (e) {
|
||||
e.preventDefault();
|
||||
this.openDomainDialog();
|
||||
},
|
||||
},
|
||||
custom_events: {
|
||||
"domain_changed": function (e) {
|
||||
if (this.options.in_dialog) return;
|
||||
this.set_value(this.domainSelector.getDomain(), true);
|
||||
},
|
||||
"domain_selected": function (e) {
|
||||
this.set_value(e.data.domain);
|
||||
},
|
||||
},
|
||||
init: function () {
|
||||
this._super.apply(this, arguments);
|
||||
|
||||
this.valid = true;
|
||||
this.debug = session.debug;
|
||||
this.options = _.defaults(this.options || {}, {
|
||||
in_dialog: false,
|
||||
model: undefined, // this option is mandatory !
|
||||
fs_filters: {}, // Field selector filters (to only show a subset of available fields @see FieldSelector)
|
||||
});
|
||||
if (this.options.model_field && !this.options.model) {
|
||||
this.options.model = this.options.model_field;
|
||||
}
|
||||
},
|
||||
start: function() {
|
||||
this.model = _get_model.call(this); // TODO get the model another way ?
|
||||
this.field_manager.on("view_content_has_changed", this, function () {
|
||||
var currentModel = this.model;
|
||||
this.model = _get_model.call(this);
|
||||
if (currentModel !== this.model) {
|
||||
this.render_value();
|
||||
}
|
||||
});
|
||||
|
||||
return this._super.apply(this, arguments);
|
||||
|
||||
function _get_model() {
|
||||
if (this.field_manager.fields[this.options.model]) {
|
||||
return this.field_manager.get_field_value(this.options.model);
|
||||
}
|
||||
return this.options.model;
|
||||
}
|
||||
},
|
||||
initialize_content: function () {
|
||||
this._super.apply(this, arguments);
|
||||
this.$panel = this.$(".o_form_field_domain_panel");
|
||||
this.$showSelectionButton = this.$panel.find(".o_domain_show_selection_button");
|
||||
this.$recordsCountDisplay = this.$showSelectionButton.find(".o_domain_records_count");
|
||||
this.$errorMessage = this.$panel.find(".o_domain_error_message");
|
||||
this.$modelMissing = this.$(".o_domain_model_missing");
|
||||
},
|
||||
set_value: function (value, noDomainSelectorRender) {
|
||||
this._noDomainSelectorRender = !!noDomainSelectorRender;
|
||||
this._super.apply(this, arguments);
|
||||
this._noDomainSelectorRender = false;
|
||||
},
|
||||
render_value: function() {
|
||||
this._super.apply(this, arguments);
|
||||
|
||||
// If there is no set model, the field should only display the corresponding error message
|
||||
this.$panel.toggleClass("o_hidden", !this.model);
|
||||
this.$modelMissing.toggleClass("o_hidden", !!this.model);
|
||||
if (!this.model) {
|
||||
if (this.domainSelector) {
|
||||
this.domainSelector.destroy();
|
||||
this.domainSelector = undefined;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
var domain = pyeval.eval("domain", this.get("value") || "[]");
|
||||
|
||||
// Recreate domain widget with new domain value
|
||||
if (!this._noDomainSelectorRender) {
|
||||
if (this.domainSelector) {
|
||||
this.domainSelector.destroy();
|
||||
}
|
||||
this.domainSelector = new DomainSelector(this, this.model, domain, {
|
||||
readonly: this.get("effective_readonly") || this.options.in_dialog,
|
||||
fs_filters: this.options.fs_filters,
|
||||
debugMode: session.debug,
|
||||
});
|
||||
this.domainSelector.prependTo(this.$el);
|
||||
}
|
||||
|
||||
// Show number of selected records
|
||||
new Model(this.model).call("search_count", [domain], {
|
||||
context: this.build_context(),
|
||||
}).then((function (data) {
|
||||
this.valid = true;
|
||||
return data;
|
||||
}).bind(this), (function (error, e) {
|
||||
e.preventDefault();
|
||||
this.valid = false;
|
||||
}).bind(this)).always((function (data) {
|
||||
this.$recordsCountDisplay.text(data || 0);
|
||||
this.$showSelectionButton.toggleClass("hidden", !this.valid);
|
||||
this.$errorMessage.toggleClass("hidden", this.valid);
|
||||
}).bind(this));
|
||||
},
|
||||
is_syntax_valid: function() {
|
||||
return this.field_manager.get("actual_mode") === "view" || this.valid;
|
||||
},
|
||||
_showSelection: function() {
|
||||
return new common.SelectCreateDialog(this, {
|
||||
title: _t("Selected records"),
|
||||
res_model: this.model,
|
||||
domain: this.get("value") || "[]",
|
||||
no_create: true,
|
||||
readonly: true,
|
||||
disable_multiple_selection: true,
|
||||
}).open();
|
||||
},
|
||||
openDomainDialog: function () {
|
||||
new DomainSelectorDialog(this, this.model, this.get("value") || "[]", {
|
||||
readonly: this.get("effective_readonly"),
|
||||
fs_filters: this.options.fs_filters,
|
||||
debugMode: session.debug,
|
||||
}).open();
|
||||
},
|
||||
});
|
||||
|
||||
// Replace char_domain widget
|
||||
core.form_widget_registry.add('char_domain', FieldDomain);
|
||||
});
|
|
@ -0,0 +1,26 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||
License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). -->
|
||||
|
||||
<odoo>
|
||||
|
||||
<template id="assets_backend" inherit_id="web.assets_backend">
|
||||
<xpath expr=".">
|
||||
<link rel="stylesheet"
|
||||
href="/web_widget_domain_v11/static/src/copied-css/domain_selector.less"/>
|
||||
<link rel="stylesheet"
|
||||
href="/web_widget_domain_v11/static/src/copied-css/model_field_selector.less"/>
|
||||
<script type="text/javascript"
|
||||
src="/web_widget_domain_v11/static/src/copied-js/domain_selector_dialog.js"/>
|
||||
<script type="text/javascript"
|
||||
src="/web_widget_domain_v11/static/src/copied-js/domain_selector.js"/>
|
||||
<script type="text/javascript"
|
||||
src="/web_widget_domain_v11/static/src/copied-js/domain_utils.js"/>
|
||||
<script type="text/javascript"
|
||||
src="/web_widget_domain_v11/static/src/copied-js/model_field_selector.js"/>
|
||||
<script type="text/javascript"
|
||||
src="/web_widget_domain_v11/static/src/js/domain_field.js"/>
|
||||
</xpath>
|
||||
</template>
|
||||
|
||||
</odoo>
|
Loading…
Reference in New Issue