mirror of https://github.com/OCA/web.git
FIX
parent
51630ecbdf
commit
1c2af267f4
|
@ -1,19 +1,6 @@
|
||||||
Add Tiles to Dashboard
|
Dashboard Tiles
|
||||||
======================
|
===============
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
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;
|
|
||||||
=======
|
|
||||||
Adds a dashboard where you can configure tiles from any view and add them as short cut.
|
Adds a dashboard where you can configure tiles from any view and add them as short cut.
|
||||||
|
|
||||||
By default, the tile displays items count of a given model restricted to a given domain.
|
By default, the tile displays items count of a given model restricted to a given domain.
|
||||||
|
@ -30,28 +17,18 @@ Tile can be:
|
||||||
- Restricted to some groups.
|
- Restricted to some groups.
|
||||||
|
|
||||||
*Note: The tile will be hidden if the current user doesn't have access to the given model.*
|
*Note: The tile will be hidden if the current user doesn't have access to the given model.*
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
|
|
||||||
Usage
|
Usage
|
||||||
=====
|
=====
|
||||||
|
|
||||||
* Dashboad sample, displaying Sale Orders to invoice:
|
* Dashboad sample, displaying Sale Orders to invoice:
|
||||||
.. image:: /web_dashboard_tile/static/src/img/screenshot_dashboard.png
|
|
||||||
|
.. image:: ./static/src/img/screenshot_dashboard.png
|
||||||
|
|
||||||
* Tree view displayed when user click on the tile:
|
* Tree view displayed when user click on the tile:
|
||||||
.. image:: /web_dashboard_tile/static/src/img/screenshot_action_click.png
|
|
||||||
|
|
||||||
Known issues / Roadmap
|
.. image:: ./static/src/img/screenshot_action_click.png
|
||||||
======================
|
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
* 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;
|
|
||||||
* Support context_today;
|
|
||||||
* Add icons;
|
|
||||||
* Support client side action (like inbox);
|
|
||||||
=======
|
|
||||||
Known issues
|
Known issues
|
||||||
============
|
============
|
||||||
* Can not edit tile from dashboard (color, sequence, function, ...).
|
* Can not edit tile from dashboard (color, sequence, function, ...).
|
||||||
|
@ -66,7 +43,6 @@ Roadmap
|
||||||
* Restore original Domain + Filter when an action is set.
|
* Restore original Domain + Filter when an action is set.
|
||||||
* Posibility to hide the tile based on a field expression.
|
* Posibility to hide the tile based on a field expression.
|
||||||
* Posibility to set the background color based on a field expression.
|
* Posibility to set the background color based on a field expression.
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
|
|
||||||
Bug Tracker
|
Bug Tracker
|
||||||
===========
|
===========
|
||||||
|
@ -74,10 +50,7 @@ Bug Tracker
|
||||||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/web/issues>`_.
|
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.
|
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
|
If you spotted it first, help us smashing it by providing a detailed and welcomed feedback
|
||||||
`here <https://github.com/OCA/
|
`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**>`_.
|
||||||
web/issues/new?body=module:%20
|
|
||||||
web_dashboard_tile%0Aversion:%20
|
|
||||||
8.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
|
|
||||||
|
|
||||||
|
|
||||||
Credits
|
Credits
|
||||||
|
@ -88,6 +61,7 @@ Contributors
|
||||||
|
|
||||||
* Markus Schneider <markus.schneider at initos.com>
|
* Markus Schneider <markus.schneider at initos.com>
|
||||||
* Sylvain Le Gal (https://twitter.com/legalsylvain)
|
* Sylvain Le Gal (https://twitter.com/legalsylvain)
|
||||||
|
* Iván Todorovich <ivan.todorovich@gmail.com>
|
||||||
|
|
||||||
Maintainer
|
Maintainer
|
||||||
----------
|
----------
|
||||||
|
@ -100,4 +74,4 @@ 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.
|
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.
|
To contribute to this module, please visit http://odoo-community.org.
|
|
@ -1,26 +1,7 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
##############################################################################
|
# © 2010-2013 OpenERP s.a. (<http://openerp.com>).
|
||||||
#
|
# © 2014 initOS GmbH & Co. KG (<http://www.initos.com>).
|
||||||
# OpenERP, Open Source Management Solution
|
# © 2015-Today GRAP
|
||||||
# Copyright (C) 2010-2013 OpenERP s.a. (<http://openerp.com>).
|
# License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html
|
||||||
# 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
|
from . import models
|
||||||
|
|
|
@ -20,6 +20,10 @@
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
#
|
#
|
||||||
##############################################################################
|
##############################################################################
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# © 2010-2013 OpenERP s.a. (<http://openerp.com>).
|
||||||
|
# © 2014 initOS GmbH & Co. KG (<http://www.initos.com>).
|
||||||
|
# License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html
|
||||||
{
|
{
|
||||||
"name": "Dashboard Tile",
|
"name": "Dashboard Tile",
|
||||||
"summary": "Add Tiles to Dashboard",
|
"summary": "Add Tiles to Dashboard",
|
||||||
|
@ -30,9 +34,16 @@
|
||||||
'mail',
|
'mail',
|
||||||
'web_widget_color',
|
'web_widget_color',
|
||||||
],
|
],
|
||||||
'author': "initOS GmbH & Co. KG,GRAP,Odoo Community Association (OCA)",
|
'author': 'initOS GmbH & Co. KG, '
|
||||||
|
'GRAP, '
|
||||||
|
'Odoo Community Association (OCA)',
|
||||||
"category": "web",
|
"category": "web",
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
|
'contributors': [
|
||||||
|
'initOS GmbH & Co. KG',
|
||||||
|
'GRAP',
|
||||||
|
'Iván Todorovich <ivan.todorovich@gmail.com>'
|
||||||
|
],
|
||||||
'data': [
|
'data': [
|
||||||
'views/tile.xml',
|
'views/tile.xml',
|
||||||
'views/templates.xml',
|
'views/templates.xml',
|
||||||
|
@ -46,5 +57,4 @@
|
||||||
'qweb': [
|
'qweb': [
|
||||||
'static/src/xml/custom_xml.xml',
|
'static/src/xml/custom_xml.xml',
|
||||||
],
|
],
|
||||||
'installable': True,
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,23 +1,7 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
##############################################################################
|
# © 2010-2013 OpenERP s.a. (<http://openerp.com>).
|
||||||
#
|
# © 2014 initOS GmbH & Co. KG (<http://www.initos.com>).
|
||||||
# OpenERP, Open Source Management Solution
|
# © 2015-Today GRAP
|
||||||
# Copyright (C) 2015-Today GRAP
|
# License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html
|
||||||
# @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
|
from . import tile_tile
|
||||||
|
|
|
@ -1,33 +1,4 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
<<<<<<< HEAD
|
|
||||||
##############################################################################
|
|
||||||
#
|
|
||||||
# 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
|
|
||||||
=======
|
|
||||||
# © 2010-2013 OpenERP s.a. (<http://openerp.com>).
|
# © 2010-2013 OpenERP s.a. (<http://openerp.com>).
|
||||||
# © 2014 initOS GmbH & Co. KG (<http://www.initos.com>).
|
# © 2014 initOS GmbH & Co. KG (<http://www.initos.com>).
|
||||||
# © 2015-Today GRAP
|
# © 2015-Today GRAP
|
||||||
|
@ -40,13 +11,10 @@ from collections import OrderedDict
|
||||||
|
|
||||||
from openerp import api, fields, models
|
from openerp import api, fields, models
|
||||||
from openerp.tools.safe_eval import safe_eval as eval
|
from openerp.tools.safe_eval import safe_eval as eval
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
from openerp.tools.translate import _
|
from openerp.tools.translate import _
|
||||||
|
from openerp.exceptions import ValidationError, except_orm
|
||||||
|
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
class TileTile(Model):
|
|
||||||
=======
|
|
||||||
def median(vals):
|
def median(vals):
|
||||||
# https://docs.python.org/3/library/statistics.html#statistics.median
|
# https://docs.python.org/3/library/statistics.html#statistics.median
|
||||||
# TODO : refactor, using statistics.median when Odoo will be available
|
# TODO : refactor, using statistics.median when Odoo will be available
|
||||||
|
@ -75,7 +43,7 @@ FIELD_FUNCTIONS = OrderedDict([
|
||||||
'help': _("Total value of '%s'")}),
|
'help': _("Total value of '%s'")}),
|
||||||
('avg', {
|
('avg', {
|
||||||
'name': 'Average',
|
'name': 'Average',
|
||||||
'func': lambda vals: sum(vals)/len(vals),
|
'func': lambda vals: sum(vals) / len(vals),
|
||||||
'help': _("Minimum value of '%s'")}),
|
'help': _("Minimum value of '%s'")}),
|
||||||
('median', {
|
('median', {
|
||||||
'name': 'Median',
|
'name': 'Median',
|
||||||
|
@ -89,56 +57,10 @@ FIELD_FUNCTION_SELECTION = [
|
||||||
|
|
||||||
|
|
||||||
class TileTile(models.Model):
|
class TileTile(models.Model):
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
_name = 'tile.tile'
|
_name = 'tile.tile'
|
||||||
_description = 'Dashboard Tile'
|
_description = 'Dashboard Tile'
|
||||||
_order = 'sequence, name'
|
_order = 'sequence, name'
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
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 _get_eval_context(self):
|
def _get_eval_context(self):
|
||||||
def _context_today():
|
def _context_today():
|
||||||
return fields.Date.from_string(fields.Date.context_today(self))
|
return fields.Date.from_string(fields.Date.context_today(self))
|
||||||
|
@ -238,13 +160,13 @@ class TileTile(models.Model):
|
||||||
self.primary_function != 'count',
|
self.primary_function != 'count',
|
||||||
self.secondary_function and
|
self.secondary_function and
|
||||||
self.secondary_function != 'count'
|
self.secondary_function != 'count'
|
||||||
]):
|
]):
|
||||||
records = model.search(eval(domain, eval_context))
|
records = model.search(eval(domain, eval_context))
|
||||||
for f in ['primary_', 'secondary_']:
|
for f in ['primary_', 'secondary_']:
|
||||||
f_function = f+'function'
|
f_function = f + 'function'
|
||||||
f_field_id = f+'field_id'
|
f_field_id = f + 'field_id'
|
||||||
f_format = f+'format'
|
f_format = f + 'format'
|
||||||
f_value = f+'value'
|
f_value = f + 'value'
|
||||||
value = 0
|
value = 0
|
||||||
if self[f_function] == 'count':
|
if self[f_function] == 'count':
|
||||||
value = count
|
value = count
|
||||||
|
@ -268,9 +190,9 @@ class TileTile(models.Model):
|
||||||
'secondary_function', 'secondary_field_id')
|
'secondary_function', 'secondary_field_id')
|
||||||
def _compute_helper(self):
|
def _compute_helper(self):
|
||||||
for f in ['primary_', 'secondary_']:
|
for f in ['primary_', 'secondary_']:
|
||||||
f_function = f+'function'
|
f_function = f + 'function'
|
||||||
f_field_id = f+'field_id'
|
f_field_id = f + 'field_id'
|
||||||
f_helper = f+'helper'
|
f_helper = f + 'helper'
|
||||||
self[f_helper] = ''
|
self[f_helper] = ''
|
||||||
field_func = FIELD_FUNCTIONS.get(self[f_function], {})
|
field_func = FIELD_FUNCTIONS.get(self[f_function], {})
|
||||||
help = field_func.get('help', False)
|
help = field_func.get('help', False)
|
||||||
|
@ -285,15 +207,13 @@ class TileTile(models.Model):
|
||||||
def _compute_active(self):
|
def _compute_active(self):
|
||||||
ima = self.env['ir.model.access']
|
ima = self.env['ir.model.access']
|
||||||
self.active = ima.check(self.model_id.model, 'read', False)
|
self.active = ima.check(self.model_id.model, 'read', False)
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
|
|
||||||
def _search_active(self, operator, value):
|
def _search_active(self, operator, value):
|
||||||
cr = self.env.cr
|
cr = self.env.cr
|
||||||
if operator != '=':
|
if operator != '=':
|
||||||
raise except_orm(
|
raise except_orm(
|
||||||
'Unimplemented Feature',
|
_('Unimplemented Feature. Search on Active field disabled.'))
|
||||||
'Search on Active field disabled.')
|
ima = self.env['ir.model.access']
|
||||||
ima_obj = self.env['ir.model.access']
|
|
||||||
ids = []
|
ids = []
|
||||||
cr.execute("""
|
cr.execute("""
|
||||||
SELECT tt.id, im.model
|
SELECT tt.id, im.model
|
||||||
|
@ -301,40 +221,10 @@ class TileTile(models.Model):
|
||||||
INNER JOIN ir_model im
|
INNER JOIN ir_model im
|
||||||
ON tt.model_id = im.id""")
|
ON tt.model_id = im.id""")
|
||||||
for result in cr.fetchall():
|
for result in cr.fetchall():
|
||||||
if (ima_obj.check(result[1], 'read', False) == value):
|
if (ima.check(result[1], 'read', False) == value):
|
||||||
ids.append(result[0])
|
ids.append(result[0])
|
||||||
return [('id', 'in', ids)]
|
return [('id', 'in', ids)]
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
# 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)
|
|
||||||
=======
|
|
||||||
# Constraints and onchanges
|
# Constraints and onchanges
|
||||||
@api.one
|
@api.one
|
||||||
@api.constrains('model_id', 'primary_field_id', 'secondary_field_id')
|
@api.constrains('model_id', 'primary_field_id', 'secondary_field_id')
|
||||||
|
@ -344,9 +234,9 @@ class TileTile(models.Model):
|
||||||
self.primary_field_id.model_id.id != self.model_id.id,
|
self.primary_field_id.model_id.id != self.model_id.id,
|
||||||
self.secondary_field_id and
|
self.secondary_field_id and
|
||||||
self.secondary_field_id.model_id.id != self.model_id.id
|
self.secondary_field_id.model_id.id != self.model_id.id
|
||||||
]):
|
]):
|
||||||
raise ValidationError(
|
raise ValidationError(
|
||||||
_("Please select a field from the selected model."))
|
_("Please select a field from the selected model."))
|
||||||
|
|
||||||
@api.onchange('model_id')
|
@api.onchange('model_id')
|
||||||
def _onchange_model_id(self):
|
def _onchange_model_id(self):
|
||||||
|
@ -359,34 +249,8 @@ class TileTile(models.Model):
|
||||||
self.primary_field_id = False
|
self.primary_field_id = False
|
||||||
if self.secondary_function in [False, 'count']:
|
if self.secondary_function in [False, 'count']:
|
||||||
self.secondary_field_id = False
|
self.secondary_field_id = False
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
|
|
||||||
# Constraint Section
|
# Action methods
|
||||||
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
|
@api.multi
|
||||||
def open_link(self):
|
def open_link(self):
|
||||||
res = {
|
res = {
|
||||||
|
@ -403,8 +267,7 @@ class TileTile(models.Model):
|
||||||
}
|
}
|
||||||
if self.action_id:
|
if self.action_id:
|
||||||
res.update(self.action_id.read(
|
res.update(self.action_id.read(
|
||||||
['view_type', 'view_mode', 'view_id', 'type'])[0])
|
['view_type', 'view_mode', 'type'])[0])
|
||||||
# FIXME: restore original Domain + Filter would be better
|
|
||||||
return res
|
return res
|
||||||
|
|
||||||
@api.model
|
@api.model
|
||||||
|
|
|
@ -1,17 +1,10 @@
|
||||||
.openerp .oe_kanban_view .oe_dashbaord_tile{
|
.openerp .oe_kanban_view .oe_dashboard_tile {
|
||||||
width: 150px;
|
width: 150px;
|
||||||
height: 150px;
|
height: 150px;
|
||||||
border: 0;
|
border: 0;
|
||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
.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;
|
|
||||||
=======
|
|
||||||
/* Disable default kanban style */
|
/* Disable default kanban style */
|
||||||
.openerp .oe_kanban_view .oe_dashboard_tile .oe_kanban_content div:first-child {
|
.openerp .oe_kanban_view .oe_dashboard_tile .oe_kanban_content div:first-child {
|
||||||
margin-right: inherit!important;
|
margin-right: inherit!important;
|
||||||
|
@ -20,44 +13,27 @@
|
||||||
.openerp .oe_kanban_view .oe_dashboard_tile .tile_label,
|
.openerp .oe_kanban_view .oe_dashboard_tile .tile_label,
|
||||||
.openerp .oe_kanban_view .oe_dashboard_tile .tile_primary_value,
|
.openerp .oe_kanban_view .oe_dashboard_tile .tile_primary_value,
|
||||||
.openerp .oe_kanban_view .oe_dashboard_tile .tile_secondary_value {
|
.openerp .oe_kanban_view .oe_dashboard_tile .tile_secondary_value {
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_label{
|
|
||||||
=======
|
|
||||||
.openerp .oe_kanban_view .oe_dashboard_tile .tile_label {
|
.openerp .oe_kanban_view .oe_dashboard_tile .tile_label {
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
padding: 5px;
|
padding: 5px;
|
||||||
font-size: 15px;
|
font-size: 15px;
|
||||||
}
|
}
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_count_without_computed_value{
|
|
||||||
font-size: 52px;
|
|
||||||
font-weight: bold;
|
|
||||||
=======
|
|
||||||
.openerp .oe_kanban_view .oe_dashboard_tile .tile_primary_value{
|
.openerp .oe_kanban_view .oe_dashboard_tile .tile_primary_value{
|
||||||
font-size: 54px;
|
font-size: 54px;
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
left: 5px;
|
left: 5px;
|
||||||
right: 5px;
|
right: 5px;
|
||||||
bottom: 5px;
|
bottom: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
.openerp .oe_kanban_view .oe_dashbaord_tile .tile_count_with_computed_value{
|
|
||||||
font-size: 38px;
|
|
||||||
font-weight: bold;
|
|
||||||
=======
|
|
||||||
.openerp .oe_kanban_view .oe_dashboard_tile .tile_secondary_value{
|
.openerp .oe_kanban_view .oe_dashboard_tile .tile_secondary_value{
|
||||||
display: none;
|
display: none;
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
left: 5px;
|
left: 5px;
|
||||||
right: 5px;
|
right: 5px;
|
||||||
|
@ -69,16 +45,6 @@
|
||||||
bottom: 30px;
|
bottom: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
<<<<<<< HEAD
|
|
||||||
.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_dashboard_tile .with_secondary .tile_secondary_value{
|
.openerp .oe_kanban_view .oe_dashboard_tile .with_secondary .tile_secondary_value{
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
@ -90,5 +56,4 @@
|
||||||
|
|
||||||
.openerp .oe_searchview_drawer .oe_opened .oe_dashboard_tile_form {
|
.openerp .oe_searchview_drawer .oe_opened .oe_dashboard_tile_form {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
>>>>>>> 3ab676d... [IMP][8.0][web_dashboard_tile] Refactor (see changes in description) (#476)
|
|
|
@ -19,7 +19,7 @@
|
||||||
//
|
//
|
||||||
//#############################################################################
|
//#############################################################################
|
||||||
|
|
||||||
openerp.web_dashboard_tile = function (instance)
|
odoo.web_dashboard_tile = function (instance)
|
||||||
{
|
{
|
||||||
var QWeb = instance.web.qweb,
|
var QWeb = instance.web.qweb,
|
||||||
_t = instance.web._t,
|
_t = instance.web._t,
|
||||||
|
@ -35,13 +35,13 @@ _.mixin({
|
||||||
var self = this;
|
var self = this;
|
||||||
this.$('#add_dashboard_tile').on('click', this, function (){
|
this.$('#add_dashboard_tile').on('click', this, function (){
|
||||||
self.save_tile();
|
self.save_tile();
|
||||||
})
|
});
|
||||||
},
|
},
|
||||||
render_data: function(dashboard_choices){
|
render_data: function(dashboard_choices){
|
||||||
var selection = instance.web.qweb.render(
|
var selection = instance.web.qweb.render(
|
||||||
"SearchView.addtodashboard.selection", {
|
"SearchView.addtodashboard.selection", {
|
||||||
selections: dashboard_choices});
|
selections: dashboard_choices});
|
||||||
this.$("form input").before(selection)
|
this.$("form input").before(selection);
|
||||||
},
|
},
|
||||||
save_tile: function () {
|
save_tile: function () {
|
||||||
var self = this;
|
var self = this;
|
||||||
|
@ -97,4 +97,4 @@ _.mixin({
|
||||||
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
};
|
||||||
|
|
|
@ -2,11 +2,11 @@
|
||||||
<templates id="template" xml:space="preserve">
|
<templates id="template" xml:space="preserve">
|
||||||
<t t-extend="SearchView.addtodashboard">
|
<t t-extend="SearchView.addtodashboard">
|
||||||
<t t-jquery="form" t-operation="after">
|
<t t-jquery="form" t-operation="after">
|
||||||
<div>
|
<div class="oe_dashboard_tile_form">
|
||||||
<label for="dashboard_tile_new_name">Tile:</label>
|
<label for="dashboard_tile_new_name">Tile:</label>
|
||||||
<input id="dashboard_tile_new_name" />
|
<input id="dashboard_tile_new_name" />
|
||||||
<button id="add_dashboard_tile">Create</button>
|
<button id="add_dashboard_tile">Create</button>
|
||||||
</div>
|
</div>
|
||||||
</t>
|
</t>
|
||||||
</t>
|
</t>
|
||||||
</templates>
|
</templates>
|
|
@ -30,4 +30,4 @@
|
||||||
</xpath>
|
</xpath>
|
||||||
</template>
|
</template>
|
||||||
</data>
|
</data>
|
||||||
</openerp>
|
</openerp>
|
|
@ -103,7 +103,7 @@
|
||||||
|
|
||||||
<templates>
|
<templates>
|
||||||
<t t-name="kanban-box">
|
<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 t-attf-class="oe_dashboard_tile oe_kanban_global_click" t-attf-style="background-color:#{record.background_color.raw_value}" >
|
||||||
<div class="oe_kanban_content">
|
<div class="oe_kanban_content">
|
||||||
<a type="object" name="open_link" args="[]" t-attf-style="color:#{record.font_color.raw_value};">
|
<a type="object" name="open_link" args="[]" t-attf-style="color:#{record.font_color.raw_value};">
|
||||||
<div style="height:100%;" t-att-class="record.secondary_function.raw_value and 'with_secondary' or 'simple'">
|
<div style="height:100%;" t-att-class="record.secondary_function.raw_value and 'with_secondary' or 'simple'">
|
||||||
|
@ -152,9 +152,9 @@
|
||||||
|
|
||||||
<record id="mail_dashboard" model="ir.ui.menu">
|
<record id="mail_dashboard" model="ir.ui.menu">
|
||||||
<field name="name">Dashboard</field>
|
<field name="name">Dashboard</field>
|
||||||
<field name="sequence" eval="9"/>
|
<field name="sequence" eval="0"/>
|
||||||
<field name="action" ref="action_kanban_dashboard_tile"/>
|
<field name="action" ref="action_kanban_dashboard_tile"/>
|
||||||
<field name="parent_id" ref="mail.mail_feeds"/>
|
<field name="parent_id" ref=""/>
|
||||||
</record>
|
</record>
|
||||||
|
|
||||||
</data>
|
</data>
|
||||||
|
|
Loading…
Reference in New Issue