Holger Brunn
10 years ago
committed by
Jairo Llopis
8 changed files with 470 additions and 0 deletions
-
56web_widget_x2many_2d_matrix/README.rst
-
20web_widget_x2many_2d_matrix/__init__.py
-
45web_widget_x2many_2d_matrix/__openerp__.py
-
BINweb_widget_x2many_2d_matrix/static/description/icon.png
-
0web_widget_x2many_2d_matrix/static/src/css/web_widget_x2many_2d_matrix.css
-
305web_widget_x2many_2d_matrix/static/src/js/web_widget_x2many_2d_matrix.js
-
33web_widget_x2many_2d_matrix/static/src/xml/web_widget_x2many_2d_matrix.xml
-
11web_widget_x2many_2d_matrix/views/templates.xml
@ -0,0 +1,56 @@ |
|||
2D matrix for x2many fields |
|||
=========================== |
|||
|
|||
This module allows to show an x2many field with 3-tuples |
|||
($x_value, $y_value, $value) in a table |
|||
|
|||
+-----------+-----------+-----------+ |
|||
| | $x_value1 | $x_value2 | |
|||
+===========+===========+===========+ |
|||
| $y_value1 | $value1/1 | $value2/1 | |
|||
+-----------+-----------+-----------+ |
|||
| $y_value2 | $value1/2 | $value2/2 | |
|||
+-----------+-----------+-----------+ |
|||
|
|||
where `valuen/n` is editable. |
|||
|
|||
|
|||
Usage |
|||
===== |
|||
|
|||
Use this widget by saying:: |
|||
|
|||
<field name="my_field" widget="x2many_2d_matrix" /> |
|||
|
|||
This assumes that my_field refers to a model with the fields `x`, `y` and |
|||
`value`. If your fields are named differently, pass the correct names as |
|||
attributes:: |
|||
|
|||
<field name="my_field" widget="x2many_2d_matrix" |
|||
field_x_axis="my_field1" field_y_axis="my_field2" field_value="my_field3" /> |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
* ... |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Holger Brunn <hbrunn@therp.nl> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: http://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: http://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 http://odoo-community.org. |
@ -0,0 +1,20 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2015 Therp BV <http://therp.nl>. |
|||
# |
|||
# This program is free software: you can redistribute it and/or modify |
|||
# it under the terms of the GNU Affero General Public License as |
|||
# published by the Free Software Foundation, either version 3 of the |
|||
# License, or (at your option) any later version. |
|||
# |
|||
# This program is distributed in the hope that it will be useful, |
|||
# but WITHOUT ANY WARRANTY; without even the implied warranty of |
|||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|||
# GNU Affero General Public License for more details. |
|||
# |
|||
# You should have received a copy of the GNU Affero General Public License |
|||
# along with this program. If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################## |
@ -0,0 +1,45 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################## |
|||
# |
|||
# OpenERP, Open Source Management Solution |
|||
# This module copyright (C) 2015 Therp BV <http://therp.nl>. |
|||
# |
|||
# This program is free software: you can redistribute it and/or modify |
|||
# it under the terms of the GNU Affero General Public License as |
|||
# published by the Free Software Foundation, either version 3 of the |
|||
# License, or (at your option) any later version. |
|||
# |
|||
# This program is distributed in the hope that it will be useful, |
|||
# but WITHOUT ANY WARRANTY; without even the implied warranty of |
|||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|||
# GNU Affero General Public License for more details. |
|||
# |
|||
# You should have received a copy of the GNU Affero General Public License |
|||
# along with this program. If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################## |
|||
{ |
|||
"name": "2D matrix for x2many fields", |
|||
"version": "1.0", |
|||
"author": "Therp BV", |
|||
"license": "AGPL-3", |
|||
"category": "Hidden/Dependency", |
|||
"summary": "Show list fields as a matrix", |
|||
"depends": [ |
|||
'web', |
|||
], |
|||
"data": [ |
|||
'views/templates.xml', |
|||
], |
|||
"qweb": [ |
|||
'static/src/xml/web_widget_x2many_2d_matrix.xml', |
|||
], |
|||
"test": [ |
|||
], |
|||
"auto_install": False, |
|||
"installable": True, |
|||
"application": False, |
|||
"external_dependencies": { |
|||
'python': [], |
|||
}, |
|||
} |
After Width: 80 | Height: 80 | Size: 1.1 KiB |
@ -0,0 +1,305 @@ |
|||
//-*- coding: utf-8 -*-
|
|||
//############################################################################
|
|||
//
|
|||
// OpenERP, Open Source Management Solution
|
|||
// This module copyright (C) 2015 Therp BV <http://therp.nl>.
|
|||
//
|
|||
// This program is free software: you can redistribute it and/or modify
|
|||
// it under the terms of the GNU Affero General Public License as
|
|||
// published by the Free Software Foundation, either version 3 of the
|
|||
// License, or (at your option) any later version.
|
|||
//
|
|||
// This program is distributed in the hope that it will be useful,
|
|||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|||
// GNU Affero General Public License for more details.
|
|||
//
|
|||
// You should have received a copy of the GNU Affero General Public License
|
|||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||
//
|
|||
//############################################################################
|
|||
|
|||
openerp.web_widget_x2many_2d_matrix = function(instance) |
|||
{ |
|||
instance.web.form.widgets.add( |
|||
'x2many_2d_matrix', |
|||
'instance.web_widget_x2many_2d_matrix.FieldX2Many2dMatrix'); |
|||
instance.web_widget_x2many_2d_matrix.FieldX2Many2dMatrix = instance.web.form.FieldOne2Many.extend({ |
|||
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: {}, |
|||
field_x_axis: 'x', |
|||
field_label_x_axis: 'x', |
|||
field_y_axis: 'y', |
|||
field_label_y_axis: 'y', |
|||
field_value: 'value', |
|||
// 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: {}, |
|||
|
|||
// read parameters
|
|||
init: function(field_manager, node) |
|||
{ |
|||
this.field_x_axis = node.attrs.field_x_axis || this.field_x_axis; |
|||
this.field_y_axis = node.attrs.field_y_axis || this.field_y_axis; |
|||
this.field_label_x_axis = node.attrs.field_label_x_axis || this.field_x_axis; |
|||
this.field_label_y_axis = node.attrs.field_label_y_axis || this.field_y_axis; |
|||
this.field_value = node.attrs.field_value || this.field_value; |
|||
this.show_row_totals = node.attrs.show_row_totals || this.show_row_totals; |
|||
this.show_column_totals = node.attrs.show_column_totals || this.show_column_totals; |
|||
return this._super.apply(this, arguments); |
|||
}, |
|||
|
|||
// return a field's value, id in case it's a one2many field
|
|||
get_field_value: function(row, field, many2one_as_name) |
|||
{ |
|||
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() |
|||
{ |
|||
var self = this, |
|||
result = this._super.apply(this, arguments); |
|||
|
|||
self.by_x_axis = {}; |
|||
self.by_y_axis = {}; |
|||
|
|||
return jQuery.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; |
|||
}).then(function() |
|||
{ |
|||
return self.dataset.read_ids(self.dataset.ids).then(function(rows) |
|||
{ |
|||
var read_many2one = {}, |
|||
many2one_fields = [ |
|||
self.field_x_axis, self.field_y_axis, |
|||
self.field_label_x_axis, self.field_label_y_axis |
|||
]; |
|||
// prepare to read many2one names if necessary (we can get (id, name) or just id as value)
|
|||
_.each(many2one_fields, function(field) |
|||
{ |
|||
if(self.fields[field].type == 'many2one') |
|||
{ |
|||
read_many2one[field] = {}; |
|||
} |
|||
}); |
|||
// setup data structure
|
|||
_.each(rows, function(row) |
|||
{ |
|||
var x = self.get_field_value(row, self.field_x_axis), |
|||
y = self.get_field_value(row, self.field_y_axis); |
|||
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] = row; |
|||
self.by_y_axis[y][x] = row; |
|||
_.each(read_many2one, function(rows, field) |
|||
{ |
|||
if(!_.isArray(row[field])) |
|||
{ |
|||
rows[row[field]] = rows[row[field]] || [] |
|||
rows[row[field]].push(row); |
|||
} |
|||
}); |
|||
}); |
|||
// read many2one fields if necessary
|
|||
var deferrends = []; |
|||
_.each(read_many2one, function(rows, field) |
|||
{ |
|||
if(_.isEmpty(rows)) |
|||
{ |
|||
return; |
|||
} |
|||
var model = new instance.web.Model(self.fields[field].relation); |
|||
deferrends.push(model.call( |
|||
'name_get', |
|||
[_.map(_.keys(rows), function(key) {return parseInt(key)})]) |
|||
.then(function(names) |
|||
{ |
|||
_.each(names, function(name) |
|||
{ |
|||
_.each(rows[name[0]], function(row) |
|||
{ |
|||
row[field] = name; |
|||
}); |
|||
}); |
|||
})); |
|||
}) |
|||
return jQuery.when.apply(jQuery, deferrends); |
|||
}); |
|||
}); |
|||
}); |
|||
}, |
|||
|
|||
// 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 x axis labels
|
|||
get_x_axis_labels: function() |
|||
{ |
|||
var self = this; |
|||
return _.map( |
|||
this.get_x_axis_values(), |
|||
function(val) |
|||
{ |
|||
return self.get_field_value( |
|||
_.first(_.values(self.by_x_axis[val])), |
|||
self.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']; |
|||
}, |
|||
|
|||
// 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) |
|||
{ |
|||
return true; |
|||
}, |
|||
|
|||
// parse a value from user input
|
|||
parse_xy_value: function(val) |
|||
{ |
|||
if(this.is_numeric) |
|||
{ |
|||
return parseFloat(val); |
|||
} |
|||
else |
|||
{ |
|||
return val; |
|||
} |
|||
}, |
|||
|
|||
// format a value from the database for display
|
|||
format_xy_value: function(val) |
|||
{ |
|||
return instance.web.format_value( |
|||
val, {'type': this.fields[this.field_value].type}); |
|||
}, |
|||
|
|||
// compute totals
|
|||
compute_totals: function() |
|||
{ |
|||
var self = this, |
|||
totals_x = {}, |
|||
totals_y = {}; |
|||
return self.dataset.read_ids(self.dataset.ids).then(function(rows) |
|||
{ |
|||
_.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); |
|||
}); |
|||
}).then(function() |
|||
{ |
|||
_.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)); |
|||
}); |
|||
}); |
|||
}, |
|||
|
|||
start: function() |
|||
{ |
|||
var self = this; |
|||
this.$el.find('input').on( |
|||
'change', |
|||
function() |
|||
{ |
|||
var $this = jQuery(this), |
|||
val = $this.val() |
|||
if(self.validate_xy_value(val)) |
|||
{ |
|||
data = {} |
|||
data[self.field_value] = self.parse_xy_value(val); |
|||
self.dataset.write($this.data('id'), data); |
|||
$this.parent().removeClass('oe_form_invalid'); |
|||
self.compute_totals(); |
|||
} |
|||
else |
|||
{ |
|||
$this.parent().addClass('oe_form_invalid'); |
|||
} |
|||
|
|||
}); |
|||
this.compute_totals(); |
|||
return this._super.apply(this, arguments); |
|||
}, |
|||
|
|||
// deactivate view related functions
|
|||
load_views: function() {}, |
|||
reload_current_view: function() {}, |
|||
get_active_view: function() {}, |
|||
}); |
|||
} |
@ -0,0 +1,33 @@ |
|||
<templates> |
|||
<t t-name="FieldX2Many2dMatrix"> |
|||
<div> |
|||
<table class="oe_list_content"> |
|||
<thead> |
|||
<tr class="oe_list_header_columns"> |
|||
<th /> |
|||
<th t-foreach="widget.get_x_axis_labels()" t-as="label"> |
|||
<t t-esc="label" /> |
|||
</th> |
|||
<th t-if="widget.show_row_totals">Total</th> |
|||
</tr> |
|||
</thead> |
|||
<tbody> |
|||
<tr t-foreach="widget.get_y_axis_values()" t-as="y"> |
|||
<th><t t-esc="widget.get_y_axis_label(y)" /></th> |
|||
<td t-foreach="widget.get_x_axis_values()" t-as="x"> |
|||
<span t-att-class="widget.get_xy_value_class()"> |
|||
<input t-att-data-x="x" t-att-data-y="y" t-att-data-id="widget.get_xy_id(x, y)" t-att-value="widget.format_xy_value(widget.get_xy_value(x, y))" /> |
|||
</span> |
|||
</td> |
|||
<td t-if="widget.show_row_totals" class="row_total" t-att-data-y="y"/> |
|||
</tr> |
|||
<tr t-if="widget.show_column_totals"> |
|||
<th>Total</th> |
|||
<td t-foreach="widget.get_x_axis_values()" t-as="x" class="column_total" t-att-data-x="x" /> |
|||
<td /> |
|||
</tr> |
|||
</tbody> |
|||
</table> |
|||
</div> |
|||
</t> |
|||
</templates> |
@ -0,0 +1,11 @@ |
|||
<?xml version="1.0" encoding="UTF-8"?> |
|||
<openerp> |
|||
<data> |
|||
<template id="assets_backend" name="web_widget_x2many_2d_matrix assets" inherit_id="web.assets_backend"> |
|||
<xpath expr="." position="inside"> |
|||
<script type="text/javascript" src="/web_widget_x2many_2d_matrix/static/src/js/web_widget_x2many_2d_matrix.js"></script> |
|||
<link rel="stylesheet" href="/web_widget_x2many_2d_matrix/static/src/css/web_widget_x2many_2d_matrix.css"/> |
|||
</xpath> |
|||
</template> |
|||
</data> |
|||
</openerp> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue