Browse Source

Merge pull request #180 from grap/8.0_PORT_web_dashboard_tile

[PORT][8.0] port web_dashboard_tile
pull/196/head
Pedro M. Baeza 9 years ago
parent
commit
bef3484fb8
  1. 66
      web_dashboard_tile/README.rst
  2. 26
      web_dashboard_tile/__init__.py
  3. 49
      web_dashboard_tile/__openerp__.py
  4. 25
      web_dashboard_tile/demo/res_groups.yml
  5. 40
      web_dashboard_tile/demo/tile_tile.yml
  6. 220
      web_dashboard_tile/i18n/fr.po
  7. 232
      web_dashboard_tile/i18n/web_dashboard_tile.pot
  8. 23
      web_dashboard_tile/models/__init__.py
  9. 179
      web_dashboard_tile/models/tile_tile.py
  10. 2
      web_dashboard_tile/security/ir.model.access.csv
  11. 13
      web_dashboard_tile/security/rules.xml
  12. BIN
      web_dashboard_tile/static/description/icon.png
  13. 44
      web_dashboard_tile/static/src/css/tile.css
  14. BIN
      web_dashboard_tile/static/src/img/avg.png
  15. BIN
      web_dashboard_tile/static/src/img/max.png
  16. BIN
      web_dashboard_tile/static/src/img/median.png
  17. BIN
      web_dashboard_tile/static/src/img/min.png
  18. BIN
      web_dashboard_tile/static/src/img/screenshot_action_click.png
  19. BIN
      web_dashboard_tile/static/src/img/screenshot_dashboard.png
  20. BIN
      web_dashboard_tile/static/src/img/sum.png
  21. 100
      web_dashboard_tile/static/src/js/custom_js.js
  22. 12
      web_dashboard_tile/static/src/xml/custom_xml.xml
  23. 33
      web_dashboard_tile/views/templates.xml
  24. 126
      web_dashboard_tile/views/tile.xml

66
web_dashboard_tile/README.rst

@ -0,0 +1,66 @@
Add Tiles to Dashboard
======================
module to give you a dashboard where you can configure tile from any view
and add them as short cut.
* 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;
Usage
=====
* 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);
Bug Tracker
===========
Bugs are tracked on `GitHub Issues <https://github.com/OCA/web/issues>`_.
In case of trouble, please check there if your issue has already been reported.
If you spotted it first, help us smashing it by providing a detailed and welcomed feedback
`here <https://github.com/OCA/web/issues/new?body=module:%20web_dashboard_tile%0Aversion:%208.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
Credits
=======
Contributors
------------
* Markus Schneider <markus.schneider at initos.com>
* Sylvain Le Gal (https://twitter.com/legalsylvain)
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.

26
web_dashboard_tile/__init__.py

@ -0,0 +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 models

49
web_dashboard_tile/__openerp__.py

@ -0,0 +1,49 @@
# -*- 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>).
# Author Markus Schneider <markus.schneider at initos.com>
#
# 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": "Dashboard Tile",
"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": "web",
'license': 'AGPL-3',
'data': [
'views/tile.xml',
'views/templates.xml',
'security/ir.model.access.csv',
'security/rules.xml',
],
'demo': [
'demo/res_groups.yml',
'demo/tile_tile.yml',
],
'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

40
web_dashboard_tile/demo/tile_tile.yml

@ -0,0 +1,40 @@
# -*- 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
- !record {model: tile.tile, id: all_currency_with_rate}:
name: Currencies (Max Rate)
model_id: base.model_res_currency
domain: []
field_function: max
field_id: base.field_res_currency_rate

220
web_dashboard_tile/i18n/fr.po

@ -0,0 +1,220 @@
# Translation of OpenERP Server.
# This file contains the translation of the following modules:
# * web_dashboard_tile
#
msgid ""
msgstr ""
"Project-Id-Version: OpenERP Server 7.0\n"
"Report-Msgid-Bugs-To: \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"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: web_dashboard_tile
#: field:tile.tile,action_id:0
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
#, python-format
msgid "Create"
msgstr "Créer"
#. module: web_dashboard_tile
#: model:ir.actions.act_window,name:web_dashboard_tile.action_kanban_dashboard_tile
#: model:ir.actions.act_window,name:web_dashboard_tile.action_tree_dashboard_tile
#: model:ir.ui.menu,name:web_dashboard_tile.mail_dashboard
msgid "Dashboard"
msgstr "Tableau de bord"
#. module: web_dashboard_tile
#: model:ir.ui.menu,name:web_dashboard_tile.menue_dashboard_tile
msgid "Dashboard Tile"
msgstr "Indicateur de tableau de bord"
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "Dashboard tiles"
msgstr "Indicateurs de tableau de bord"
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "Delete"
msgstr "Supprimer"
#. module: web_dashboard_tile
#: field:tile.tile,domain:0
msgid "Domain"
msgstr "Domaine"
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "Edit..."
msgstr "Editer..."
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:61
#, python-format
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
#, python-format
msgid "Filter name is required."
msgstr "Le nom du filtre est requis."
#. module: web_dashboard_tile
#: 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
#, python-format
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"
msgstr "Nom de l'indicateur"
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:100
#, python-format
msgid "Tile is created"
msgstr "L'indicateur a été créé"
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/xml/custom_xml.xml:6
#, python-format
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"

232
web_dashboard_tile/i18n/web_dashboard_tile.pot

@ -0,0 +1,232 @@
# Translation of OpenERP Server.
# This file contains the translation of the following modules:
# * web_dashboard_tile
#
msgid ""
msgstr ""
"Project-Id-Version: OpenERP Server 7.0\n"
"Report-Msgid-Bugs-To: \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"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: web_dashboard_tile
#: field:tile.tile,action_id:0
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
#, python-format
msgid "Create"
msgstr ""
#. module: web_dashboard_tile
#: model:ir.actions.act_window,name:web_dashboard_tile.action_kanban_dashboard_tile
#: model:ir.actions.act_window,name:web_dashboard_tile.action_tree_dashboard_tile
#: model:ir.ui.menu,name:web_dashboard_tile.mail_dashboard
msgid "Dashboard"
msgstr ""
#. module: web_dashboard_tile
#: model:ir.ui.menu,name:web_dashboard_tile.menue_dashboard_tile
msgid "Dashboard Tile"
msgstr ""
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "Dashboard tiles"
msgstr ""
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "Delete"
msgstr ""
#. module: web_dashboard_tile
#: field:tile.tile,domain:0
msgid "Domain"
msgstr ""
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "Edit..."
msgstr ""
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:61
#, python-format
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
#, python-format
msgid "Filter name is required."
msgstr ""
#. module: web_dashboard_tile
#: 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
#: field:tile.tile,model_id:0
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
#, python-format
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"
msgstr ""
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/js/custom_js.js:100
#, python-format
msgid "Tile is created"
msgstr ""
#. module: web_dashboard_tile
#. openerp-web
#: code:addons/web_dashboard_tile/static/src/xml/custom_xml.xml:6
#, python-format
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"
msgstr ""
#. module: web_dashboard_tile
#: code:_description:0
#: model:ir.model,name:web_dashboard_tile.model_tile_tile
#, python-format
msgid "tile.tile"
msgstr ""
#. module: web_dashboard_tile
#: view:tile.tile:0
msgid "í"
msgstr ""

23
web_dashboard_tile/models/__init__.py

@ -0,0 +1,23 @@
# -*- 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/>.
#
##############################################################################
from . import tile_tile

179
web_dashboard_tile/models/tile_tile.py

@ -0,0 +1,179 @@
# -*- 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 openerp import api, fields
from openerp.models import Model
from openerp.exceptions import except_orm
from openerp.tools.translate import _
class TileTile(Model):
_name = 'tile.tile'
_order = 'sequence, name'
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):
ima_obj = self.env['ir.model.access']
res = {}
for r in self:
r.active = False
r.count = 0
r.computed_value = 0
r.helper = ''
if ima_obj.check(r.model_id.model, 'read', False):
# Compute count item
model = self.env[r.model_id.model]
r.count = model.search_count(eval(r.domain))
r.active = True
# Compute datas for field_id depending of field_function
if r.field_function and r.field_id and r.count != 0:
records = model.search(eval(r.domain))
vals = [x[r.field_id.name] for x in records]
desc = r.field_id.field_description
if r.field_function == 'min':
r.computed_value = min(vals)
r.helper = _("Minimum value of '%s'") % desc
elif r.field_function == 'max':
r.computed_value = max(vals)
r.helper = _("Maximum value of '%s'") % desc
elif r.field_function == 'sum':
r.computed_value = sum(vals)
r.helper = _("Total value of '%s'") % desc
elif r.field_function == 'avg':
r.computed_value = sum(vals) / len(vals)
r.helper = _("Average value of '%s'") % desc
elif r.field_function == 'median':
r.computed_value = self.median(vals)
r.helper = _("Median value of '%s'") % desc
return res
def _search_active(self, operator, value):
cr = self.env.cr
if operator != '=':
raise except_orm(
'Unimplemented Feature',
'Search on Active field disabled.')
ima_obj = self.env['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(result[1], 'read', False) == value):
ids.append(result[0])
return [('id', 'in', ids)]
# Column Section
name = fields.Char(required=True)
model_id = fields.Many2one(
comodel_name='ir.model', string='Model', required=True)
user_id = fields.Many2one(
comodel_name='res.users', string='User')
domain = fields.Text(default='[]')
action_id = fields.Many2one(
comodel_name='ir.actions.act_window', string='Action')
count = fields.Integer(compute='_get_tile_info')
computed_value = fields.Float(compute='_get_tile_info')
helper = fields.Char(compute='_get_tile_info')
field_function = fields.Selection(selection=[
('min', 'Minimum'),
('max', 'Maximum'),
('sum', 'Sum'),
('avg', 'Average'),
('median', 'Median'),
], string='Function')
field_id = fields.Many2one(
comodel_name='ir.model.fields', string='Field',
domain="[('model_id', '=', model_id),"
" ('ttype', 'in', ['float', 'int'])]")
active = fields.Boolean(
compute='_get_tile_info', readonly=True, search='_search_active')
background_color = fields.Char(default='#0E6C7E', oldname='color')
font_color = fields.Char(default='#FFFFFF')
sequence = fields.Integer(default=0, 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']),
]
# View / action Section
@api.multi
def open_link(self):
res = {
'name': self.name,
'view_type': 'form',
'view_mode': 'tree',
'view_id': [False],
'res_model': self.model_id.model,
'type': 'ir.actions.act_window',
'context': self.env.context,
'nodestroy': True,
'target': 'current',
'domain': self.domain,
}
if self.action_id:
res.update(self.action_id.read(
['view_type', 'view_mode', 'view_id', 'type'])[0])
# FIXME: restore original Domain + Filter would be better
return res
@api.model
def add(self, vals):
if 'model_id' in vals and not vals['model_id'].isdigit():
# need to replace model_name with its id
vals['model_id'] = self.env['ir.model'].search(
[('model', '=', vals['model_id'])]).id
self.create(vals)

2
web_dashboard_tile/security/ir.model.access.csv

@ -0,0 +1,2 @@
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
tile_user,tile_user,model_tile_tile,base.group_user,1,1,1,1

13
web_dashboard_tile/security/rules.xml

@ -0,0 +1,13 @@
<?xml version="1.0" encoding="utf-8"?>
<openerp>
<data noupdate="1">
<record id="model_tile_rule" model="ir.rule">
<field name="name">tile.owner</field>
<field name="model_id" ref="model_tile_tile" />
<field name="groups" eval="[(4, ref('base.group_user'))]"/>
<field name="domain_force">[('user_id','in',[False,user.id])]</field>
</record>
</data>
</openerp>

BIN
web_dashboard_tile/static/description/icon.png

After

Width: 142  |  Height: 142  |  Size: 1.0 KiB

44
web_dashboard_tile/static/src/css/tile.css

@ -0,0 +1,44 @@
.openerp .oe_kanban_view .oe_dashbaord_tile{
width: 150px;
height: 150px;
border: 0;
border-radius: 0;
}
.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: 5px;
font-size: 15px;
}
.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_dashbaord_tile .tile_count_with_computed_value{
font-size: 38px;
font-weight: bold;
position: absolute;
left: 5px;
bottom: 30px;
}
.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;
}

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

100
web_dashboard_tile/static/src/js/custom_js.js

@ -0,0 +1,100 @@
// @@@ web_dashboard_tile custom JS @@@
//#############################################################################
//
// Copyright (C) 2010-2013 OpenERP s.a. (<http://www.openerp.com>)
// Copyright (C) 2014 initOS GmbH & Co. KG (<http://initos.com>)
//
// 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_dashboard_tile = function (instance)
{
var QWeb = instance.web.qweb,
_t = instance.web._t,
_lt = instance.web._lt;
_.mixin({
sum: function (obj) { return _.reduce(obj, function (a, b) { return a + b; }, 0); }
});
var module = instance.board.AddToDashboard;
module.include({
start: function () {
this._super();
var self = this;
this.$('#add_dashboard_tile').on('click', this, function (){
self.save_tile();
})
},
render_data: function(dashboard_choices){
var selection = instance.web.qweb.render(
"SearchView.addtodashboard.selection", {
selections: dashboard_choices});
this.$("form input").before(selection)
},
save_tile: function () {
var self = this;
var view_parent = this.getParent().getParent();
var $name = this.$('#dashboard_tile_new_name');
this.tile = new instance.web.Model('tile.tile');
var private_filter = !this.$('#oe_searchview_custom_public').prop('checked');
if (_.isEmpty($name.val())){
this.do_warn(_t("Error"), _t("Filter name is required."));
return false;
}
var search = this.view.build_search_data();
var context = new instance.web.CompoundContext(view_parent.dataset.get_context() || []);
var domain = new instance.web.CompoundDomain(view_parent.dataset.get_domain() || []);
_.each(search.contexts, context.add, context);
_.each(search.domains, domain.add, domain);
var c = instance.web.pyeval.eval('context', context);
for(var k in c) {
if (c.hasOwnProperty(k) && /^search_default_/.test(k)) {
delete c[k];
}
}
// TODO: replace this 6.1 workaround by attribute on <action/>
c.dashboard_merge_domains_contexts = false;
var d = instance.web.pyeval.eval('domain', domain);
context.add({
group_by: instance.web.pyeval.eval('groupbys', search.groupbys || [])
});
// Don't save user_context keys in the custom filter, otherwise end
// up with e.g. wrong uid or lang stored *and used in subsequent
// reqs*
var ctx = context;
_(_.keys(instance.session.user_context)).each(function (key) {
delete ctx[key];
});
var filter = {
name: $name.val(),
user_id: private_filter ? instance.session.uid : false,
model_id: self.view.model,
//context: context,
domain: d,
action_id: view_parent.action.id,
};
// FIXME: current context?
return self.tile.call('add', [filter]).done(function (id) {
self.do_warn(_t("Success"), _t("Tile is created"));
});
}
});
}

12
web_dashboard_tile/static/src/xml/custom_xml.xml

@ -0,0 +1,12 @@
<?xml version="1.0" encoding="UTF-8"?>
<templates id="template" xml:space="preserve">
<t t-extend="SearchView.addtodashboard">
<t t-jquery="form" t-operation="after">
<div>
<label for="dashboard_tile_new_name">Tile:</label>
<input id="dashboard_tile_new_name" />
<button id="add_dashboard_tile">Create</button>
</div>
</t>
</t>
</templates>

33
web_dashboard_tile/views/templates.xml

@ -0,0 +1,33 @@
<?xml version="1.0" encoding="UTF-8"?>
<!--***************************************************************************
Web - Custom Element Number in ListView module for Odoo
Copyright (C) 2015-Today Akretion (http://www.akretion.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/>.
****************************************************************************-->
<openerp>
<data>
<template id="assets_backend" name="web_dashboard_tile assets" inherit_id="web.assets_backend">
<xpath expr="." position="inside">
<link rel="stylesheet"
href="/web_dashboard_tile/static/src/css/tile.css"/>
<script type="text/javascript"
src="/web_dashboard_tile/static/src/js/custom_js.js">
</script>
</xpath>
</template>
</data>
</openerp>

126
web_dashboard_tile/views/tile.xml

@ -0,0 +1,126 @@
<?xml version="1.0" encoding="utf-8"?>
<openerp>
<data>
<record model="ir.ui.view" id="dashboard_tile_tile_tree_view">
<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="field_function"/>
<field name="field_id"/>
<field name="user_id"/>
<field name="background_color" widget="color"/>
</tree>
</field>
</record>
<record model="ir.ui.view" id="dashboard_tile_tile_form_view">
<field name="model">tile.tile</field>
<field name="arch" type="xml">
<form string="Dashboard tiles">
<sheet>
<h1>
<field name="name"/>
</h1>
<group col="4">
<separator string="Display" colspan="4"/>
<field name="background_color" widget="color"/>
<field name="font_color" widget="color"/>
<field name="user_id"/>
<separator string="Technical Informations" colspan="4"/>
<field name="model_id"/>
<field name="action_id"/>
<field name="domain" colspan="4"/>
<separator string="Optional Field Informations" colspan="4"/>
<field name="field_function"/>
<field name="field_id"/>
<field name="helper" colspan="4"/>
</group>
</sheet>
</form>
</field>
</record>
<!-- CRM Lead Kanban View -->
<record model="ir.ui.view" id="dashboard_tile_tile_kanban_view">
<field name="model">tile.tile</field>
<field name="arch" type="xml">
<kanban edit="false" create="false">
<field name="name"/>
<field name="domain"/>
<field name="model_id"/>
<field name="action_id"/>
<field name="count"/>
<field name="background_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_dashbaord_tile oe_kanban_global_click" t-attf-style="background-color:#{record.background_color.raw_value}" >
<div class="oe_kanban_content">
<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>
<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>
</div>
</t>
</templates>
</kanban>
</field>
</record>
<record model="ir.actions.act_window" id="action_tree_dashboard_tile">
<field name="name">Dashboard</field>
<field name="res_model">tile.tile</field>
<field name="view_type">form</field>
<field name="view_mode">tree,kanban,form</field>
<field name="view_id" ref="dashboard_tile_tile_tree_view"/>
</record>
<menuitem id="menue_dashboard_tile"
name="Dashboard Tile" parent="base.next_id_2"
action="action_tree_dashboard_tile" sequence="50"/>
<record model="ir.actions.act_window" id="action_kanban_dashboard_tile">
<field name="name">Dashboard</field>
<field name="res_model">tile.tile</field>
<field name="view_type">form</field>
<field name="view_mode">kanban</field>
<field name="domain">['|',('user_id','=',False),('user_id','=',uid)]</field>
<field name="view_id" ref="dashboard_tile_tile_tree_view"/>
</record>
<record id="mail_dashboard" model="ir.ui.menu">
<field name="name">Dashboard</field>
<field name="sequence" eval="9"/>
<field name="action" ref="action_kanban_dashboard_tile"/>
<field name="parent_id" ref="mail.mail_feeds"/>
</record>
</data>
</openerp>
Loading…
Cancel
Save