Browse Source

Merge pull request #111 from grap/7.0-web_dashboard_tile-improve

[7.0] web_dashboard_tile : Fixes & Improvements
pull/142/head
Sylvain LE GAL 10 years ago
parent
commit
885147e5b2
  1. 25
      web_dashboard_tile/__init__.py
  2. 86
      web_dashboard_tile/__openerp__.py
  3. 25
      web_dashboard_tile/demo/res_groups.yml
  4. 33
      web_dashboard_tile/demo/tile_tile.yml
  5. 128
      web_dashboard_tile/i18n/fr.po
  6. 119
      web_dashboard_tile/i18n/web_dashboard_tile.pot
  7. 48
      web_dashboard_tile/migrations/7.0.1.0/post-migration-color.py
  8. 95
      web_dashboard_tile/static/src/css/tile.css
  9. BIN
      web_dashboard_tile/static/src/img/avg.png
  10. BIN
      web_dashboard_tile/static/src/img/max.png
  11. BIN
      web_dashboard_tile/static/src/img/median.png
  12. BIN
      web_dashboard_tile/static/src/img/min.png
  13. BIN
      web_dashboard_tile/static/src/img/screenshot_action_click.png
  14. BIN
      web_dashboard_tile/static/src/img/screenshot_dashboard.png
  15. BIN
      web_dashboard_tile/static/src/img/sum.png
  16. 145
      web_dashboard_tile/tile.py
  17. 45
      web_dashboard_tile/view/tile.xml

25
web_dashboard_tile/__init__.py

@ -1 +1,26 @@
# -*- coding: utf-8 -*-
##############################################################################
#
# OpenERP, Open Source Management Solution
# Copyright (C) 2010-2013 OpenERP s.a. (<http://openerp.com>).
# Copyright (C) 2014 initOS GmbH & Co. KG (<http://www.initos.com>).
# Copyright (C) 2015-Today GRAP
# Author Markus Schneider <markus.schneider at initos.com>
# @author Sylvain LE GAL (https://twitter.com/legalsylvain)
#
# 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/>.
#
##############################################################################
from . import tile

86
web_dashboard_tile/__openerp__.py

@ -22,41 +22,71 @@
##############################################################################
{
"name": "Dashboard Tile",
"version": "0.4",
"depends": ['web', 'board', 'mail'],
'author': "initOS GmbH & Co. KG,Odoo Community Association (OCA)",
"summary": "Add Tiles to Dashboard",
"version": "1.0",
"depends": [
'web',
'board',
'mail',
'web_widget_color',
],
'author': "initOS GmbH & Co. KG,GRAP,Odoo Community Association (OCA)",
"category": "",
'license': 'AGPL-3',
"description": """
module to give you a dashboard where you can configure tile from any view
and add them as short cut.
Add Tiles to Dashboard
======================
Features:
---------
module to give you a dashboard where you can configure tile from any view
and add them as short cut.
Kown issues/limits:
* change color picks wrong color
* can not edit tile from dashboard
* context are ignored
* date filter can not be relative
* combine domain of menue and filter so can not restore origin filter
* Tile can be:
* displayed only for a user;
* global for all users (In that case, some tiles will be hidden if
the current user doesn't have access to the given model);
* The tile displays items count of a given model restricted to a given domain;
* Optionnaly, the tile can display the result of a function of a field;
* Function is one of sum/avg/min/max/median;
* Field must be integer or float;
possible future improvments:
* support context_today
* add icons
* support client side action (like inbox)
* support select int/float column with min/max/avg/sum to display
* change position (maybe drag&drop)
""",
"summary": "Add tile to dashboard",
'data': ['tile.xml',
'security/ir.model.access.csv',
'security/rules.xml'],
'css': ['static/src/css/tile.css'],
Screenshot:
-----------
* Dashboad sample, displaying Sale Orders to invoice:
.. image:: web_dashboard_tile/static/src/img/screenshot_dashboard.png
* Tree view displayed when user click on the tile:
.. image:: web_dashboard_tile/static/src/img/screenshot_action_click.png
Kown issues/limits:
-------------------
* can not edit tile from dashboard (color, sequence, function, ...);
* context are ignored;
* date filter can not be relative;
* combine domain of menue and filter so can not restore origin filter;
possible future improvments:
----------------------------
* support context_today;
* add icons;
* support client side action (like inbox);
""",
'data': [
'view/tile.xml',
'security/ir.model.access.csv',
'security/rules.xml',
],
'css': [
'static/src/css/tile.css',
],
'demo': [
'demo/res_groups.yml',
'demo/tile_tile.yml',
],
'js': [
'static/src/js/custom_js.js',
],
'test': [
'qweb': [
'static/src/xml/custom_xml.xml',
],
'installable': True,
'auto_install': False,
'js': ['static/src/js/custom_js.js'],
'qweb': ['static/src/xml/custom_xml.xml'],
}

25
web_dashboard_tile/demo/res_groups.yml

@ -0,0 +1,25 @@
# -*- coding: utf-8 -*-
##############################################################################
#
# OpenERP, Open Source Management Solution
# Copyright (C) 2015-Today GRAP
# @author Sylvain LE GAL (https://twitter.com/legalsylvain)
#
# 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/>.
#
##############################################################################
- !record {model: res.groups, id: base.group_no_one}:
users:
- base.user_root

33
web_dashboard_tile/demo/tile_tile.yml

@ -0,0 +1,33 @@
# -*- coding: utf-8 -*-
##############################################################################
#
# OpenERP, Open Source Management Solution
# Copyright (C) 2015-Today GRAP
# @author Sylvain LE GAL (https://twitter.com/legalsylvain)
#
# 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/>.
#
##############################################################################
- !record {model: tile.tile, id: installed_modules}:
name: Installed Modules
model_id: base.model_ir_module_module
domain: [['state', 'in', ['installed', 'to upgrade', 'to remove']]]
action_id: base.open_module_tree
- !record {model: tile.tile, id: installed_OCA_modules}:
name: Installed OCA Modules
model_id: base.model_ir_module_module
domain: [['state', 'in', ['installed', 'to upgrade', 'to remove']], ['author', 'ilike', 'Odoo Community Association (OCA)']]
action_id: base.open_module_tree

128
web_dashboard_tile/i18n/fr.po

@ -6,8 +6,8 @@ msgid ""
msgstr ""
"Project-Id-Version: OpenERP Server 7.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2015-04-05 19:36+0000\n"
"PO-Revision-Date: 2015-04-05 19:36+0000\n"
"POT-Creation-Date: 2015-04-10 01:04+0000\n"
"PO-Revision-Date: 2015-04-10 01:04+0000\n"
"Last-Translator: <>\n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
@ -20,6 +20,37 @@ msgstr ""
msgid "Action"
msgstr "Action"
#. module: web_dashboard_tile
#: field:tile.tile,active:0
msgid "Active"
msgstr "Actif"
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Average"
msgstr "Moyenne"
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:82
#, python-format
msgid "Average value of '%s'"
msgstr "Valeur moyenne du champ '%s'"
#. module: web_dashboard_tile
#: field:tile.tile,color:0
msgid "Background color"
msgstr "Couleur de fond"
#. module: web_dashboard_tile
#: field:tile.tile,computed_value:0
msgid "Computed Value"
msgstr "Valeur calculée"
#. module: web_dashboard_tile
#: field:tile.tile,count:0
msgid "Count"
msgstr "Quantité"
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/xml/custom_xml.xml:8
@ -66,6 +97,21 @@ msgstr "Editer..."
msgid "Error"
msgstr "Erreur"
#. module: web_dashboard_tile
#: constraint:tile.tile:0
msgid "Error ! Please select a field of the selected model."
msgstr "Erreur ! Veuillez sélectioner un champ qui correspond au modèle."
#. module: web_dashboard_tile
#: constraint:tile.tile:0
msgid "Error ! Please set both fields: 'Field' and 'Function'."
msgstr "Erreur ! Veuillez renseigner les deux champs : 'Champ' et 'Fonction'."
#. module: web_dashboard_tile
#: field:tile.tile,field_id:0
msgid "Field"
msgstr "Champ"
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:61
@ -74,15 +120,63 @@ msgid "Filter name is required."
msgstr "Le nom du filtre est requis."
#. module: web_dashboard_tile
#: field:tile.tile,color:0
msgid "Kanban Color"
msgstr "Couleur dans la vue Kanban"
#: field:tile.tile,font_color:0
msgid "Font Color"
msgstr "Couleur du texte"
#. module: web_dashboard_tile
#: field:tile.tile,field_function:0
msgid "Function"
msgstr "Fonction"
#. module: web_dashboard_tile
#: field:tile.tile,helper:0
msgid "Helper Text"
msgstr "Texte Descriptif"
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Maximum"
msgstr "Maximum"
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:76
#, python-format
msgid "Maximum value of '%s'"
msgstr "Valeur maximale du champ '%s'"
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Median"
msgstr "Médiane"
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:85
#, python-format
msgid "Median value of '%s'"
msgstr "Valeur médian du champ '%s'"
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Minimum"
msgstr "Minimum"
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:73
#, python-format
msgid "Minimum value of '%s'"
msgstr "Valeur minimale du champ '%s'"
#. module: web_dashboard_tile
#: field:tile.tile,model_id:0
msgid "Model"
msgstr "Modèle"
#. module: web_dashboard_tile
#: field:tile.tile,sequence:0
msgid "Sequence"
msgstr "Séquence"
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:100
@ -90,6 +184,11 @@ msgstr "Modèle"
msgid "Success"
msgstr "Succès"
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Sum"
msgstr "Somme"
#. module: web_dashboard_tile
#: field:tile.tile,name:0
msgid "Tile Name"
@ -100,7 +199,7 @@ msgstr "Nom de l'indicateur"
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:100
#, python-format
msgid "Tile is created"
msgstr "L'indicateur a été crée"
msgstr "L'indicateur a été créé"
#. module: web_dashboard_tile
#. openerp-web
@ -109,18 +208,13 @@ msgstr "L'indicateur a été crée"
msgid "Tile:"
msgstr "Indicateur :"
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:79
#, python-format
msgid "Total value of '%s'"
msgstr "Somme du champ '%s'"
#. module: web_dashboard_tile
#: field:tile.tile,user_id:0
msgid "User"
msgstr "Utilisateur"
#. module: web_dashboard_tile
#: field:tile.tile,count:0
msgid "unknown"
msgstr "inconnu"
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "í"
msgstr "í"

119
web_dashboard_tile/i18n/web_dashboard_tile.pot

@ -6,8 +6,8 @@ msgid ""
msgstr ""
"Project-Id-Version: OpenERP Server 7.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2015-04-05 19:42+0000\n"
"PO-Revision-Date: 2015-04-05 19:42+0000\n"
"POT-Creation-Date: 2015-04-10 01:03+0000\n"
"PO-Revision-Date: 2015-04-10 01:03+0000\n"
"Last-Translator: <>\n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
@ -20,6 +20,37 @@ msgstr ""
msgid "Action"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,active:0
msgid "Active"
msgstr ""
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Average"
msgstr ""
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:82
#, python-format
msgid "Average value of '%s'"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,color:0
msgid "Background color"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,computed_value:0
msgid "Computed Value"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,count:0
msgid "Count"
msgstr ""
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/xml/custom_xml.xml:8
@ -66,6 +97,21 @@ msgstr ""
msgid "Error"
msgstr ""
#. module: web_dashboard_tile
#: constraint:tile.tile:0
msgid "Error ! Please select a field of the selected model."
msgstr ""
#. module: web_dashboard_tile
#: constraint:tile.tile:0
msgid "Error ! Please set both fields: 'Field' and 'Function'."
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,field_id:0
msgid "Field"
msgstr ""
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:61
@ -74,8 +120,51 @@ msgid "Filter name is required."
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,color:0
msgid "Kanban Color"
#: field:tile.tile,font_color:0
msgid "Font Color"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,field_function:0
msgid "Function"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,helper:0
msgid "Helper Text"
msgstr ""
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Maximum"
msgstr ""
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:76
#, python-format
msgid "Maximum value of '%s'"
msgstr ""
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Median"
msgstr ""
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:85
#, python-format
msgid "Median value of '%s'"
msgstr ""
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Minimum"
msgstr ""
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:73
#, python-format
msgid "Minimum value of '%s'"
msgstr ""
#. module: web_dashboard_tile
@ -83,6 +172,11 @@ msgstr ""
msgid "Model"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,sequence:0
msgid "Sequence"
msgstr ""
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:100
@ -90,6 +184,11 @@ msgstr ""
msgid "Success"
msgstr ""
#. module: web_dashboard_tile
#: selection:tile.tile,field_function:0
msgid "Sum"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,name:0
msgid "Tile Name"
@ -109,6 +208,12 @@ msgstr ""
msgid "Tile:"
msgstr ""
#. module: web_dashboard_tile
#: code:addons/web_dashboard_tile/tile.py:79
#, python-format
msgid "Total value of '%s'"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,user_id:0
msgid "User"
@ -121,13 +226,7 @@ msgstr ""
msgid "tile.tile"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,count:0
msgid "unknown"
msgstr ""
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "í"
msgstr ""

48
web_dashboard_tile/migrations/7.0.1.0/post-migration-color.py

@ -0,0 +1,48 @@
# -*- coding: utf-8 -*-
##############################################################################
#
# OpenERP, Open Source Management Solution
# Copyright (C) 2015-Today GRAP
# @author Sylvain LE GAL (https://twitter.com/legalsylvain)
#
# 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/>.
#
##############################################################################
COLOR_NUMERIC_TO_RVB = {
0: '#006015',
1: '#CD2513',
2: '#CDC713',
3: '#57158A',
4: '#0E9B2D',
5: '#7F0C00',
6: '#7F7B00',
7: '#320455',
8: '#CD6E13',
9: '#0E6C7E',
}
def migrate_color(cr):
for old, new in COLOR_NUMERIC_TO_RVB.iteritems():
cr.execute("""
UPDATE tile_tile
SET color='%s', font_color='#FFFFFF'
WHERE color='%s'
""" % (new, old))
def migrate(cr, installed_version):
migrate_color(cr)

95
web_dashboard_tile/static/src/css/tile.css

@ -4,72 +4,41 @@
border: 0;
border-radius: 0;
}
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_count{
font-size: 48px;
font-weight: bold;
position: absolute;
left: 9px;
bottom: 9px;
}
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_label,
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_count_without_computed_value,
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_count_with_computed_value,
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_computed_value {
width: 140px;
text-align: center;
}
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_label{
padding: 9px;
padding: 5px;
font-size: 15px;
}
.openerp .oe_kanban_view .oe_dashbaord_tile a{
color: #fff;
}
.openerp .oe_kanban_view .oe_dashbaord_tile.oe_tile_color_,
.openerp .oe_kanban_view .oe_dashbaord_tile.oe_tile_color_ a,
.openerp .oe_kanban_view .oe_dashbaord_tile .oe_dropdown_menu a{
color: #000;
}
.openerp .oe_kanban_view .oe_tile_color_1,
.openerp .oe_dashbaord_tile a.oe_kanban_color_1{
background: #CD2513;
color: #fff;
}
.openerp .oe_kanban_view .oe_tile_color_2,
.openerp .oe_dashbaord_tile a.oe_kanban_color_2{
background: #CDC713;
color: #fff;
}
.openerp .oe_kanban_view .oe_tile_color_3,
.openerp .oe_dashbaord_tile a.oe_kanban_color_3{
background: #57158A;
color: #fff;
}
.openerp .oe_kanban_view .oe_tile_color_4,
.openerp .oe_dashbaord_tile a.oe_kanban_color_4{
background: #0E9B2D;
color: #fff;
}
.openerp .oe_kanban_view .oe_tile_color_5,
.openerp .oe_dashbaord_tile a.oe_kanban_color_5{
background: #7F0C00;
color: #fff;
}
.openerp .oe_kanban_view .oe_tile_color_6,
.openerp .oe_dashbaord_tile a.oe_kanban_color_6{
background: #7F7B00;
color: #fff;
}
.openerp .oe_kanban_view .oe_tile_color_7,
.openerp .oe_dashbaord_tile a.oe_kanban_color_7{
background: #320455;
color: #fff;
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_count_without_computed_value{
font-size: 52px;
font-weight: bold;
position: absolute;
left: 5px;
bottom: 5px;
}
.openerp .oe_kanban_view .oe_tile_color_8,
.openerp .oe_dashbaord_tile a.oe_kanban_color_8{
background: #CD6E13;
color: #fff;
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_count_with_computed_value{
font-size: 38px;
font-weight: bold;
position: absolute;
left: 5px;
bottom: 30px;
}
.openerp .oe_kanban_view .oe_tile_color_9,
.openerp .oe_dashbaord_tile a.oe_kanban_color_9{
background: #0E6C7E;
color: #fff;
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_computed_value{
font-size: 18px;
font-weight: bold;
position: absolute;
right: 10px;
bottom: 5px;
font-style: italic;
}
.openerp .oe_kanban_view .oe_tile_color_0,
.openerp .oe_dashbaord_tile a.oe_kanban_color_0{
background: #006015;
color: #fff;
}

BIN
web_dashboard_tile/static/src/img/avg.png

After

Width: 24  |  Height: 24  |  Size: 340 B

BIN
web_dashboard_tile/static/src/img/max.png

After

Width: 24  |  Height: 24  |  Size: 264 B

BIN
web_dashboard_tile/static/src/img/median.png

After

Width: 24  |  Height: 24  |  Size: 287 B

BIN
web_dashboard_tile/static/src/img/min.png

After

Width: 24  |  Height: 24  |  Size: 283 B

BIN
web_dashboard_tile/static/src/img/screenshot_action_click.png

After

Width: 754  |  Height: 168  |  Size: 24 KiB

BIN
web_dashboard_tile/static/src/img/screenshot_dashboard.png

After

Width: 796  |  Height: 162  |  Size: 45 KiB

BIN
web_dashboard_tile/static/src/img/sum.png

After

Width: 24  |  Height: 24  |  Size: 305 B

145
web_dashboard_tile/tile.py

@ -4,7 +4,9 @@
# OpenERP, Open Source Management Solution
# Copyright (C) 2010-2013 OpenERP s.a. (<http://openerp.com>).
# Copyright (C) 2014 initOS GmbH & Co. KG (<http://www.initos.com>).
# Copyright (C) 2015-Today GRAP
# Author Markus Schneider <markus.schneider at initos.com>
# @author Sylvain LE GAL (https://twitter.com/legalsylvain)
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
@ -22,35 +24,150 @@
##############################################################################
from openerp.osv import orm, fields
import random
from openerp.tools.translate import _
class tile(orm.Model):
_name = 'tile.tile'
_order = 'sequence, name'
def _get_tile_count(self, cr, uid, ids, field_name, field_value,
arg, context=None):
result = {}
records = self.browse(cr, uid, ids)
def median(self, aList):
# https://docs.python.org/3/library/statistics.html#statistics.median
# TODO : refactor, using statistics.median when Odoo will be available
# in Python 3.4
even = (0 if len(aList) % 2 else 1) + 1
half = (len(aList) - 1) / 2
return sum(sorted(aList)[half:half + even]) / float(even)
def _get_tile_info(self, cr, uid, ids, fields, args, context=None):
ima_obj = self.pool['ir.model.access']
res = {}
records = self.browse(cr, uid, ids, context=context)
for r in records:
model = self.pool.get(r.model_id.model)
result[r.id] = model.search_count(cr, uid, eval(r.domain), context)
return result
res[r.id] = {
'active': False,
'count': 0,
'computed_value': 0,
'helper': '',
}
if ima_obj.check(
cr, uid, r.model_id.model, 'read', False, context):
# Compute count item
model = self.pool.get(r.model_id.model)
count = model.search_count(
cr, uid, eval(r.domain), context=context)
res[r.id].update({
'active': True,
'count': count,
})
# Compute datas for field_id depending of field_function
if r.field_function and r.field_id and count != 0:
ids = model.search(
cr, uid, eval(r.domain), context=context)
vals = [x[r.field_id.name] for x in model.read(
cr, uid, ids, [r.field_id.name], context=context)]
desc = r.field_id.field_description
if r.field_function == 'min':
value = min(vals)
helper = _("Minimum value of '%s'") % desc
elif r.field_function == 'max':
value = max(vals)
helper = _("Maximum value of '%s'") % desc
elif r.field_function == 'sum':
value = sum(vals)
helper = _("Total value of '%s'") % desc
elif r.field_function == 'avg':
value = sum(vals) / len(vals)
helper = _("Average value of '%s'") % desc
elif r.field_function == 'median':
value = self.median(vals)
helper = _("Median value of '%s'") % desc
res[r.id].update({
'computed_value': value,
'helper': helper,
})
return res
def _search_active(self, cr, uid, obj, name, arg, context=None):
ima_obj = self.pool['ir.model.access']
ids = []
cr.execute("""
SELECT tt.id, im.model
FROM tile_tile tt
INNER JOIN ir_model im
ON tt.model_id = im.id""")
for result in cr.fetchall():
if (ima_obj.check(cr, uid, result[1], 'read', False) ==
arg[0][2]):
ids.append(result[0])
return [('id', 'in', ids)]
_columns = {
'name': fields.char('Tile Name'),
'model_id': fields.many2one('ir.model', 'Model'),
'model_id': fields.many2one('ir.model', 'Model', required=True),
'user_id': fields.many2one('res.users', 'User'),
'domain': fields.text('Domain'),
'action_id': fields.many2one('ir.actions.act_window', 'Action'),
'count': fields.function(_get_tile_count, type='int', String='Count',
readonly=True),
'color': fields.char('Kanban Color')
'count': fields.function(
_get_tile_info, type='int', string='Count',
multi='tile_info', readonly=True),
'computed_value': fields.function(
_get_tile_info, type='float', string='Computed Value',
multi='tile_info', readonly=True),
'helper': fields.function(
_get_tile_info, type='char', string='Helper Text',
multi='tile_info', readonly=True),
'field_function': fields.selection([
('min', 'Minimum'),
('max', 'Maximum'),
('sum', 'Sum'),
('avg', 'Average'),
('median', 'Median'),
], 'Function'),
'field_id': fields.many2one(
'ir.model.fields', 'Field',
domain="[('model_id', '=', model_id),"
" ('ttype', 'in', ['float', 'int'])]"),
'active': fields.function(
_get_tile_info, type='boolean', string='Active',
multi='tile_info', readonly=True, fnct_search=_search_active),
'color': fields.char('Background color'),
'font_color': fields.char('Font Color'),
'sequence': fields.integer(
'Sequence', required=True),
}
# Constraint Section
def _check_model_id_field_id(self, cr, uid, ids, context=None):
for t in self.browse(cr, uid, ids, context=context):
if t.field_id and t.field_id.model_id.id != t.model_id.id:
return False
return True
def _check_field_id_field_function(self, cr, uid, ids, context=None):
for t in self.browse(cr, uid, ids, context=context):
if t.field_id and not t.field_function or\
t.field_function and not t.field_id:
return False
return True
_constraints = [
(
_check_model_id_field_id,
"Error ! Please select a field of the selected model.",
['model_id', 'field_id']),
(
_check_field_id_field_function,
"Error ! Please set both fields: 'Field' and 'Function'.",
['field_id', 'field_function']),
]
_defaults = {
'domain': '[]',
'color': 0,
'color': '#0E6C7E',
'font_color': '#FFFFFF',
'sequence': 0,
}
def open_link(self, cr, uid, ids, context=None):
@ -89,6 +206,4 @@ class tile(orm.Model):
[('model', '=',
vals['model_id'])])
vals['model_id'] = model_ids[0]
if 'color' not in vals:
vals['color'] = random.randint(1, 10)
return self.create(cr, uid, vals, context)

45
web_dashboard_tile/tile.xml → web_dashboard_tile/view/tile.xml

@ -6,11 +6,13 @@
<field name="model">tile.tile</field>
<field name="arch" type="xml">
<tree string="Dashboard tiles">
<field name="sequence" widget="handle"/>
<field name="name"/>
<field name="domain"/>
<field name="model_id"/>
<field name="action_id"/>
<field name="count"/>
<field name="field_function"/>
<field name="field_id"/>
<field name="user_id"/>
</tree>
</field>
</record>
@ -26,6 +28,10 @@
<field name="model_id"/>
<field name="user_id"/>
<field name="action_id"/>
<field name="field_function"/>
<field name="field_id"/>
<field name="color" widget="color"/>
<field name="font_color" widget="color"/>
</group>
</form>
</field>
@ -43,30 +49,35 @@
<field name="action_id"/>
<field name="count"/>
<field name="color"/>
<field name="font_color"/>
<field name="field_id" />
<field name="field_function" />
<field name="helper" />
<templates>
<t t-name="kanban-box">
<div t-attf-class="oe_tile_color_#{kanban_getcolor(record.color.raw_value)} oe_dashbaord_tile oe_kanban_global_click" >
<!-- FIXME: icon is hidden and edit not working if you have no form view
need more JS in the future -->
<div class="oe_dropdown_toggle oe_dropdown_kanban">
<span class="oe_e">í</span>
<ul class="oe_dropdown_menu">
<t t-if="widget.view.is_action_enabled('edit')"><li><a type="edit">Edit...</a></li></t>
<t t-if="widget.view.is_action_enabled('delete')"><li><a type="delete">Delete</a></li></t>
<li><ul class="oe_kanban_colorpicker" data-field="color"/></li>
</ul>
</div>
<div t-attf-class="oe_dashbaord_tile oe_kanban_global_click" t-attf-style="background-color:#{record.color.raw_value}" >
<div class="oe_kanban_content">
<a type="object" name="open_link" args="[]">
<a type="object" name="open_link" args="[]" t-attf-style="color:#{record.font_color.raw_value};">
<div class="tile_label">
<b><field name="name"/></b>
</div>
<div style="padding-left: 0.5em; height: 115px;">
</div>
<div class="tile_count">
<span><field name="count"/></span>
</div>
<t t-if="record.field_id.raw_value != '' and record.field_function.raw_value != '' and record.count.raw_value !=0">
<div class="tile_count_with_computed_value">
<span><field name="count"/></span>
</div>
<div class="tile_computed_value" t-att-title="record.helper.raw_value">
<img t-att-src="_s + '/web_dashboard_tile/static/src/img/' + record.field_function.raw_value + '.png'"/>
<span><field name="computed_value"/></span>
</div>
</t>
<t t-if="!(record.field_id.raw_value != '' and record.field_function.raw_value != '' and record.count.raw_value !=0)">
<div class="tile_count_without_computed_value">
<span><field name="count"/></span>
</div>
</t>
</a>
</div>
<div class="oe_clear"></div>
Loading…
Cancel
Save