[MIG] web_tree_dynamic_colored_field: Migration to 17.0

- Implementation now is based on updating `style` attribute of `cell/td` instead of  manipulating element's `css` value in js.

- Cleanup docs about no more functioning  `colors` parameter for `tree` (since 13.0)
pull/3107/head
Jurgis Pralgauskis 2024-03-14 10:21:15 +02:00 committed by Enric Tobella
parent 4c780be3d5
commit df68fb69d4
10 changed files with 119 additions and 247 deletions

View File

@ -63,7 +63,7 @@ Usage
</field>
...
With this example, column which renders 'name' field will have its background colored in red.
With this example, column which renders 'name' field will have its **background** colored in red on customer records.
- In the tree view declaration, put
``options='{"fg_color": "white:customer == True"}'`` attribute in the
@ -81,26 +81,7 @@ Usage
</field>
...
With this example, column which renders 'name' field will have its text colored in white on a customer records.
- In the tree view declaration, use
``options='"color_field": "my_color"'`` attribute in the ``tree``
tag:
::
...
<field name="arch" type="xml">
<tree string="View name" colors="color_field: my_color" >
...
<field name="my_color" invisible="1"/>
...
</tree>
</field>
...
- You can also use ``colors="bg_color_field: my_color"`` to defined the
field name that will be used for the background color of the line.
With this example, column which renders 'name' field will have its **text** colored in white on customer records.
- If you want to use more than one color, you can split the attributes
using ';':
@ -123,12 +104,6 @@ Example:
</field>
...
With this example, the content of the field named `my_color` will be used to
populate the `my_color` CSS value. Use a function field to return whichever
color you want depending on the other record values. Note that this
overrides the rest of `colors` attributes, and that you need the tree
to load your field in the first place by adding it as invisible field.
- Can use strings too... In the tree view declaration, put
``options="{'fg_color': 'green:customer_state == \'success\''}"``
attribute in the ``field`` tag:
@ -158,6 +133,8 @@ Known issues / Roadmap
``colors`` attribute is no longer in the RelaxNG schema of the tree
view, so we can't use it anymore. This feature has then been dropped,
but could be reimplement in another way.
- Since version 17.0 coloring is written into ``style`` attribute of
(td) element
Bug Tracker
===========
@ -187,6 +164,7 @@ Contributors
- Guewen Baconnier <guewen.baconnier@camptocamp.com>
- Phuc Tran Thanh <phuc@trobz.com>
- Sylvain LE GAL <https://twitter.com/legalsylvain>
- Jurgis Pralgauskis <jurgis@versada.eu>
Other credits
-------------
@ -194,6 +172,7 @@ Other credits
The development of this module has been financially supported by:
- Camptocamp
- Versada
Maintainers
-----------

View File

@ -4,7 +4,7 @@
"name": "Colorize field in tree views",
"summary": "Allows you to dynamically color fields on tree views",
"category": "Hidden/Dependency",
"version": "15.0.1.0.1",
"version": "17.0.1.0.0",
"depends": ["web"],
"author": "Camptocamp, Therp BV, Odoo Community Association (OCA)",
"license": "AGPL-3",
@ -13,7 +13,8 @@
"installable": True,
"assets": {
"web.assets_backend": [
"/web_tree_dynamic_colored_field/static/src/js/web_tree_dynamic_colored_field.js",
"web_tree_dynamic_colored_field/static/src/xml/list.xml",
"web_tree_dynamic_colored_field/static/src/js/list_renderer.esm.js",
],
},
}

View File

@ -4,3 +4,4 @@
- Guewen Baconnier \<<guewen.baconnier@camptocamp.com>\>
- Phuc Tran Thanh \<<phuc@trobz.com>\>
- Sylvain LE GAL \<<https://twitter.com/legalsylvain>\>
- Jurgis Pralgauskis \<<jurgis@versada.eu>\>

View File

@ -1,3 +1,4 @@
The development of this module has been financially supported by:
- Camptocamp
- Versada

View File

@ -4,3 +4,4 @@
`colors` attribute is no longer in the RelaxNG schema of the tree
view, so we can't use it anymore. This feature has then been dropped,
but could be reimplement in another way.
- Since version 17.0 coloring is written into ``style`` attribute of (td) element

View File

@ -12,7 +12,7 @@
</field>
...
With this example, column which renders 'name' field will have its background colored in red.
With this example, column which renders 'name' field will have its **background** colored in red on customer records.
- In the tree view declaration, put
`options='{"fg_color": "white:customer == True"}'` attribute in the
@ -28,28 +28,12 @@
</field>
...
With this example, column which renders 'name' field will have its text colored in white on a customer records.
- In the tree view declaration, use
`options='"color_field": "my_color"'` attribute in the `tree` tag:
...
<field name="arch" type="xml">
<tree string="View name" colors="color_field: my_color" >
...
<field name="my_color" invisible="1"/>
...
</tree>
</field>
...
- You can also use `colors="bg_color_field: my_color"` to defined the
field name that will be used for the background color of the line.
With this example, column which renders 'name' field will have its **text** colored in white on customer records.
- If you want to use more than one color, you can split the attributes
using ';':
```
```
options='{"fg_color": "red:red_color == True; green:green_color == True"}'
```
@ -65,12 +49,6 @@ Example:
</tree>
</field>
...
With this example, the content of the field named `my_color` will be used to
populate the `my_color` CSS value. Use a function field to return whichever
color you want depending on the other record values. Note that this
overrides the rest of `colors` attributes, and that you need the tree
to load your field in the first place by adding it as invisible field.
```
- Can use strings too... In the tree view declaration, put

View File

@ -1,4 +1,3 @@
<?xml version="1.0" encoding="utf-8"?>
<!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>
@ -406,7 +405,7 @@ color of a cell in tree view</li>
&lt;/field&gt;
...
With this example, column which renders 'name' field will have its background colored in red.
With this example, column which renders 'name' field will have its **background** colored in red on customer records.
</pre>
</li>
<li><p class="first">In the tree view declaration, put
@ -423,27 +422,9 @@ With this example, column which renders 'name' field will have its background co
&lt;/field&gt;
...
With this example, column which renders 'name' field will have its text colored in white on a customer records.
With this example, column which renders 'name' field will have its **text** colored in white on customer records.
</pre>
</li>
<li><p class="first">In the tree view declaration, use
<tt class="docutils literal"><span class="pre">options='&quot;color_field&quot;:</span> &quot;my_color&quot;'</tt> attribute in the <tt class="docutils literal">tree</tt>
tag:</p>
<pre class="literal-block">
...
&lt;field name=&quot;arch&quot; type=&quot;xml&quot;&gt;
&lt;tree string=&quot;View name&quot; colors=&quot;color_field: my_color&quot; &gt;
...
&lt;field name=&quot;my_color&quot; invisible=&quot;1&quot;/&gt;
...
&lt;/tree&gt;
&lt;/field&gt;
...
</pre>
</li>
<li><p class="first">You can also use <tt class="docutils literal"><span class="pre">colors=&quot;bg_color_field:</span> my_color&quot;</tt> to defined the
field name that will be used for the background color of the line.</p>
</li>
<li><p class="first">If you want to use more than one color, you can split the attributes
using ;:</p>
</li>
@ -461,13 +442,7 @@ options='{&quot;fg_color&quot;: &quot;red:red_color == True; green:green_color =
</span>...<span class="w">
</span><span class="nt">&lt;/tree&gt;</span><span class="w">
</span><span class="nt">&lt;/field&gt;</span><span class="w">
</span>...<span class="w">
</span>With<span class="w"> </span>this<span class="w"> </span>example,<span class="w"> </span>the<span class="w"> </span>content<span class="w"> </span>of<span class="w"> </span>the<span class="w"> </span>field<span class="w"> </span>named<span class="w"> </span>`my_color`<span class="w"> </span>will<span class="w"> </span>be<span class="w"> </span>used<span class="w"> </span>to<span class="w">
</span>populate<span class="w"> </span>the<span class="w"> </span>`my_color`<span class="w"> </span>CSS<span class="w"> </span>value.<span class="w"> </span>Use<span class="w"> </span>a<span class="w"> </span>function<span class="w"> </span>field<span class="w"> </span>to<span class="w"> </span>return<span class="w"> </span>whichever<span class="w">
</span>color<span class="w"> </span>you<span class="w"> </span>want<span class="w"> </span>depending<span class="w"> </span>on<span class="w"> </span>the<span class="w"> </span>other<span class="w"> </span>record<span class="w"> </span>values.<span class="w"> </span>Note<span class="w"> </span>that<span class="w"> </span>this<span class="w">
</span>overrides<span class="w"> </span>the<span class="w"> </span>rest<span class="w"> </span>of<span class="w"> </span>`colors`<span class="w"> </span>attributes,<span class="w"> </span>and<span class="w"> </span>that<span class="w"> </span>you<span class="w"> </span>need<span class="w"> </span>the<span class="w"> </span>tree<span class="w">
</span>to<span class="w"> </span>load<span class="w"> </span>your<span class="w"> </span>field<span class="w"> </span>in<span class="w"> </span>the<span class="w"> </span>first<span class="w"> </span>place<span class="w"> </span>by<span class="w"> </span>adding<span class="w"> </span>it<span class="w"> </span>as<span class="w"> </span>invisible<span class="w"> </span>field.
</span>...
</pre>
<ul>
<li><p class="first">Can use strings too… In the tree view declaration, put
@ -499,6 +474,8 @@ with the name of the field on the <tt class="docutils literal">&lt;tree&gt;</tt>
<tt class="docutils literal">colors</tt> attribute is no longer in the RelaxNG schema of the tree
view, so we cant use it anymore. This feature has then been dropped,
but could be reimplement in another way.</li>
<li>Since version 17.0 coloring is written into <tt class="docutils literal">style</tt> attribute of
(td) element</li>
</ul>
</div>
<div class="section" id="bug-tracker">
@ -529,6 +506,7 @@ If you spotted it first, help us to smash it by providing a detailed and welcome
<li>Guewen Baconnier &lt;<a class="reference external" href="mailto:guewen.baconnier&#64;camptocamp.com">guewen.baconnier&#64;camptocamp.com</a>&gt;</li>
<li>Phuc Tran Thanh &lt;<a class="reference external" href="mailto:phuc&#64;trobz.com">phuc&#64;trobz.com</a>&gt;</li>
<li>Sylvain LE GAL &lt;<a class="reference external" href="https://twitter.com/legalsylvain">https://twitter.com/legalsylvain</a>&gt;</li>
<li>Jurgis Pralgauskis &lt;<a class="reference external" href="mailto:jurgis&#64;versada.eu">jurgis&#64;versada.eu</a>&gt;</li>
</ul>
</div>
<div class="section" id="other-credits">
@ -536,6 +514,7 @@ If you spotted it first, help us to smash it by providing a detailed and welcome
<p>The development of this module has been financially supported by:</p>
<ul class="simple">
<li>Camptocamp</li>
<li>Versada</li>
</ul>
</div>
<div class="section" id="maintainers">

View File

@ -0,0 +1,78 @@
/** @odoo-module **/
import {patch} from "@web/core/utils/patch";
import {ListRenderer} from "@web/views/list/list_renderer";
import {evaluateBooleanExpr} from "@web/core/py_js/py";
patch(ListRenderer.prototype, {
/**
* @param {Object} column represents field
* @param {Record} record
* @returns {String} style code for the html element
*/
getDynamicColoredStyle(column, record) {
let style = "";
let color = this.getDynamicColor(column, record, "bg_color");
if (color !== undefined) {
style += `background-color: ${color};`;
}
color = this.getDynamicColor(column, record, "fg_color");
if (color !== undefined) {
// $td.css('color', color);
style += `color: ${color};`;
}
return style;
},
/**
* Return the `color` that has truthfull expresssion
*
* @param column {Object} represents field
* @param record {Record}
* @param color_target {String} 'bg_color' or 'fg_color'
* @returns {String | undefined} color
*/
getDynamicColor(column, record, color_target) {
if (color_target in column.options) {
const definition = column.options[color_target];
let result = "";
for (const color_def of definition.split(";")) {
const color_to_expression = this.pairColorParse(color_def);
if (color_to_expression !== undefined) {
const [color, expression] = color_to_expression;
if (
evaluateBooleanExpr(
expression,
record.evalContextWithVirtualIds
)
) {
// We don't return first match,
// as it can be default color (with "True" expression),
// and later more precise condition may be found.
result = color;
}
}
}
return result || undefined;
}
},
/**
* @param {String} pairColor `color: expression` pair
* @returns {Array} undefined or array of color, expression
*/
pairColorParse: function (pairColor) {
if (pairColor !== "") {
var pairList = pairColor.split(":"),
color = pairList[0],
// If one passes a bare color instead of an expression,
// then we consider that color is to be shown in any case
expression = pairList[1] ? pairList[1] : "True";
return [color, expression];
}
return undefined;
},
});

View File

@ -1,165 +0,0 @@
odoo.define("web_tree_dynamic_colored_field", function (require) {
"use strict";
var ListRenderer = require("web.ListRenderer");
var pyUtils = require("web.py_utils");
var py = window.py;
ListRenderer.include({
/**
* Look up for a `color_field` or ``bg_color_field`` parameter in tree `colors` attribute
*
* @override
*/
_renderBody: function () {
if (this.arch.attrs.colors) {
var colorAttr = this.arch.attrs.colors.split(";");
if (colorAttr.length > 0) {
var colorType = colorAttr[0].split(":")[0].trim();
var colorField = colorAttr[0].split(":")[1].trim();
// Validate the presence of that field in tree view
if (
this.state.data.length &&
colorField in this.state.data[0].data
) {
if (colorType === "color_field") {
this.colorField = colorField;
} else if (colorType === "bg_color_field") {
this.bgColorField = colorField;
}
} else {
console.warn(
"No field named '" + colorField + "' present in view."
);
}
}
}
return this._super();
},
/**
* Colorize a cell during it's render
*
* @override
*/
_renderBodyCell: function (record, node) {
var $td = this._super.apply(this, arguments);
var ctx = this.getEvalContext(record);
this.applyColorize($td, record, node, ctx);
return $td;
},
/**
* Colorize the current cell depending on expressions provided.
*
* @param {Element} $td a <td> tag inside a table representing a list view
* @param {Object} record
* @param {Object} node an XML node (must be a <field>)
* @param {Object} ctx evaluation context for the record
*/
applyColorize: function ($td, record, node, ctx) {
if (!node.attrs.options) {
return;
}
if (node.tag !== "field") {
return;
}
var treeBgColor = record.data[this.bgColorField];
if (treeBgColor) {
$td.css("background-color", treeBgColor);
}
// Apply <field>'s own `options`
if (!node.attrs.options) {
return;
}
if (node.tag !== "field") {
return;
}
var nodeOptions = node.attrs.options;
if (!_.isObject(nodeOptions)) {
nodeOptions = pyUtils.py_eval(nodeOptions);
}
this.applyColorizeHelper($td, nodeOptions, node, "fg_color", "color", ctx);
this.applyColorizeHelper(
$td,
nodeOptions,
node,
"bg_color",
"background-color",
ctx
);
},
/**
* @param {Element} $td a <td> tag inside a table representing a list view
* @param {Object} nodeOptions a mapping of nodeOptions parameters to the color itself
* @param {Object} node an XML node (must be a <field>)
* @param {String} nodeAttribute an attribute of a node to apply a style onto
* @param {String} cssAttribute a real CSS-compatible attribute
* @param {Object} ctx evaluation context for the record
*/
applyColorizeHelper: function (
$td,
nodeOptions,
node,
nodeAttribute,
cssAttribute,
ctx
) {
if (nodeOptions[nodeAttribute]) {
var colors = _(nodeOptions[nodeAttribute].split(";"))
.chain()
.map(this.pairColors)
.value()
.filter(function CheckUndefined(value) {
return value !== undefined;
});
for (var i = 0, len = colors.length; i < len; ++i) {
var pair = colors[i],
color = pair[0],
expression = pair[1];
if (py.evaluate(expression, ctx).toJSON()) {
$td.css(cssAttribute, color);
}
}
}
},
/**
* Parse `<color>: <field> <operator> <value>` forms to
* evaluable expressions
*
* @param {String} pairColor `color: expression` pair
* @returns {Array} undefined or array of color, parsed expression,
* original expression
*/
pairColors: function (pairColor) {
if (pairColor !== "") {
var pairList = pairColor.split(":"),
color = pairList[0],
// If one passes a bare color instead of an expression,
// then we consider that color is to be shown in any case
expression = pairList[1] ? pairList[1] : "True";
return [color, py.parse(py.tokenize(expression)), expression];
}
return undefined;
},
/**
* Construct domain evaluation context, mostly by passing
* record's fields's values to local scope.
*
* @param {Object} record a record to build a context from
* @returns {Object} evaluation context for the record
*/
getEvalContext: function (record) {
var ctx = _.extend({}, record.data, pyUtils.context());
for (var key in ctx) {
var value = ctx[key];
if (ctx[key] instanceof moment) {
// Date/datetime fields are represented w/ Moment objects
// docs: https://momentjs.com/
ctx[key] = value.format("YYYY-MM-DD hh:mm:ss");
}
}
return ctx;
},
});
});

View File

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="UTF-8" ?>
<templates xml:space="preserve">
<t
t-name="web_tree_dynamic_colored_field.ListRenderer.RecordRow"
t-inherit="web.ListRenderer.RecordRow"
t-inherit-mode="extension"
>
<xpath
expr="//t[@t-if=&quot;column.type === 'field'&quot;]//td"
position="attributes"
>
<attribute
name="t-attf-style"
>{{getDynamicColoredStyle(column, record)}}</attribute>
</xpath>
</t>
</templates>