[MIG+REF][11] web_widget_x2many_2d_matrix
The widget has been completely refactored to benefit from the new MVC paradigm introduced in v11.pull/871/head
-
1setup/web_widget_x2many_2d_matrix/odoo/addons/web_widget_x2many_2d_matrix
-
2setup/web_widget_x2many_2d_matrix/setup.cfg
-
6setup/web_widget_x2many_2d_matrix/setup.py
-
47web_widget_x2many_2d_matrix/README.rst
-
7web_widget_x2many_2d_matrix/__manifest__.py
-
BINweb_widget_x2many_2d_matrix/static/description/icon.png
-
BINweb_widget_x2many_2d_matrix/static/description/screenshot.png
-
9web_widget_x2many_2d_matrix/static/src/css/web_widget_x2many_2d_matrix.css
-
416web_widget_x2many_2d_matrix/static/src/js/2d_matrix_renderer.js
-
433web_widget_x2many_2d_matrix/static/src/js/web_widget_x2many_2d_matrix.js
-
172web_widget_x2many_2d_matrix/static/src/js/widget_x2many_2d_matrix.js
-
36web_widget_x2many_2d_matrix/static/src/xml/web_widget_x2many_2d_matrix.xml
-
3web_widget_x2many_2d_matrix/views/assets.xml
@ -0,0 +1 @@ |
|||||
|
../../../../web_widget_x2many_2d_matrix |
@ -0,0 +1,2 @@ |
|||||
|
[bdist_wheel] |
||||
|
universal=1 |
@ -0,0 +1,6 @@ |
|||||
|
import setuptools |
||||
|
|
||||
|
setuptools.setup( |
||||
|
setup_requires=['setuptools-odoo'], |
||||
|
odoo_addon=True, |
||||
|
) |
Before Width: 80 | Height: 80 | Size: 5.0 KiB After Width: 90 | Height: 90 | Size: 2.4 KiB |
Before Width: 914 | Height: 349 | Size: 19 KiB After Width: 1240 | Height: 471 | Size: 22 KiB |
@ -1,8 +1,3 @@ |
|||||
.oe_form_field_x2many_2d_matrix th.oe_link |
|
||||
{ |
|
||||
cursor: pointer; |
|
||||
} |
|
||||
.oe_form_field_x2many_2d_matrix .oe_list_content > tbody > tr > td.oe_list_field_cell |
|
||||
{ |
|
||||
white-space: normal; |
|
||||
|
.o_field_x2many_2d_matrix .row-total { |
||||
|
font-weight: bold; |
||||
} |
} |
@ -0,0 +1,416 @@ |
|||||
|
/* Copyright 2018 Simone Orsi <simone.orsi@camptocamp.com> |
||||
|
* License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). */
|
||||
|
|
||||
|
odoo.define('web_widget_x2many_2d_matrix.X2Many2dMatrixRenderer', function (require) { |
||||
|
"use strict"; |
||||
|
|
||||
|
// heavily inspired by Odoo's `ListRenderer`
|
||||
|
var BasicRenderer = require('web.BasicRenderer'); |
||||
|
var config = require('web.config'); |
||||
|
var field_utils = require('web.field_utils'); |
||||
|
var utils = require('web.utils'); |
||||
|
var FIELD_CLASSES = { |
||||
|
// copied from ListRenderer
|
||||
|
float: 'o_list_number', |
||||
|
integer: 'o_list_number', |
||||
|
monetary: 'o_list_number', |
||||
|
text: 'o_list_text', |
||||
|
}; |
||||
|
|
||||
|
var X2Many2dMatrixRenderer = BasicRenderer.extend({ |
||||
|
|
||||
|
init: function (parent, state, params) { |
||||
|
this._super.apply(this, arguments); |
||||
|
this.editable = params.editable; |
||||
|
this.columns = params.matrix_data.columns; |
||||
|
this.rows = params.matrix_data.rows; |
||||
|
this.matrix_data = params.matrix_data; |
||||
|
}, |
||||
|
/** |
||||
|
* Main render function for the matrix widget. It is rendered as a table. For now, |
||||
|
* this method does not wait for the field widgets to be ready. |
||||
|
* |
||||
|
* @override |
||||
|
* @private |
||||
|
* returns {Deferred} this deferred is resolved immediately |
||||
|
*/ |
||||
|
_renderView: function () { |
||||
|
var self = this; |
||||
|
|
||||
|
this.$el |
||||
|
.removeClass('table-responsive') |
||||
|
.empty(); |
||||
|
|
||||
|
var $table = $('<table>').addClass('o_list_view table table-condensed table-striped'); |
||||
|
this.$el |
||||
|
.addClass('table-responsive') |
||||
|
.append($table); |
||||
|
|
||||
|
this._computeColumnAggregates(); |
||||
|
this._computeRowAggregates(); |
||||
|
|
||||
|
$table |
||||
|
.append(this._renderHeader()) |
||||
|
.append(this._renderBody()); |
||||
|
if (self.matrix_data.show_column_totals) { |
||||
|
$table.append(this._renderFooter()); |
||||
|
} |
||||
|
return this._super(); |
||||
|
}, |
||||
|
/** |
||||
|
* Render the table body. Looks for the table body and renders the rows in it. |
||||
|
* Also it sets the tabindex on every input element. |
||||
|
* |
||||
|
* @private |
||||
|
* return {jQueryElement} The table body element that was just filled. |
||||
|
*/ |
||||
|
_renderBody: function () { |
||||
|
var $body = $('<tbody>').append(this._renderRows()); |
||||
|
_.each($body.find('input'), function (td, i) { |
||||
|
$(td).attr('tabindex', i); |
||||
|
}); |
||||
|
return $body; |
||||
|
}, |
||||
|
/** |
||||
|
* Render the table head of our matrix. Looks for the first table head |
||||
|
* and inserts the header into it. |
||||
|
* |
||||
|
* @private |
||||
|
* @return {jQueryElement} The thead element that was inserted into. |
||||
|
*/ |
||||
|
_renderHeader: function () { |
||||
|
var $tr = $('<tr>').append('<th/>'); |
||||
|
$tr= $tr.append(_.map(this.columns, this._renderHeaderCell.bind(this))); |
||||
|
if (this.matrix_data.show_row_totals) { |
||||
|
$tr.append($('<th/>', {class: 'total'})); |
||||
|
} |
||||
|
return $('<thead>').append($tr); |
||||
|
}, |
||||
|
/** |
||||
|
* Render a single header cell. Creates a th and adds the description as text. |
||||
|
* |
||||
|
* @private |
||||
|
* @param {jQueryElement} node |
||||
|
* @returns {jQueryElement} the created <th> node. |
||||
|
*/ |
||||
|
_renderHeaderCell: function (node) { |
||||
|
var name = node.attrs.name; |
||||
|
var field = this.state.fields[name]; |
||||
|
var $th = $('<th>'); |
||||
|
if (!field) { |
||||
|
return $th; |
||||
|
} |
||||
|
var description; |
||||
|
if (node.attrs.widget) { |
||||
|
description = this.state.fieldsInfo.list[name].Widget.prototype.description; |
||||
|
} |
||||
|
if (description === undefined) { |
||||
|
description = node.attrs.string || field.string; |
||||
|
} |
||||
|
$th.text(description).data('name', name); |
||||
|
|
||||
|
if (field.type === 'float' || field.type === 'integer' || field.type === 'monetary') { |
||||
|
$th.addClass('text-right'); |
||||
|
} |
||||
|
|
||||
|
if (config.debug) { |
||||
|
var fieldDescr = { |
||||
|
field: field, |
||||
|
name: name, |
||||
|
string: description || name, |
||||
|
record: this.state, |
||||
|
attrs: node.attrs, |
||||
|
}; |
||||
|
this._addFieldTooltip(fieldDescr, $th); |
||||
|
} |
||||
|
return $th; |
||||
|
}, |
||||
|
/** |
||||
|
* Proxy call to function rendering single row. |
||||
|
* |
||||
|
* @private |
||||
|
* @returns {String} a string with the generated html. |
||||
|
* |
||||
|
*/ |
||||
|
|
||||
|
_renderRows: function () { |
||||
|
return _.map(this.rows, this._renderRow.bind(this)); |
||||
|
}, |
||||
|
/** |
||||
|
* Render a single row with all its columns. Renders all the cells and then wraps them with a <tr>. |
||||
|
* If aggregate is set on the row it also will generate the aggregate cell. |
||||
|
* |
||||
|
* @private |
||||
|
* @param {Object} row: The row that will be rendered. |
||||
|
* @returns {jQueryElement} the <tr> element that has been rendered. |
||||
|
*/ |
||||
|
_renderRow: function (row) { |
||||
|
var self = this; |
||||
|
var $tr = $('<tr/>', {class: 'o_data_row'}); |
||||
|
$tr = $tr.append(self._renderLabelCell(row.data[0])); |
||||
|
var $cells = _.map(this.columns, function (node, index) { |
||||
|
var record = row.data[index]; |
||||
|
// make the widget use our field value for each cell
|
||||
|
node.attrs.name = self.matrix_data.field_value; |
||||
|
return self._renderBodyCell(record, node, index, {mode:''}); |
||||
|
}); |
||||
|
$tr = $tr.append($cells); |
||||
|
if (row.aggregate) { |
||||
|
$tr.append(self._renderAggregateRowCell(row)); |
||||
|
} |
||||
|
return $tr; |
||||
|
}, |
||||
|
/** |
||||
|
* Renders the label for a specific row. |
||||
|
* |
||||
|
* @private |
||||
|
* @params {Object} record: Contains the information about the record. |
||||
|
* @params {jQueryElement} the cell that was rendered. |
||||
|
*/ |
||||
|
_renderLabelCell: function(record) { |
||||
|
var $td = $('<td>'); |
||||
|
var value = record.data[this.matrix_data.field_y_axis]; |
||||
|
if (value.type == 'record') { |
||||
|
// we have a related record
|
||||
|
value = value.data.display_name; |
||||
|
} |
||||
|
// get 1st column filled w/ Y label
|
||||
|
$td.text(value); |
||||
|
return $td; |
||||
|
}, |
||||
|
/** |
||||
|
* Create a cell and fill it with the aggregate value. |
||||
|
* |
||||
|
* @private |
||||
|
* @param {Object} row: the row object to aggregate. |
||||
|
* @returns {jQueryElement} The rendered cell. |
||||
|
*/ |
||||
|
_renderAggregateRowCell: function (row) { |
||||
|
var $cell = $('<td/>', {class: 'row-total text-right'}); |
||||
|
this._apply_aggregate_value($cell, row.aggregate); |
||||
|
return $cell; |
||||
|
}, |
||||
|
/** |
||||
|
* Render a single body Cell. |
||||
|
* Gets the field and renders the widget. We force the edit mode, since |
||||
|
* we always want the widget to be editable. |
||||
|
* |
||||
|
* @private |
||||
|
* @param {Object} record: Contains the data for this cell |
||||
|
* @param {jQueryElement} node: The HTML of the field. |
||||
|
* @param {int} colIndex: The index of the current column. |
||||
|
* @param {Object} options: The obtions used for the widget |
||||
|
* @returns {jQueryElement} the rendered cell. |
||||
|
*/ |
||||
|
_renderBodyCell: function (record, node, colIndex, options) { |
||||
|
var tdClassName = 'o_data_cell'; |
||||
|
if (node.tag === 'button') { |
||||
|
tdClassName += ' o_list_button'; |
||||
|
} else if (node.tag === 'field') { |
||||
|
var typeClass = FIELD_CLASSES[this.state.fields[node.attrs.name].type]; |
||||
|
if (typeClass) { |
||||
|
tdClassName += (' ' + typeClass); |
||||
|
} |
||||
|
if (node.attrs.widget) { |
||||
|
tdClassName += (' o_' + node.attrs.widget + '_cell'); |
||||
|
} |
||||
|
} |
||||
|
// TODO roadmap: here we should collect possible extra params
|
||||
|
// the user might want to attach to each single cell.
|
||||
|
var $td = $('<td>', { |
||||
|
'class': tdClassName, |
||||
|
'data-form-id': record.id, |
||||
|
'data-id': record.data.id, |
||||
|
}); |
||||
|
// We register modifiers on the <td> element so that it gets the correct
|
||||
|
// modifiers classes (for styling)
|
||||
|
var modifiers = this._registerModifiers(node, record, $td, _.pick(options, 'mode')); |
||||
|
// If the invisible modifiers is true, the <td> element is left empty.
|
||||
|
// Indeed, if the modifiers was to change the whole cell would be
|
||||
|
// rerendered anyway.
|
||||
|
if (modifiers.invisible && !(options && options.renderInvisible)) { |
||||
|
return $td; |
||||
|
} |
||||
|
options.mode = 'edit'; // enforce edit mode
|
||||
|
var widget = this._renderFieldWidget(node, record, _.pick(options, 'mode')); |
||||
|
this._handleAttributes(widget.$el, node); |
||||
|
return $td.append(widget.$el); |
||||
|
}, |
||||
|
/** |
||||
|
* Wraps the column aggregate with a tfoot element |
||||
|
* |
||||
|
* @private |
||||
|
* @returns {jQueryElement} The footer element with the cells in it. |
||||
|
*/ |
||||
|
_renderFooter: function () { |
||||
|
var $cells = this._renderAggregateColCells(); |
||||
|
if ($cells) { |
||||
|
return $('<tfoot>').append($('<tr>').append('<td/>').append($cells)); |
||||
|
} |
||||
|
return; |
||||
|
}, |
||||
|
/** |
||||
|
* Render the Aggregate cells for the column. |
||||
|
* |
||||
|
* @private |
||||
|
* @returns {List} the rendered cells |
||||
|
*/ |
||||
|
_renderAggregateColCells: function () { |
||||
|
var self = this; |
||||
|
return _.map(this.columns, function (column, index) { |
||||
|
var $cell = $('<td>', {class: 'col-total text-right'}); |
||||
|
if (column.aggregate) { |
||||
|
self._apply_aggregate_value($cell, column.aggregate); |
||||
|
} |
||||
|
return $cell; |
||||
|
}); |
||||
|
}, |
||||
|
/** |
||||
|
* Compute the column aggregates. |
||||
|
* This function is called everytime the value is changed. |
||||
|
* |
||||
|
* @private |
||||
|
*/ |
||||
|
_computeColumnAggregates: function () { |
||||
|
if (!this.matrix_data.show_column_totals) { |
||||
|
return; |
||||
|
} |
||||
|
var self = this, |
||||
|
fname = this.matrix_data.field_value, |
||||
|
field = this.state.fields[fname]; |
||||
|
if (!field) { return; } |
||||
|
var type = field.type; |
||||
|
if (type !== 'integer' && type !== 'float' && type !== 'monetary') { |
||||
|
return; |
||||
|
} |
||||
|
_.each(self.columns, function (column, index) { |
||||
|
column.aggregate = { |
||||
|
fname: fname, |
||||
|
ftype: type, |
||||
|
// TODO: translate
|
||||
|
help: 'Sum', |
||||
|
value: 0 |
||||
|
}; |
||||
|
_.each(self.rows, function (row) { |
||||
|
// var record = _.findWhere(self.state.data, {id: col.data.id});
|
||||
|
column.aggregate.value += row.data[index].data[fname]; |
||||
|
}); |
||||
|
}); |
||||
|
}, |
||||
|
/** |
||||
|
* Compute the row aggregates. |
||||
|
* This function is called everytime the value is changed. |
||||
|
* |
||||
|
* @private |
||||
|
*/ |
||||
|
_computeRowAggregates: function () { |
||||
|
if (!this.matrix_data.show_row_totals) { |
||||
|
return; |
||||
|
} |
||||
|
var self = this, |
||||
|
fname = this.matrix_data.field_value, |
||||
|
field = this.state.fields[fname]; |
||||
|
if (!field) { return; } |
||||
|
var type = field.type; |
||||
|
if (type !== 'integer' && type !== 'float' && type !== 'monetary') { |
||||
|
return; |
||||
|
} |
||||
|
_.each(self.rows, function (row) { |
||||
|
row.aggregate = { |
||||
|
fname: fname, |
||||
|
ftype: type, |
||||
|
// TODO: translate
|
||||
|
help: 'Sum', |
||||
|
value: 0 |
||||
|
}; |
||||
|
_.each(row.data, function (col) { |
||||
|
row.aggregate.value += col.data[fname]; |
||||
|
}); |
||||
|
}); |
||||
|
}, |
||||
|
/** |
||||
|
* Takes the given Value, formats it and adds it to the given cell. |
||||
|
* |
||||
|
* @private |
||||
|
* @param {jQueryElement} $cell: The Cell where the aggregate should be added. |
||||
|
* @param {Object} aggregate: The object which contains the information about the aggregate value |
||||
|
*/ |
||||
|
_apply_aggregate_value: function ($cell, aggregate) { |
||||
|
var field = this.state.fields[aggregate.fname], |
||||
|
formatter = field_utils.format[field.type]; |
||||
|
var formattedValue = formatter(aggregate.value, field, {escape: true, }); |
||||
|
$cell.addClass('total').attr('title', aggregate.help).html(formattedValue); |
||||
|
}, |
||||
|
/** |
||||
|
* Check if the change was successful and then update the grid. |
||||
|
* This function is required on relational fields. |
||||
|
* |
||||
|
* @params {Object} state: Contains the current state of the field & all the data |
||||
|
* @params {String} id: the id of the updated object. |
||||
|
* @params {Array} fields: The fields we have in the view. |
||||
|
* @params {Object} ev: The event object. |
||||
|
* @returns {Deferred} The deferred object thats gonna be resolved when the change is made. |
||||
|
*/ |
||||
|
confirmUpdate: function (state, id, fields, ev) { |
||||
|
var self = this; |
||||
|
this.state = state; |
||||
|
return this.confirmChange(state, id, fields, ev).then(function () { |
||||
|
self._refresh(id); |
||||
|
}); |
||||
|
}, |
||||
|
/** |
||||
|
* Refresh our grid. |
||||
|
* |
||||
|
* @private |
||||
|
*/ |
||||
|
_refresh: function (id) { |
||||
|
this._updateRow(id); |
||||
|
this._refreshColTotals(); |
||||
|
this._refreshRowTotals(); |
||||
|
}, |
||||
|
/** |
||||
|
*Update row data in our internal rows. |
||||
|
* |
||||
|
* @params {String} id: The id of the row that needs to be updated. |
||||
|
*/ |
||||
|
_updateRow: function (id) { |
||||
|
var self = this, |
||||
|
record = _.findWhere(self.state.data, {id: id}); |
||||
|
_.each(self.rows, function(row) { |
||||
|
_.each(row.data, function(col, i) { |
||||
|
if (col.id == id) { |
||||
|
row.data[i] = record; |
||||
|
} |
||||
|
}); |
||||
|
}); |
||||
|
}, |
||||
|
/** |
||||
|
* Update the row total. |
||||
|
*/ |
||||
|
_refreshColTotals: function () { |
||||
|
this._computeColumnAggregates(); |
||||
|
this.$('tfoot').replaceWith(this._renderFooter()); |
||||
|
}, |
||||
|
/** |
||||
|
* Update the column total. |
||||
|
*/ |
||||
|
_refreshRowTotals: function () { |
||||
|
var self = this; |
||||
|
this._computeRowAggregates(); |
||||
|
var $rows = self.$el.find('tr.o_data_row'); |
||||
|
_.each(self.rows, function(row, i) { |
||||
|
if (row.aggregate) { |
||||
|
$($rows[i]).find('.row-total') |
||||
|
.replaceWith(self._renderAggregateRowCell(row)); |
||||
|
} |
||||
|
}); |
||||
|
}, |
||||
|
/* |
||||
|
x2m fields expect this |
||||
|
*/ |
||||
|
getEditableRecordID: function (){ return false;} |
||||
|
|
||||
|
}); |
||||
|
|
||||
|
return X2Many2dMatrixRenderer; |
||||
|
}); |
@ -1,433 +0,0 @@ |
|||||
/* Copyright 2015 Holger Brunn <hbrunn@therp.nl> |
|
||||
* Copyright 2016 Pedro M. Baeza <pedro.baeza@tecnativa.com> |
|
||||
* License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). */
|
|
||||
|
|
||||
odoo.define('web_widget_x2many_2d_matrix.widget', function (require) { |
|
||||
"use strict"; |
|
||||
|
|
||||
var core = require('web.core'); |
|
||||
var FieldManagerMixin = require('web.FieldManagerMixin'); |
|
||||
var Widget = require('web.Widget'); |
|
||||
var fieldRegistry = require('web.field_registry'); |
|
||||
var widgetRegistry = require('web.widget_registry'); |
|
||||
var widgetOne2many = widgetRegistry.get('one2many'); |
|
||||
var data = require('web.data'); |
|
||||
var $ = require('jquery'); |
|
||||
|
|
||||
var WidgetX2Many2dMatrix = widgetOne2Many.extend(FieldManagerMixin, { |
|
||||
template: 'FieldX2Many2dMatrix', |
|
||||
widget_class: 'oe_form_field_x2many_2d_matrix', |
|
||||
|
|
||||
// those will be filled with rows from the dataset
|
|
||||
by_x_axis: {}, |
|
||||
by_y_axis: {}, |
|
||||
by_id: {}, |
|
||||
// configuration values
|
|
||||
field_x_axis: 'x', |
|
||||
field_label_x_axis: 'x', |
|
||||
field_y_axis: 'y', |
|
||||
field_label_y_axis: 'y', |
|
||||
field_value: 'value', |
|
||||
x_axis_clickable: true, |
|
||||
y_axis_clickable: true, |
|
||||
// information about our datatype
|
|
||||
is_numeric: false, |
|
||||
show_row_totals: true, |
|
||||
show_column_totals: true, |
|
||||
// this will be filled with the model's fields_get
|
|
||||
fields: {}, |
|
||||
// Store fields used to fill HTML attributes
|
|
||||
fields_att: {}, |
|
||||
|
|
||||
parse_boolean: function(val) |
|
||||
{ |
|
||||
if (val.toLowerCase() === 'true' || val === '1') { |
|
||||
return true; |
|
||||
} |
|
||||
return false; |
|
||||
}, |
|
||||
|
|
||||
// read parameters
|
|
||||
init: function (parent, fieldname, record, therest) { |
|
||||
var res = this._super(parent, fieldname, record, therest); |
|
||||
FieldManagerMixin.init.call(this); |
|
||||
var node = record.fieldsInfo[therest.viewType][fieldname]; |
|
||||
|
|
||||
this.field_x_axis = node.field_x_axis || this.field_x_axis; |
|
||||
this.field_y_axis = node.field_y_axis || this.field_y_axis; |
|
||||
this.field_label_x_axis = node.field_label_x_axis || this.field_x_axis; |
|
||||
this.field_label_y_axis = node.field_label_y_axis || this.field_y_axis; |
|
||||
this.x_axis_clickable = this.parse_boolean(node.x_axis_clickable || '1'); |
|
||||
this.y_axis_clickable = this.parse_boolean(node.y_axis_clickable || '1'); |
|
||||
this.field_value = node.field_value || this.field_value; |
|
||||
for (var property in node) { |
|
||||
if (property.startsWith("field_att_")) { |
|
||||
this.fields_att[property.substring(10)] = node[property]; |
|
||||
} |
|
||||
} |
|
||||
this.field_editability = node.field_editability || this.field_editability; |
|
||||
this.show_row_totals = this.parse_boolean(node.show_row_totals || '1'); |
|
||||
this.show_column_totals = this.parse_boolean(node.show_column_totals || '1'); |
|
||||
this.init_fields(); |
|
||||
// this.set_value(undefined);
|
|
||||
|
|
||||
return res; |
|
||||
}, |
|
||||
|
|
||||
init_fields: function() { |
|
||||
return; |
|
||||
}, |
|
||||
|
|
||||
// return a field's value, id in case it's a one2many field
|
|
||||
get_field_value: function(row, field, many2one_as_name) |
|
||||
// FIXME looks silly
|
|
||||
{ |
|
||||
if(this.fields[field].type == 'many2one' && _.isArray(row[field])) |
|
||||
{ |
|
||||
if(many2one_as_name) |
|
||||
{ |
|
||||
return row[field][1]; |
|
||||
} |
|
||||
else |
|
||||
{ |
|
||||
return row[field][0]; |
|
||||
} |
|
||||
} |
|
||||
return row[field]; |
|
||||
}, |
|
||||
|
|
||||
// setup our datastructure for simple access in the template
|
|
||||
set_value: function(value_) |
|
||||
{ |
|
||||
var self = this, |
|
||||
result = this._super(value_); |
|
||||
|
|
||||
self.by_x_axis = {}; |
|
||||
self.by_y_axis = {}; |
|
||||
self.by_id = {}; |
|
||||
|
|
||||
return $.when(result).then(function() |
|
||||
{ |
|
||||
return self.dataset._model.call('fields_get').then(function(fields) |
|
||||
{ |
|
||||
self.fields = fields; |
|
||||
self.is_numeric = fields[self.field_value].type == 'float'; |
|
||||
self.show_row_totals &= self.is_numeric; |
|
||||
self.show_column_totals &= self.is_numeric; |
|
||||
}) |
|
||||
// if there are cached writes on the parent dataset, read below
|
|
||||
// only returns the written data, which is not enough to properly
|
|
||||
// set up our data structure. Read those ids here and patch the
|
|
||||
// cache
|
|
||||
.then(function() |
|
||||
{ |
|
||||
var ids_written = _.map( |
|
||||
self.dataset.to_write, function(x) { return x.id }); |
|
||||
if(!ids_written.length) |
|
||||
{ |
|
||||
return; |
|
||||
} |
|
||||
return (new data.Query(self.dataset._model)) |
|
||||
.filter([['id', 'in', ids_written]]) |
|
||||
.all() |
|
||||
.then(function(rows) |
|
||||
{ |
|
||||
_.each(rows, function(row) |
|
||||
{ |
|
||||
var cache = _.find( |
|
||||
self.dataset.cache, |
|
||||
function(x) { return x.id == row.id } |
|
||||
); |
|
||||
_.extend(cache.values, row, _.clone(cache.values)); |
|
||||
}) |
|
||||
}) |
|
||||
}) |
|
||||
.then(function() |
|
||||
{ |
|
||||
return self.dataset.read_ids(self.dataset.ids, self.fields).then(function(rows) |
|
||||
{ |
|
||||
// setup data structure
|
|
||||
_.each(rows, function(row) |
|
||||
{ |
|
||||
self.add_xy_row(row); |
|
||||
}); |
|
||||
if(self.is_started && !self.no_rerender) |
|
||||
{ |
|
||||
self.renderElement(); |
|
||||
self.compute_totals(); |
|
||||
self.setup_many2one_axes(); |
|
||||
self.$el.find('.edit').on( |
|
||||
'change', self.proxy(self.xy_value_change)); |
|
||||
self.effective_readonly_change(); |
|
||||
} |
|
||||
}); |
|
||||
}); |
|
||||
}); |
|
||||
}, |
|
||||
|
|
||||
// do whatever needed to setup internal data structure
|
|
||||
add_xy_row: function(row) |
|
||||
{ |
|
||||
var x = this.get_field_value(row, this.field_x_axis), |
|
||||
y = this.get_field_value(row, this.field_y_axis); |
|
||||
// row is a *copy* of a row in dataset.cache, fetch
|
|
||||
// a reference to this row in order to have the
|
|
||||
// internal data structure point to the same data
|
|
||||
// the dataset manipulates
|
|
||||
_.every(this.dataset.cache, function(cached_row) |
|
||||
{ |
|
||||
if(cached_row.id == row.id) |
|
||||
{ |
|
||||
row = cached_row.values; |
|
||||
// new rows don't have that
|
|
||||
row.id = cached_row.id; |
|
||||
return false; |
|
||||
} |
|
||||
return true; |
|
||||
}); |
|
||||
this.by_x_axis[x] = this.by_x_axis[x] || {}; |
|
||||
this.by_y_axis[y] = this.by_y_axis[y] || {}; |
|
||||
this.by_x_axis[x][y] = row; |
|
||||
this.by_y_axis[y][x] = row; |
|
||||
this.by_id[row.id] = row; |
|
||||
}, |
|
||||
|
|
||||
// get x axis values in the correct order
|
|
||||
get_x_axis_values: function() |
|
||||
{ |
|
||||
return _.keys(this.by_x_axis); |
|
||||
}, |
|
||||
|
|
||||
// get y axis values in the correct order
|
|
||||
get_y_axis_values: function() |
|
||||
{ |
|
||||
return _.keys(this.by_y_axis); |
|
||||
}, |
|
||||
|
|
||||
// get the label for a value on the x axis
|
|
||||
get_x_axis_label: function(x) |
|
||||
{ |
|
||||
return this.get_field_value( |
|
||||
_.first(_.values(this.by_x_axis[x])), |
|
||||
this.field_label_x_axis, true); |
|
||||
}, |
|
||||
|
|
||||
// get the label for a value on the y axis
|
|
||||
get_y_axis_label: function(y) |
|
||||
{ |
|
||||
return this.get_field_value( |
|
||||
_.first(_.values(this.by_y_axis[y])), |
|
||||
this.field_label_y_axis, true); |
|
||||
}, |
|
||||
|
|
||||
// return the class(es) the inputs should have
|
|
||||
get_xy_value_class: function() |
|
||||
{ |
|
||||
var classes = 'oe_form_field oe_form_required'; |
|
||||
if(this.is_numeric) |
|
||||
{ |
|
||||
classes += ' oe_form_field_float'; |
|
||||
} |
|
||||
return classes; |
|
||||
}, |
|
||||
|
|
||||
// return row id of a coordinate
|
|
||||
get_xy_id: function(x, y) |
|
||||
{ |
|
||||
return this.by_x_axis[x][y]['id']; |
|
||||
}, |
|
||||
|
|
||||
get_xy_att: function(x, y) |
|
||||
{ |
|
||||
var vals = {}; |
|
||||
for (var att in this.fields_att) { |
|
||||
var val = this.get_field_value( |
|
||||
this.by_x_axis[x][y], this.fields_att[att]); |
|
||||
// Discard empty values
|
|
||||
if (val) { |
|
||||
vals[att] = val; |
|
||||
} |
|
||||
} |
|
||||
return vals; |
|
||||
}, |
|
||||
|
|
||||
// return the value of a coordinate
|
|
||||
get_xy_value: function(x, y) |
|
||||
{ |
|
||||
return this.get_field_value( |
|
||||
this.by_x_axis[x][y], this.field_value); |
|
||||
}, |
|
||||
|
|
||||
// validate a value
|
|
||||
validate_xy_value: function(val) |
|
||||
{ |
|
||||
try |
|
||||
{ |
|
||||
this.parse_xy_value(val); |
|
||||
} |
|
||||
catch(e) |
|
||||
{ |
|
||||
return false; |
|
||||
} |
|
||||
return true; |
|
||||
}, |
|
||||
|
|
||||
// parse a value from user input
|
|
||||
parse_xy_value: function(val) |
|
||||
{ |
|
||||
return val; |
|
||||
}, |
|
||||
|
|
||||
// format a value from the database for display
|
|
||||
format_xy_value: function(val) |
|
||||
{ |
|
||||
return val; |
|
||||
}, |
|
||||
|
|
||||
// compute totals
|
|
||||
compute_totals: function() |
|
||||
{ |
|
||||
var self = this, |
|
||||
grand_total = 0, |
|
||||
totals_x = {}, |
|
||||
totals_y = {}, |
|
||||
rows = this.by_id, |
|
||||
deferred = $.Deferred(); |
|
||||
_.each(rows, function(row) |
|
||||
{ |
|
||||
var key_x = self.get_field_value(row, self.field_x_axis), |
|
||||
key_y = self.get_field_value(row, self.field_y_axis); |
|
||||
totals_x[key_x] = (totals_x[key_x] || 0) + self.get_field_value(row, self.field_value); |
|
||||
totals_y[key_y] = (totals_y[key_y] || 0) + self.get_field_value(row, self.field_value); |
|
||||
grand_total += self.get_field_value(row, self.field_value); |
|
||||
}); |
|
||||
_.each(totals_y, function(total, y) |
|
||||
{ |
|
||||
self.$el.find( |
|
||||
_.str.sprintf('td.row_total[data-y="%s"]', y)).text( |
|
||||
self.format_xy_value(total)); |
|
||||
}); |
|
||||
_.each(totals_x, function(total, x) |
|
||||
{ |
|
||||
self.$el.find( |
|
||||
_.str.sprintf('td.column_total[data-x="%s"]', x)).text( |
|
||||
self.format_xy_value(total)); |
|
||||
}); |
|
||||
self.$el.find('.grand_total').text( |
|
||||
self.format_xy_value(grand_total)) |
|
||||
deferred.resolve({ |
|
||||
totals_x: totals_x, |
|
||||
totals_y: totals_y, |
|
||||
grand_total: grand_total, |
|
||||
rows: rows, |
|
||||
}); |
|
||||
return deferred; |
|
||||
}, |
|
||||
|
|
||||
setup_many2one_axes: function() |
|
||||
{ |
|
||||
if(this.fields[this.field_x_axis].type == 'many2one' && this.x_axis_clickable) |
|
||||
{ |
|
||||
this.$el.find('th[data-x]').addClass('oe_link') |
|
||||
.click(_.partial( |
|
||||
this.proxy(this.many2one_axis_click), |
|
||||
this.field_x_axis, 'x')); |
|
||||
} |
|
||||
if(this.fields[this.field_y_axis].type == 'many2one' && this.y_axis_clickable) |
|
||||
{ |
|
||||
this.$el.find('tr[data-y] th').addClass('oe_link') |
|
||||
.click(_.partial( |
|
||||
this.proxy(this.many2one_axis_click), |
|
||||
this.field_y_axis, 'y')); |
|
||||
} |
|
||||
}, |
|
||||
|
|
||||
many2one_axis_click: function(field, id_attribute, e) |
|
||||
{ |
|
||||
this.do_action({ |
|
||||
type: 'ir.actions.act_window', |
|
||||
name: this.fields[field].string, |
|
||||
res_model: this.fields[field].relation, |
|
||||
res_id: $(e.currentTarget).data(id_attribute), |
|
||||
views: [[false, 'form']], |
|
||||
target: 'current', |
|
||||
}) |
|
||||
}, |
|
||||
|
|
||||
start: function() |
|
||||
{ |
|
||||
var self = this; |
|
||||
this.$el.find('.edit').on( |
|
||||
'change', self.proxy(this.xy_value_change)); |
|
||||
this.compute_totals(); |
|
||||
this.setup_many2one_axes(); |
|
||||
this.on("change:effective_readonly", |
|
||||
this, this.proxy(this.effective_readonly_change)); |
|
||||
this.effective_readonly_change(); |
|
||||
return this._super(); |
|
||||
}, |
|
||||
|
|
||||
xy_value_change: function(e) |
|
||||
{ |
|
||||
var $this = $(e.currentTarget), |
|
||||
val = $this.val(); |
|
||||
if(this.validate_xy_value(val)) |
|
||||
{ |
|
||||
var data = {}, value = this.parse_xy_value(val); |
|
||||
data[this.field_value] = value; |
|
||||
|
|
||||
$this.siblings('.read').text(this.format_xy_value(value)); |
|
||||
$this.val(this.format_xy_value(value)); |
|
||||
|
|
||||
this.dataset.write($this.data('id'), data); |
|
||||
this.by_id[$this.data('id')][this.field_value] = value; |
|
||||
$this.parent().removeClass('oe_form_invalid'); |
|
||||
this.compute_totals(); |
|
||||
} |
|
||||
else |
|
||||
{ |
|
||||
$this.parent().addClass('oe_form_invalid'); |
|
||||
} |
|
||||
|
|
||||
}, |
|
||||
|
|
||||
effective_readonly_change: function() |
|
||||
{ |
|
||||
this.$el |
|
||||
.find('tbody .edit') |
|
||||
.toggle(!this.get('effective_readonly')); |
|
||||
this.$el |
|
||||
.find('tbody .read') |
|
||||
.toggle(this.get('effective_readonly')); |
|
||||
this.$el.find('.edit').first().focus(); |
|
||||
}, |
|
||||
|
|
||||
is_syntax_valid: function() |
|
||||
{ |
|
||||
return this.$el.find('.oe_form_invalid').length == 0; |
|
||||
}, |
|
||||
|
|
||||
load_views: function() { |
|
||||
// Needed for removing the initial empty tree view when the widget
|
|
||||
// is loaded
|
|
||||
var self = this, |
|
||||
result = this._super(); |
|
||||
|
|
||||
return $.when(result).then(function() |
|
||||
{ |
|
||||
self.renderElement(); |
|
||||
self.compute_totals(); |
|
||||
self.$el.find('.edit').on( |
|
||||
'change', self.proxy(self.xy_value_change)); |
|
||||
}); |
|
||||
}, |
|
||||
}); |
|
||||
|
|
||||
fieldRegistry.add( |
|
||||
'x2many_2d_matrix', WidgetX2Many2dMatrix |
|
||||
); |
|
||||
|
|
||||
return { |
|
||||
WidgetX2Many2dMatrix: WidgetX2Many2dMatrix |
|
||||
}; |
|
||||
}); |
|
@ -0,0 +1,172 @@ |
|||||
|
/* Copyright 2015 Holger Brunn <hbrunn@therp.nl> |
||||
|
* Copyright 2016 Pedro M. Baeza <pedro.baeza@tecnativa.com> |
||||
|
* Copyright 2018 Simone Orsi <simone.orsi@camptocamp.com> |
||||
|
* License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). */
|
||||
|
|
||||
|
odoo.define('web_widget_x2many_2d_matrix.widget', function (require) { |
||||
|
"use strict"; |
||||
|
|
||||
|
var core = require('web.core'); |
||||
|
// var FieldManagerMixin = require('web.FieldManagerMixin');
|
||||
|
var field_registry = require('web.field_registry'); |
||||
|
var relational_fields = require('web.relational_fields'); |
||||
|
var weContext = require('web_editor.context'); |
||||
|
// var Helpers = require('web_widget_x2many_2d_matrix.helpers');
|
||||
|
var AbstractField = require('web.AbstractField'); |
||||
|
var X2Many2dMatrixRenderer = require('web_widget_x2many_2d_matrix.X2Many2dMatrixRenderer'); |
||||
|
|
||||
|
var WidgetX2Many2dMatrix = relational_fields.FieldOne2Many.extend({ |
||||
|
widget_class: 'o_form_field_x2many_2d_matrix', |
||||
|
/** |
||||
|
* Initialize the widget & parameters. |
||||
|
* |
||||
|
* @param {Object} parent: contains the form view. |
||||
|
* @param {String} name: the name of the field. |
||||
|
* @param {Object} record: Contains the information about the database records. |
||||
|
* @param {Object} options: Contains the view options. |
||||
|
*/ |
||||
|
init: function (parent, name, record, options) { |
||||
|
this._super(parent, name, record, options); |
||||
|
this.init_params(); |
||||
|
}, |
||||
|
|
||||
|
/** |
||||
|
* Initialize the widget specific parameters. |
||||
|
* Sets the axis and the values. |
||||
|
*/ |
||||
|
init_params: function () { |
||||
|
var node = this.attrs; |
||||
|
this.by_x_axis = {}; |
||||
|
this.by_y_axis = {}; |
||||
|
this.field_x_axis = node.field_x_axis || this.field_x_axis; |
||||
|
this.field_y_axis = node.field_y_axis || this.field_y_axis; |
||||
|
this.field_label_x_axis = node.field_label_x_axis || this.field_x_axis; |
||||
|
this.field_label_y_axis = node.field_label_y_axis || this.field_y_axis; |
||||
|
this.x_axis_clickable = this.parse_boolean(node.x_axis_clickable || '1'); |
||||
|
this.y_axis_clickable = this.parse_boolean(node.y_axis_clickable || '1'); |
||||
|
this.field_value = node.field_value || this.field_value; |
||||
|
// TODO: is this really needed? Holger?
|
||||
|
for (var property in node) { |
||||
|
if (property.startsWith("field_att_")) { |
||||
|
this.fields_att[property.substring(10)] = node[property]; |
||||
|
} |
||||
|
} |
||||
|
// and this?
|
||||
|
this.field_editability = node.field_editability || this.field_editability; |
||||
|
this.show_row_totals = this.parse_boolean(node.show_row_totals || '1'); |
||||
|
this.show_column_totals = this.parse_boolean(node.show_column_totals || '1'); |
||||
|
this.init_matrix(); |
||||
|
}, |
||||
|
/** |
||||
|
* Initializes the Value matrix. |
||||
|
* Puts the values in the grid. If we have related items we use the display name. |
||||
|
*/ |
||||
|
init_matrix: function(){ |
||||
|
var self = this, |
||||
|
records = self.recordData[this.name].data; |
||||
|
_.each(records, function(record) { |
||||
|
var x = record.data[self.field_x_axis], |
||||
|
y = record.data[self.field_y_axis]; |
||||
|
if (x.type == 'record') { |
||||
|
// we have a related record
|
||||
|
x = x.data.display_name; |
||||
|
} |
||||
|
if (y.type == 'record') { |
||||
|
// we have a related record
|
||||
|
y = y.data.display_name; |
||||
|
} |
||||
|
self.by_x_axis[x] = self.by_x_axis[x] || {}; |
||||
|
self.by_y_axis[y] = self.by_y_axis[y] || {}; |
||||
|
self.by_x_axis[x][y] = record; |
||||
|
self.by_y_axis[y][x] = record; |
||||
|
}); |
||||
|
// init columns
|
||||
|
self.columns = []; |
||||
|
$.each(self.by_x_axis, function(x){ |
||||
|
self.columns.push(self._make_column(x)); |
||||
|
}); |
||||
|
self.rows = []; |
||||
|
$.each(self.by_y_axis, function(y){ |
||||
|
self.rows.push(self._make_row(y)); |
||||
|
}); |
||||
|
self.matrix_data = { |
||||
|
'field_value': self.field_value, |
||||
|
'field_x_axis': self.field_x_axis, |
||||
|
'field_y_axis': self.field_y_axis, |
||||
|
'columns': self.columns, |
||||
|
'rows': self.rows, |
||||
|
'show_row_totals': self.show_row_totals, |
||||
|
'show_column_totals': self.show_column_totals |
||||
|
}; |
||||
|
|
||||
|
}, |
||||
|
/** |
||||
|
* Create scaffold for a column. |
||||
|
* |
||||
|
* @params {String} x: The string used as a column title |
||||
|
*/ |
||||
|
_make_column: function(x){ |
||||
|
return { |
||||
|
// simulate node parsed on xml arch
|
||||
|
'tag': 'field', |
||||
|
'attrs': { |
||||
|
'name': this.field_x_axis, |
||||
|
'string': x |
||||
|
} |
||||
|
}; |
||||
|
}, |
||||
|
/** |
||||
|
* Create scaffold for a row. |
||||
|
* |
||||
|
* @params {String} x: The string used as a row title |
||||
|
*/ |
||||
|
_make_row: function(y){ |
||||
|
var self = this; |
||||
|
// use object so that we can attach more data if needed
|
||||
|
var row = {'data': []}; |
||||
|
$.each(self.by_x_axis, function(x) { |
||||
|
row.data.push(self.by_y_axis[y][x]); |
||||
|
}); |
||||
|
return row; |
||||
|
}, |
||||
|
/** |
||||
|
*Parse a String containing a Python bool or 1 and convert it to a proper bool. |
||||
|
* |
||||
|
* @params {String} val: the string to be parsed. |
||||
|
* @returns {Boolean} The parsed boolean. |
||||
|
*/ |
||||
|
parse_boolean: function(val) { |
||||
|
if (val.toLowerCase() === 'true' || val === '1') { |
||||
|
return true; |
||||
|
} |
||||
|
return false; |
||||
|
}, |
||||
|
/** |
||||
|
*Create the matrix renderer and add its output to our element |
||||
|
* |
||||
|
* @returns {Deferred} A deferred object to be completed when it finished rendering. |
||||
|
*/ |
||||
|
_render: function () { |
||||
|
if (!this.view) { |
||||
|
return this._super(); |
||||
|
} |
||||
|
var arch = this.view.arch, |
||||
|
viewType = 'list'; |
||||
|
this.renderer = new X2Many2dMatrixRenderer(this, this.value, { |
||||
|
arch: arch, |
||||
|
editable: true, |
||||
|
viewType: viewType, |
||||
|
matrix_data: this.matrix_data |
||||
|
}); |
||||
|
this.$el.addClass('o_field_x2many o_field_x2many_2d_matrix'); |
||||
|
return this.renderer.appendTo(this.$el); |
||||
|
} |
||||
|
|
||||
|
}); |
||||
|
|
||||
|
field_registry.add('x2many_2d_matrix', WidgetX2Many2dMatrix); |
||||
|
|
||||
|
return { |
||||
|
WidgetX2Many2dMatrix: WidgetX2Many2dMatrix |
||||
|
}; |
||||
|
}); |
@ -1,36 +0,0 @@ |
|||||
<templates> |
|
||||
<t t-name="FieldX2Many2dMatrix"> |
|
||||
<div t-att-class="widget.widget_class"> |
|
||||
<table class="o_list_view table table-condensed table-striped"> |
|
||||
<thead> |
|
||||
<tr class="oe_list_header_columns"> |
|
||||
<th /> |
|
||||
<th t-foreach="widget.get_x_axis_values()" t-as="x" t-att-data-x="x"> |
|
||||
<t t-esc="widget.get_x_axis_label(x)" /> |
|
||||
</th> |
|
||||
<th t-if="widget.show_row_totals">Total</th> |
|
||||
</tr> |
|
||||
</thead> |
|
||||
<tbody> |
|
||||
<tr t-foreach="widget.get_y_axis_values()" t-as="y" t-att-data-y="y"> |
|
||||
<th><t t-esc="widget.get_y_axis_label(y)" /></th> |
|
||||
<td t-foreach="widget.get_x_axis_values()" t-as="x" t-att-class="'' + (widget.is_numeric ? ' oe_number' : '')" t-att-data-x="x"> |
|
||||
<span t-att-class="widget.get_xy_value_class()"> |
|
||||
<input class="edit o_form_input oe_edit_only" t-att-data-id="widget.get_xy_id(x, y)" t-att-value="widget.format_xy_value(widget.get_xy_value(x, y))" t-att="widget.get_xy_att(x, y)"/> |
|
||||
<span class="read oe_read_only"><t t-esc="widget.format_xy_value(widget.get_xy_value(x, y))" /></span> |
|
||||
</span> |
|
||||
</td> |
|
||||
<td t-if="widget.show_row_totals" class="row_total oe_number" t-att-data-y="y"/> |
|
||||
</tr> |
|
||||
</tbody> |
|
||||
<tfoot t-if="widget.show_column_totals"> |
|
||||
<tr> |
|
||||
<th>Total</th> |
|
||||
<td t-foreach="widget.get_x_axis_values()" t-as="x" class="oe_list_footer oe_number column_total" t-att-data-x="x" /> |
|
||||
<td class="grand_total oe_number" /> |
|
||||
</tr> |
|
||||
</tfoot> |
|
||||
</table> |
|
||||
</div> |
|
||||
</t> |
|
||||
</templates> |
|