mirror of https://github.com/OCA/web.git
[MIG] web_timeline: Migration to 13.0
parent
4190881650
commit
18267a7cc2
|
@ -23,12 +23,12 @@ Web timeline
|
|||
:target: https://runbot.odoo-community.org/runbot/162/12.0
|
||||
:alt: Try me on Runbot
|
||||
|
||||
|badge1| |badge2| |badge3| |badge4| |badge5|
|
||||
|badge1| |badge2| |badge3| |badge4| |badge5|
|
||||
|
||||
Define a new view displaying events in an interactive visualization chart.
|
||||
|
||||
The widget is based on the external library
|
||||
http://visjs.org/timeline_examples.html
|
||||
https://visjs.github.io/vis-timeline/examples/timeline
|
||||
|
||||
**Table of contents**
|
||||
|
||||
|
@ -173,6 +173,7 @@ Authors
|
|||
* Tecnativa
|
||||
* Monk Software
|
||||
* Onestein
|
||||
* Trobz
|
||||
|
||||
Contributors
|
||||
~~~~~~~~~~~~
|
||||
|
@ -183,6 +184,7 @@ Contributors
|
|||
* Leonardo Donelli <donelli@webmonks.it>
|
||||
* Adrien Didenot <adrien.didenot@horanet.com>
|
||||
* Dennis Sluijk <d.sluijk@onestein.nl>
|
||||
* Thong Nguyen Van <thongnv@trobz.com>
|
||||
|
||||
Other credits
|
||||
~~~~~~~~~~~~~
|
||||
|
@ -211,7 +213,7 @@ promote its widespread use.
|
|||
|
||||
Current `maintainer <https://odoo-community.org/page/maintainer-role>`__:
|
||||
|
||||
|maintainer-tarteo|
|
||||
|maintainer-tarteo|
|
||||
|
||||
This module is part of the `OCA/web <https://github.com/OCA/web/tree/12.0/web_timeline>`_ project on GitHub.
|
||||
|
||||
|
|
|
@ -4,20 +4,21 @@
|
|||
{
|
||||
"name": "Web timeline",
|
||||
"summary": "Interactive visualization chart to show events in time",
|
||||
"version": "12.0.1.0.5",
|
||||
"version": "13.0.1.0.0",
|
||||
"development_status": "Production/Stable",
|
||||
"author": "ACSONE SA/NV, "
|
||||
"Tecnativa, "
|
||||
"Monk Software, "
|
||||
"Onestein, "
|
||||
"Trobz, "
|
||||
"Odoo Community Association (OCA)",
|
||||
"category": "web",
|
||||
"license": "AGPL-3",
|
||||
"application": False,
|
||||
"installable": True,
|
||||
"website": "https://github.com/OCA/web",
|
||||
"depends": ["web"],
|
||||
"qweb": ["static/src/xml/web_timeline.xml"],
|
||||
"data": ["views/web_timeline.xml"],
|
||||
"maintainers": ["tarteo"],
|
||||
"application": False,
|
||||
"installable": True,
|
||||
}
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
# Copyright 2016 ACSONE SA/NV (<http://acsone.eu>)
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from . import ir_view
|
||||
from . import ir_ui_view
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
# Copyright 2016 ACSONE SA/NV (<http://acsone.eu>)
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from openerp import fields, models
|
||||
from odoo import fields, models
|
||||
|
||||
TIMELINE_VIEW = ("timeline", "Timeline")
|
||||
|
|
@ -4,3 +4,4 @@
|
|||
* Leonardo Donelli <donelli@webmonks.it>
|
||||
* Adrien Didenot <adrien.didenot@horanet.com>
|
||||
* Dennis Sluijk <d.sluijk@onestein.nl>
|
||||
* Thong Nguyen Van <thongnv@trobz.com>
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
Define a new view displaying events in an interactive visualization chart.
|
||||
|
||||
The widget is based on the external library
|
||||
http://visjs.org/timeline_examples.html
|
||||
https://visjs.github.io/vis-timeline/examples/timeline
|
||||
|
|
|
@ -370,7 +370,7 @@ ul.auto-toc {
|
|||
<p><a class="reference external" href="https://odoo-community.org/page/development-status"><img alt="Production/Stable" src="https://img.shields.io/badge/maturity-Production%2FStable-green.png" /></a> <a class="reference external" href="http://www.gnu.org/licenses/agpl-3.0-standalone.html"><img alt="License: AGPL-3" src="https://img.shields.io/badge/licence-AGPL--3-blue.png" /></a> <a class="reference external" href="https://github.com/OCA/web/tree/12.0/web_timeline"><img alt="OCA/web" src="https://img.shields.io/badge/github-OCA%2Fweb-lightgray.png?logo=github" /></a> <a class="reference external" href="https://translation.odoo-community.org/projects/web-12-0/web-12-0-web_timeline"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external" href="https://runbot.odoo-community.org/runbot/162/12.0"><img alt="Try me on Runbot" src="https://img.shields.io/badge/runbot-Try%20me-875A7B.png" /></a></p>
|
||||
<p>Define a new view displaying events in an interactive visualization chart.</p>
|
||||
<p>The widget is based on the external library
|
||||
<a class="reference external" href="http://visjs.org/timeline_examples.html">http://visjs.org/timeline_examples.html</a></p>
|
||||
<a class="reference external" href="https://visjs.github.io/vis-timeline/examples/timeline">https://visjs.github.io/vis-timeline/examples/timeline</a></p>
|
||||
<p><strong>Table of contents</strong></p>
|
||||
<div class="contents local topic" id="contents">
|
||||
<ul class="simple">
|
||||
|
@ -549,6 +549,7 @@ If you spotted it first, help us smashing it by providing a detailed and welcome
|
|||
<li>Tecnativa</li>
|
||||
<li>Monk Software</li>
|
||||
<li>Onestein</li>
|
||||
<li>Trobz</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="contributors">
|
||||
|
@ -560,6 +561,7 @@ If you spotted it first, help us smashing it by providing a detailed and welcome
|
|||
<li>Leonardo Donelli <<a class="reference external" href="mailto:donelli@webmonks.it">donelli@webmonks.it</a>></li>
|
||||
<li>Adrien Didenot <<a class="reference external" href="mailto:adrien.didenot@horanet.com">adrien.didenot@horanet.com</a>></li>
|
||||
<li>Dennis Sluijk <<a class="reference external" href="mailto:d.sluijk@onestein.nl">d.sluijk@onestein.nl</a>></li>
|
||||
<li>Thong Nguyen Van <<a class="reference external" href="mailto:thongnv@trobz.com">thongnv@trobz.com</a>></li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="other-credits">
|
||||
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -1,31 +0,0 @@
|
|||
/* Button style */
|
||||
.openerp .oe_view_manager .oe_view_manager_switch .oe_vm_switch_timeline:after {
|
||||
content: "N";
|
||||
}
|
||||
|
||||
/* very light gray background in weekends */
|
||||
.vis-timeline .vis-grid.vis-saturday,
|
||||
.vis-timeline .vis-grid.vis-sunday {
|
||||
background: #DCDCDC;
|
||||
}
|
||||
|
||||
.vis-item .vis-item-overflow {
|
||||
overflow: visible;
|
||||
}
|
||||
|
||||
.oe_chatter_toggle {
|
||||
padding: 15px;
|
||||
}
|
||||
|
||||
.oe_timeline_view .vlabel .inner:hover{
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.oe_timeline_view svg.oe_timeline_view_canvas {
|
||||
display: block;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
position: absolute;
|
||||
left: 0px;
|
||||
top: 0px;
|
||||
}
|
|
@ -15,7 +15,7 @@ odoo.define('web_timeline.TimelineCanvas', function (require) {
|
|||
* Clears all drawings (svg elements) from the canvas.
|
||||
*/
|
||||
clear: function () {
|
||||
this.$el.find(' > :not(defs)').remove();
|
||||
this.$(' > :not(defs)').remove();
|
||||
},
|
||||
|
||||
/**
|
||||
|
@ -33,9 +33,7 @@ odoo.define('web_timeline.TimelineCanvas', function (require) {
|
|||
* @param {Number} breakAt The space between the line turns
|
||||
* @returns {Array} Each item represents a coordinate
|
||||
*/
|
||||
get_polyline_points: function (coordx1, coordy1, coordx2, coordy2,
|
||||
width1, height1, width2, height2,
|
||||
widthMarker, breakAt) {
|
||||
get_polyline_points: function (coordx1, coordy1, coordx2, coordy2, width1, height1, width2, height2, widthMarker, breakAt) {
|
||||
var halfHeight1 = height1 / 2;
|
||||
var halfHeight2 = height2 / 2;
|
||||
var x1 = coordx1 - widthMarker;
|
||||
|
@ -52,14 +50,14 @@ odoo.define('web_timeline.TimelineCanvas', function (require) {
|
|||
if (xDiff > threshold) {
|
||||
points.push([x1 - breakAt, y1]);
|
||||
points.push([x1 - breakAt, y1 - yDiff]);
|
||||
} else if (xDiff <= threshold) {
|
||||
} else {
|
||||
var yDiffSpace = yDiff > 0 ? spaceY : -spaceY;
|
||||
points.push([x1 - breakAt, y1]);
|
||||
points.push([x1 - breakAt, y2 + yDiffSpace]);
|
||||
points.push([x2 + breakAt, y2 + yDiffSpace]);
|
||||
points.push([x2 + breakAt, y2]);
|
||||
}
|
||||
} else if(x1 < x2) {
|
||||
} else if (x1 < x2) {
|
||||
points.push([x1 - breakAt, y1]);
|
||||
points.push([x1 - breakAt, y1 + spaceY]);
|
||||
points.push([x2 + breakAt, y2 + spaceY]);
|
||||
|
@ -105,11 +103,9 @@ odoo.define('web_timeline.TimelineCanvas', function (require) {
|
|||
width2 = to.clientWidth,
|
||||
height2 = to.clientHeight;
|
||||
|
||||
var points = this.get_polyline_points(
|
||||
x1, y1, x2, y2, width1, height1, width2, height2, widthMarker, breakLineAt
|
||||
);
|
||||
var points = this.get_polyline_points(x1, y1, x2, y2, width1, height1, width2, height2, widthMarker, breakLineAt);
|
||||
|
||||
var polyline_points = _.map(points, function(point) {
|
||||
var polyline_points = _.map(points, function (point) {
|
||||
return point.join(',');
|
||||
}).join();
|
||||
|
||||
|
|
|
@ -19,7 +19,6 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
}),
|
||||
|
||||
/**
|
||||
* @constructor
|
||||
* @override
|
||||
*/
|
||||
init: function (parent, model, renderer, params) {
|
||||
|
@ -38,11 +37,11 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
*/
|
||||
update: function (params, options) {
|
||||
var res = this._super.apply(this, arguments);
|
||||
if (_.isEmpty(params)){
|
||||
if (_.isEmpty(params)) {
|
||||
return res;
|
||||
}
|
||||
var defaults = _.defaults({}, options, {
|
||||
adjust_window: true
|
||||
adjust_window: true,
|
||||
});
|
||||
var self = this;
|
||||
var domains = params.domain;
|
||||
|
@ -50,14 +49,11 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
var group_bys = params.groupBy;
|
||||
this.last_domains = domains;
|
||||
this.last_contexts = contexts;
|
||||
// select the group by
|
||||
var n_group_bys = [];
|
||||
if (this.renderer.arch.attrs.default_group_by) {
|
||||
// Select the group by
|
||||
var n_group_bys = group_bys;
|
||||
if (!n_group_bys.length && this.renderer.arch.attrs.default_group_by) {
|
||||
n_group_bys = this.renderer.arch.attrs.default_group_by.split(',');
|
||||
}
|
||||
if (group_bys.length) {
|
||||
n_group_bys = group_bys;
|
||||
}
|
||||
this.renderer.last_group_bys = n_group_bys;
|
||||
this.renderer.last_domains = domains;
|
||||
|
||||
|
@ -83,6 +79,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
* Gets triggered when a group in the timeline is clicked (by the TimelineRenderer).
|
||||
*
|
||||
* @private
|
||||
* @param {EventObject} event
|
||||
* @returns {jQuery.Deferred}
|
||||
*/
|
||||
_onGroupClick: function (event) {
|
||||
|
@ -92,7 +89,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
res_model: this.renderer.view.fields[groupField].relation,
|
||||
res_id: event.data.item.group,
|
||||
target: 'new',
|
||||
views: [[false, 'form']]
|
||||
views: [[false, 'form']],
|
||||
});
|
||||
},
|
||||
|
||||
|
@ -100,18 +97,19 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
* Opens a form view of a clicked timeline item (triggered by the TimelineRenderer).
|
||||
*
|
||||
* @private
|
||||
* @param {EventObject} event
|
||||
*/
|
||||
_onUpdate: function (event) {
|
||||
var self = this;
|
||||
this.renderer = event.data.renderer;
|
||||
var rights = event.data.rights;
|
||||
var item = event.data.item;
|
||||
var id = item.evt.id;
|
||||
var id = Number(item.evt.id) || item.evt.id;
|
||||
var title = item.evt.__name;
|
||||
if (this.open_popup_action) {
|
||||
new dialogs.FormViewDialog(this, {
|
||||
res_model: this.model.modelName,
|
||||
res_id: parseInt(id, 10).toString() === id ? parseInt(id, 10) : id,
|
||||
res_id: id,
|
||||
context: this.getSession().user_context,
|
||||
title: title,
|
||||
view_id: Number(this.open_popup_action),
|
||||
|
@ -126,7 +124,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
}
|
||||
this.trigger_up('switch_view', {
|
||||
view_type: 'form',
|
||||
res_id: parseInt(id, 10).toString() === id ? parseInt(id, 10) : id,
|
||||
res_id: id,
|
||||
mode: mode,
|
||||
model: this.model.modelName,
|
||||
});
|
||||
|
@ -137,6 +135,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
* Gets triggered when a timeline item is moved (triggered by the TimelineRenderer).
|
||||
*
|
||||
* @private
|
||||
* @param {EventObject} event
|
||||
*/
|
||||
_onMove: function (event) {
|
||||
var item = event.data.item;
|
||||
|
@ -149,7 +148,8 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
group = item.group;
|
||||
}
|
||||
var data = {};
|
||||
// In case of a move event, the date_delay stay the same, only date_start and stop must be updated
|
||||
// In case of a move event, the date_delay stay the same,
|
||||
// only date_start and stop must be updated
|
||||
data[this.date_start] = time.auto_date_to_str(event_start, fields[this.date_start].type);
|
||||
if (this.date_stop) {
|
||||
// In case of instantaneous event, item.end is not defined
|
||||
|
@ -170,7 +170,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
this.moveQueue.push({
|
||||
id: event.data.item.id,
|
||||
data: data,
|
||||
event: event
|
||||
event: event,
|
||||
});
|
||||
|
||||
this.debouncedInternalMove();
|
||||
|
@ -184,10 +184,10 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
*/
|
||||
internalMove: function () {
|
||||
var self = this;
|
||||
var queue = this.moveQueue.slice();
|
||||
var queues = this.moveQueue.slice();
|
||||
this.moveQueue = [];
|
||||
var defers = [];
|
||||
_.each(queue, function(item) {
|
||||
for (const item of queues) {
|
||||
defers.push(self._rpc({
|
||||
model: self.model.modelName,
|
||||
method: 'write',
|
||||
|
@ -196,13 +196,13 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
item.data,
|
||||
],
|
||||
context: self.getSession().user_context,
|
||||
}).then(function() {
|
||||
}).then(function () {
|
||||
item.event.data.callback(item.event.data.item);
|
||||
}));
|
||||
});
|
||||
return $.when.apply($, defers).done(function() {
|
||||
}
|
||||
return $.when.apply($, defers).done(function () {
|
||||
self.write_completed({
|
||||
adjust_window: false
|
||||
adjust_window: false,
|
||||
});
|
||||
});
|
||||
},
|
||||
|
@ -212,51 +212,30 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
* Requires user confirmation before it gets actually deleted.
|
||||
*
|
||||
* @private
|
||||
* @param {EventObject} event
|
||||
* @returns {jQuery.Deferred}
|
||||
*/
|
||||
_onRemove: function (e) {
|
||||
_onRemove: function (event) {
|
||||
var self = this;
|
||||
|
||||
function do_it(event) {
|
||||
return self._rpc({
|
||||
model: self.model.modelName,
|
||||
method: 'unlink',
|
||||
args: [
|
||||
[event.data.item.id],
|
||||
],
|
||||
context: self.getSession().user_context,
|
||||
}).then(function () {
|
||||
var unlink_index = false;
|
||||
for (var i = 0; i < self.model.data.data.length; i++) {
|
||||
if (self.model.data.data[i].id === event.data.item.id) {
|
||||
unlink_index = i;
|
||||
}
|
||||
}
|
||||
if (!isNaN(unlink_index)) {
|
||||
self.model.data.data.splice(unlink_index, 1);
|
||||
}
|
||||
|
||||
event.data.callback(event.data.item);
|
||||
});
|
||||
}
|
||||
|
||||
var message = _t("Are you sure you want to delete this record?");
|
||||
var def = $.Deferred();
|
||||
Dialog.confirm(this, message, {
|
||||
|
||||
Dialog.confirm(this, _t("Are you sure you want to delete this record?"), {
|
||||
title: _t("Warning"),
|
||||
confirm_callback: function() {
|
||||
do_it(e)
|
||||
.done(def.resolve.bind(def, true))
|
||||
.fail(def.reject.bind(def));
|
||||
confirm_callback: function () {
|
||||
self.remove_completed(event).then(def.resolve.bind(def));
|
||||
},
|
||||
cancel_callback: def.resolve.bind(def),
|
||||
});
|
||||
return def.promise();
|
||||
|
||||
return def;
|
||||
},
|
||||
|
||||
/**
|
||||
* Triggered when a timeline item gets added and opens a form view.
|
||||
*
|
||||
* @private
|
||||
* @param {EventObject} event
|
||||
* @returns {dialogs.FormViewDialog}
|
||||
*/
|
||||
_onAdd: function (event) {
|
||||
var self = this;
|
||||
|
@ -300,6 +279,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
* Triggered upon completion of a new record.
|
||||
* Updates the timeline view with the new record.
|
||||
*
|
||||
* @param {RecordId} id
|
||||
* @returns {jQuery.Deferred}
|
||||
*/
|
||||
create_completed: function (id) {
|
||||
|
@ -312,8 +292,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
this.model.fieldNames,
|
||||
],
|
||||
context: this.context,
|
||||
})
|
||||
.then(function (records) {
|
||||
}).then(function (records) {
|
||||
var new_event = self.renderer.event_data_transform(records[0]);
|
||||
var items = self.renderer.timeline.itemsData;
|
||||
items.add(new_event);
|
||||
|
@ -324,6 +303,7 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
|
||||
/**
|
||||
* Triggered upon completion of writing a record.
|
||||
* @param {ControllerOptions} options
|
||||
*/
|
||||
write_completed: function (options) {
|
||||
var params = {
|
||||
|
@ -331,9 +311,34 @@ odoo.define('web_timeline.TimelineController', function (require) {
|
|||
context: this.context,
|
||||
groupBy: this.renderer.last_group_bys,
|
||||
};
|
||||
|
||||
this.update(params, options);
|
||||
},
|
||||
|
||||
/**
|
||||
* Triggered upon confirm of removing a record.
|
||||
* @param {EventObject} event
|
||||
* @returns {jQuery.Deferred}
|
||||
*/
|
||||
remove_completed: function (event) {
|
||||
var self = this;
|
||||
return self._rpc({
|
||||
model: self.modelName,
|
||||
method: 'unlink',
|
||||
args: [[event.data.item.id]],
|
||||
context: self.getSession().user_context,
|
||||
}).then(function () {
|
||||
var unlink_index = false;
|
||||
for (var i = 0; i < self.model.data.data.length; i++) {
|
||||
if (self.model.data.data[i].id === event.data.item.id) {
|
||||
unlink_index = i;
|
||||
}
|
||||
}
|
||||
if (!isNaN(unlink_index)) {
|
||||
self.model.data.data.splice(unlink_index, 1);
|
||||
}
|
||||
event.data.callback(event.data.item);
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
return TimelineController;
|
||||
|
|
|
@ -5,16 +5,10 @@ odoo.define('web_timeline.TimelineModel', function (require) {
|
|||
|
||||
var TimelineModel = AbstractModel.extend({
|
||||
|
||||
/**
|
||||
* @constructor
|
||||
*/
|
||||
init: function () {
|
||||
this._super.apply(this, arguments);
|
||||
},
|
||||
|
||||
/**
|
||||
* @override
|
||||
*/
|
||||
load: function (params) {
|
||||
var self = this;
|
||||
this.modelName = params.modelName;
|
||||
|
@ -22,10 +16,22 @@ odoo.define('web_timeline.TimelineModel', function (require) {
|
|||
if (!this.preload_def) {
|
||||
this.preload_def = $.Deferred();
|
||||
$.when(
|
||||
this._rpc({model: this.modelName, method: 'check_access_rights', args: ["write", false]}),
|
||||
this._rpc({model: this.modelName, method: 'check_access_rights', args: ["unlink", false]}),
|
||||
this._rpc({model: this.modelName, method: 'check_access_rights', args: ["create", false]}))
|
||||
.then(function (write, unlink, create) {
|
||||
this._rpc({
|
||||
model: this.modelName,
|
||||
method: 'check_access_rights',
|
||||
args: ["write", false],
|
||||
}),
|
||||
this._rpc({
|
||||
model: this.modelName,
|
||||
method: 'check_access_rights',
|
||||
args: ["unlink", false],
|
||||
}),
|
||||
this._rpc({
|
||||
model: this.modelName,
|
||||
method: 'check_access_rights',
|
||||
args: ["create", false],
|
||||
})
|
||||
).then(function (write, unlink, create) {
|
||||
self.write_right = write;
|
||||
self.unlink_right = unlink;
|
||||
self.create_right = create;
|
||||
|
@ -55,8 +61,7 @@ odoo.define('web_timeline.TimelineModel', function (require) {
|
|||
context: self.data.context,
|
||||
fields: self.fieldNames,
|
||||
domain: self.data.domain,
|
||||
})
|
||||
.then(function (events) {
|
||||
}).then(function (events) {
|
||||
self.data.data = events;
|
||||
self.data.rights = {
|
||||
'unlink': self.unlink_right,
|
||||
|
|
|
@ -24,9 +24,6 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
'click .oe_timeline_button_scale_year': '_onScaleYearClicked',
|
||||
}),
|
||||
|
||||
/**
|
||||
* @constructor
|
||||
*/
|
||||
init: function (parent, state, params) {
|
||||
this._super.apply(this, arguments);
|
||||
this.modelName = params.model;
|
||||
|
@ -53,11 +50,11 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
var attrs = this.arch.attrs;
|
||||
this.current_window = {
|
||||
start: new moment(),
|
||||
end: new moment().add(24, 'hours')
|
||||
end: new moment().add(24, 'hours'),
|
||||
};
|
||||
|
||||
this.$el.addClass(attrs.class);
|
||||
this.$timeline = this.$el.find(".oe_timeline_widget");
|
||||
this.$timeline = this.$('.oe_timeline_widget');
|
||||
|
||||
if (!this.date_start) {
|
||||
throw new Error(_t("Timeline view has not defined 'date_start' attribute."));
|
||||
|
@ -68,11 +65,11 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
/**
|
||||
* Triggered when the timeline is attached to the DOM.
|
||||
*/
|
||||
on_attach_callback: function() {
|
||||
var height = this.$el.parent().height() - this.$el.find('.oe_timeline_buttons').height();
|
||||
if (height > this.min_height) {
|
||||
on_attach_callback: function () {
|
||||
var height = this.$el.parent().height() - this.$('.oe_timeline_buttons').height();
|
||||
if (height > this.min_height && this.timeline) {
|
||||
this.timeline.setOptions({
|
||||
height: height
|
||||
height: height,
|
||||
});
|
||||
}
|
||||
},
|
||||
|
@ -99,7 +96,7 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
_onTodayClicked: function () {
|
||||
this.current_window = {
|
||||
start: new moment(),
|
||||
end: new moment().add(24, 'hours')
|
||||
end: new moment().add(24, 'hours'),
|
||||
};
|
||||
|
||||
if (this.timeline) {
|
||||
|
@ -131,7 +128,7 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
* @private
|
||||
*/
|
||||
_onScaleMonthClicked: function () {
|
||||
this._scaleCurrentWindow(24 * 30);
|
||||
this._scaleCurrentWindow(24 * moment(this.current_window.start).daysInMonth());
|
||||
},
|
||||
|
||||
/**
|
||||
|
@ -140,7 +137,7 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
* @private
|
||||
*/
|
||||
_onScaleYearClicked: function () {
|
||||
this._scaleCurrentWindow(24 * 365);
|
||||
this._scaleCurrentWindow(24 * (moment(this.current_window.start).isLeapYear() ? 366 : 365));
|
||||
},
|
||||
|
||||
/**
|
||||
|
@ -166,30 +163,30 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
if (this.mode) {
|
||||
var start = false, end = false;
|
||||
switch (this.mode) {
|
||||
case 'day':
|
||||
start = new moment().startOf('day');
|
||||
end = new moment().endOf('day');
|
||||
break;
|
||||
case 'week':
|
||||
start = new moment().startOf('week');
|
||||
end = new moment().endOf('week');
|
||||
break;
|
||||
case 'month':
|
||||
start = new moment().startOf('month');
|
||||
end = new moment().endOf('month');
|
||||
break;
|
||||
case 'day':
|
||||
start = new moment().startOf('day');
|
||||
end = new moment().endOf('day');
|
||||
break;
|
||||
case 'week':
|
||||
start = new moment().startOf('week');
|
||||
end = new moment().endOf('week');
|
||||
break;
|
||||
case 'month':
|
||||
start = new moment().startOf('month');
|
||||
end = new moment().endOf('month');
|
||||
break;
|
||||
}
|
||||
if (end && start) {
|
||||
this.options.start = start;
|
||||
this.options.end = end;
|
||||
} else {
|
||||
this.mode = 'fit';
|
||||
this.mode = 'fit';
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
* Initializes the timeline (http://visjs.org/docs/timeline/).
|
||||
* Initializes the timeline (https://visjs.github.io/vis-timeline/docs/timeline).
|
||||
*
|
||||
* @private
|
||||
*/
|
||||
|
@ -197,32 +194,32 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
var self = this;
|
||||
this._computeMode();
|
||||
this.options.editable = {
|
||||
// add new items by double tapping
|
||||
// Add new items by double tapping
|
||||
add: this.modelClass.data.rights.create,
|
||||
// drag items horizontally
|
||||
// Drag items horizontally
|
||||
updateTime: this.modelClass.data.rights.write,
|
||||
// drag items from one group to another
|
||||
// Drag items from one group to another
|
||||
updateGroup: this.modelClass.data.rights.write,
|
||||
// delete an item by tapping the delete button top right
|
||||
// Delete an item by tapping the delete button top right
|
||||
remove: this.modelClass.data.rights.unlink,
|
||||
};
|
||||
$.extend(this.options, {
|
||||
onAdd: self.on_add,
|
||||
onMove: self.on_move,
|
||||
onUpdate: self.on_update,
|
||||
onRemove: self.on_remove
|
||||
onRemove: self.on_remove,
|
||||
});
|
||||
this.qweb = new QWeb(session.debug, {_s: session.origin}, false);
|
||||
if (this.arch.children.length) {
|
||||
var tmpl = utils.json_node_to_xml(
|
||||
_.filter(this.arch.children, function(item) {
|
||||
_.filter(this.arch.children, function (item) {
|
||||
return item.tag === 'templates';
|
||||
})[0]
|
||||
);
|
||||
this.qweb.add_template(tmpl);
|
||||
}
|
||||
|
||||
this.timeline = new vis.Timeline(self.$timeline.empty().get(0));
|
||||
this.timeline = new vis.Timeline(self.$timeline.get(0));
|
||||
this.timeline.setOptions(this.options);
|
||||
if (self.mode && self['on_scale_' + self.mode + '_clicked']) {
|
||||
self['on_scale_' + self.mode + '_clicked']();
|
||||
|
@ -235,11 +232,12 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
this.$centerContainer = $(this.timeline.dom.centerContainer);
|
||||
this.canvas = new TimelineCanvas(this);
|
||||
this.canvas.appendTo(this.$centerContainer);
|
||||
this.timeline.on('changed', function() {
|
||||
this.timeline.on('changed', function () {
|
||||
self.draw_canvas();
|
||||
self.canvas.$el.attr(
|
||||
'style',
|
||||
self.$el.find('.vis-content').attr('style') + self.$el.find('.vis-itemset').attr('style')
|
||||
self.$('.vis-content').attr('style') +
|
||||
self.$('.vis-itemset').attr('style')
|
||||
);
|
||||
});
|
||||
},
|
||||
|
@ -264,16 +262,16 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
draw_dependencies: function () {
|
||||
var self = this;
|
||||
var items = this.timeline.itemSet.items;
|
||||
_.each(items, function(item) {
|
||||
for (const item of items) {
|
||||
if (!item.data.evt) {
|
||||
return;
|
||||
}
|
||||
_.each(item.data.evt[self.dependency_arrow], function(id) {
|
||||
for (const id of item.data.evt[self.dependency_arrow]) {
|
||||
if (id in items) {
|
||||
self.draw_dependency(item, items[id]);
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
|
@ -293,7 +291,7 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
|
||||
var defaults = _.defaults({}, options, {
|
||||
line_color: 'black',
|
||||
line_width: 1
|
||||
line_width: 1,
|
||||
});
|
||||
|
||||
this.canvas.draw_arrow(from.dom.box, to.dom.box, defaults.line_color, defaults.line_width);
|
||||
|
@ -315,12 +313,12 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
ids,
|
||||
],
|
||||
context: this.getSession().user_context,
|
||||
}).then(function(names) {
|
||||
}).then(function (names) {
|
||||
var nevents = _.map(events, function (event) {
|
||||
return _.extend({
|
||||
__name: _.detect(names, function (name) {
|
||||
return name[0] === event.id;
|
||||
})[1]
|
||||
})[1],
|
||||
}, event);
|
||||
});
|
||||
return self.on_data_loaded_2(nevents, group_bys, adjust_window);
|
||||
|
@ -337,11 +335,11 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
var data = [];
|
||||
var groups = [];
|
||||
this.grouped_by = group_bys;
|
||||
_.each(events, function (event) {
|
||||
if (event[self.date_start]) {
|
||||
data.push(self.event_data_transform(event));
|
||||
for (const evt of events) {
|
||||
if (evt[self.date_start]) {
|
||||
data.push(self.event_data_transform(evt));
|
||||
}
|
||||
});
|
||||
}
|
||||
groups = this.split_groups(events, group_bys);
|
||||
this.timeline.setGroups(groups);
|
||||
this.timeline.setItems(data);
|
||||
|
@ -364,24 +362,22 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
}
|
||||
var groups = [];
|
||||
groups.push({id: -1, content: _t('-')});
|
||||
_.each(events, function (event) {
|
||||
var group_name = event[_.first(group_bys)];
|
||||
for (const evt of events) {
|
||||
var group_name = evt[_.first(group_bys)];
|
||||
if (group_name) {
|
||||
if (group_name instanceof Array) {
|
||||
var group = _.find(groups, function (existing_group) {
|
||||
return _.isEqual(existing_group.id, group_name[0]);
|
||||
return existing_group.id === group_name[0];
|
||||
});
|
||||
|
||||
if (_.isUndefined(group)) {
|
||||
group = {
|
||||
groups.push({
|
||||
id: group_name[0],
|
||||
content: group_name[1]
|
||||
};
|
||||
groups.push(group);
|
||||
content: group_name[1],
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
return groups;
|
||||
},
|
||||
|
||||
|
@ -412,7 +408,7 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
}
|
||||
|
||||
if (!date_stop && date_delay) {
|
||||
date_stop = moment(date_start).add(date_delay, 'hours').toDate();
|
||||
date_stop = date_start.clone().add(date_delay, 'hours').toDate();
|
||||
}
|
||||
|
||||
var group = evt[self.last_group_bys[0]];
|
||||
|
@ -421,13 +417,14 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
} else {
|
||||
group = -1;
|
||||
}
|
||||
_.each(self.colors, function (color) {
|
||||
if (eval("'" + evt[color.field] + "' " + color.opt + " '" + color.value + "'")) {
|
||||
|
||||
for (const color of self.colors) {
|
||||
if (py.eval("'" + evt[color.field] + "' " + color.opt + " '" + color.value + "'")) {
|
||||
self.color = color.color;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
var content = _.isUndefined(evt.__name) ? evt.display_name : evt.__name;
|
||||
var content = evt.__name || evt.display_name;
|
||||
if (this.arch.children.length) {
|
||||
content = this.render_timeline_item(evt);
|
||||
}
|
||||
|
@ -438,13 +435,14 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
'id': evt.id,
|
||||
'group': group,
|
||||
'evt': evt,
|
||||
'style': 'background-color: ' + self.color + ';'
|
||||
'style': `background-color: ${this.color};`,
|
||||
};
|
||||
// Check if the event is instantaneous, if so, display it with a point on the timeline (no 'end')
|
||||
// Check if the event is instantaneous,
|
||||
// if so, display it with a point on the timeline (no 'end')
|
||||
if (date_stop && !moment(date_start).isSame(date_stop)) {
|
||||
r.end = date_stop;
|
||||
}
|
||||
self.color = null;
|
||||
this.color = null;
|
||||
return r;
|
||||
},
|
||||
|
||||
|
@ -456,10 +454,10 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
* @returns {String} Rendered template
|
||||
*/
|
||||
render_timeline_item: function (evt) {
|
||||
if(this.qweb.has_template('timeline-item')) {
|
||||
if (this.qweb.has_template('timeline-item')) {
|
||||
return this.qweb.render('timeline-item', {
|
||||
'record': evt,
|
||||
'field_utils': field_utils
|
||||
'field_utils': field_utils,
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -475,7 +473,7 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
*/
|
||||
on_group_click: function (e) {
|
||||
if (e.what === 'group-label' && e.group !== -1) {
|
||||
this._trigger(e, function() {
|
||||
this._trigger(e, function () {
|
||||
// Do nothing
|
||||
}, 'onGroupClick');
|
||||
}
|
||||
|
@ -518,7 +516,7 @@ odoo.define('web_timeline.TimelineRenderer', function (require) {
|
|||
},
|
||||
|
||||
/**
|
||||
* trigger_up encapsulation adds by default the rights, and the renderer.
|
||||
* Trigger_up encapsulation adds by default the rights, and the renderer.
|
||||
*
|
||||
* @private
|
||||
*/
|
||||
|
|
|
@ -3,25 +3,18 @@
|
|||
* Copyright 2016 Pedro M. Baeza <pedro.baeza@tecnativa.com>
|
||||
* License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). */
|
||||
|
||||
_.str.toBoolElse = function (str, elseValues, trueValues, falseValues) {
|
||||
var ret = _.str.toBool(str, trueValues, falseValues);
|
||||
if (_.isUndefined(ret)) {
|
||||
return elseValues;
|
||||
}
|
||||
return ret;
|
||||
};
|
||||
|
||||
|
||||
odoo.define('web_timeline.TimelineView', function (require) {
|
||||
"use strict";
|
||||
|
||||
var core = require('web.core');
|
||||
var utils = require('web.utils');
|
||||
var view_registry = require('web.view_registry');
|
||||
var AbstractView = require('web.AbstractView');
|
||||
var TimelineRenderer = require('web_timeline.TimelineRenderer');
|
||||
var TimelineController = require('web_timeline.TimelineController');
|
||||
var TimelineModel = require('web_timeline.TimelineModel');
|
||||
|
||||
|
||||
var _lt = core._lt;
|
||||
|
||||
function isNullOrUndef(value) {
|
||||
|
@ -31,8 +24,8 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
var TimelineView = AbstractView.extend({
|
||||
display_name: _lt('Timeline'),
|
||||
icon: 'fa-clock-o',
|
||||
jsLibs: ['/web_timeline/static/lib/vis/vis-timeline-graph2d.min.js'],
|
||||
cssLibs: ['/web_timeline/static/lib/vis/vis-timeline-graph2d.min.css'],
|
||||
jsLibs: ['/web_timeline/static/lib/vis-timeline/vis-timeline-graph2d.min.js'],
|
||||
cssLibs: ['/web_timeline/static/lib/vis-timeline/vis-timeline-graph2d.min.css'],
|
||||
config: {
|
||||
Model: TimelineModel,
|
||||
Controller: TimelineController,
|
||||
|
@ -40,7 +33,6 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
},
|
||||
|
||||
/**
|
||||
* @constructor
|
||||
* @override
|
||||
*/
|
||||
init: function (viewInfo, params) {
|
||||
|
@ -65,17 +57,17 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
|
||||
fieldsToGather.push(attrs.default_group_by);
|
||||
|
||||
_.each(fieldsToGather, function (field) {
|
||||
for (const field of fieldsToGather) {
|
||||
if (attrs[field]) {
|
||||
var fieldName = attrs[field];
|
||||
mapping[field] = fieldName;
|
||||
fieldNames.push(fieldName);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
var archFieldNames = _.map(_.filter(this.arch.children, function(item) {
|
||||
var archFieldNames = _.map(_.filter(this.arch.children, function (item) {
|
||||
return item.tag === 'field';
|
||||
}), function(item) {
|
||||
}), function (item) {
|
||||
return item.attrs.name;
|
||||
});
|
||||
fieldNames = _.union(
|
||||
|
@ -84,8 +76,8 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
);
|
||||
|
||||
this.parse_colors();
|
||||
for (var i=0; i<this.colors.length; i++) {
|
||||
fieldNames.push(this.colors[i].field);
|
||||
for (const color of this.colors) {
|
||||
fieldNames.push(color.field);
|
||||
}
|
||||
|
||||
if (attrs.dependency_arrow) {
|
||||
|
@ -107,13 +99,13 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
|
||||
this.current_window = {
|
||||
start: new moment(),
|
||||
end: new moment().add(24, 'hours')
|
||||
end: new moment().add(24, 'hours'),
|
||||
};
|
||||
if (!isNullOrUndef(attrs.quick_create_instance)) {
|
||||
self.quick_create_instance = 'instance.' + attrs.quick_create_instance;
|
||||
}
|
||||
this.stack = true;
|
||||
if (!isNullOrUndef(attrs.stack) && !_.str.toBoolElse(attrs.stack, "true")) {
|
||||
if (!isNullOrUndef(attrs.stack) && !utils.toBoolElse(attrs.stack, true)) {
|
||||
this.stack = false;
|
||||
}
|
||||
this.options = {
|
||||
|
@ -124,9 +116,9 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
showCurrentTime: true,
|
||||
stack: this.stack,
|
||||
margin: JSON.parse(this.margin),
|
||||
zoomKey: this.zoomKey
|
||||
zoomKey: this.zoomKey,
|
||||
};
|
||||
if (isNullOrUndef(attrs.event_open_popup) || !_.str.toBoolElse(attrs.event_open_popup, true)) {
|
||||
if (isNullOrUndef(attrs.event_open_popup) || !utils.toBoolElse(attrs.event_open_popup, true)) {
|
||||
this.open_popup_action = false;
|
||||
} else {
|
||||
this.open_popup_action = attrs.event_open_popup;
|
||||
|
@ -152,23 +144,23 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
this.controllerParams.date_stop = this.date_stop;
|
||||
this.controllerParams.date_delay = this.date_delay;
|
||||
this.controllerParams.actionContext = this.action.context;
|
||||
return this;
|
||||
this.withSearchPanel = false;
|
||||
},
|
||||
|
||||
/**
|
||||
* Order function for groups.
|
||||
* @returns {Integer}
|
||||
*/
|
||||
group_order: function (grp1, grp2) {
|
||||
// display non grouped elements first
|
||||
// Display non grouped elements first
|
||||
if (grp1.id === -1) {
|
||||
return -1;
|
||||
}
|
||||
if (grp2.id === -1) {
|
||||
return +1;
|
||||
return 1;
|
||||
}
|
||||
|
||||
return grp1.content.localeCompare(grp2.content);
|
||||
|
||||
},
|
||||
|
||||
/**
|
||||
|
@ -185,7 +177,7 @@ odoo.define('web_timeline.TimelineView', function (require) {
|
|||
'color': color,
|
||||
'field': temp.expressions[0].value,
|
||||
'opt': temp.operators[0],
|
||||
'value': temp.expressions[1].value
|
||||
'value': temp.expressions[1].value,
|
||||
};
|
||||
}).value();
|
||||
} else {
|
||||
|
|
|
@ -0,0 +1,17 @@
|
|||
.oe_timeline_view .vis-timeline {
|
||||
.vis-grid {
|
||||
.vis-saturday, .vis-sunday {
|
||||
// very light gray background in weekends
|
||||
background: #DCDCDC;
|
||||
}
|
||||
}
|
||||
|
||||
.vis-item {
|
||||
&.vis-box:hover {
|
||||
cursor: pointer !important;
|
||||
}
|
||||
&.vis-item-overflow {
|
||||
overflow: visible;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,26 +1,26 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<template>
|
||||
<t t-name="TimelineView">
|
||||
<div class="oe_timeline_view">
|
||||
<div class="oe_timeline_buttons">
|
||||
<button class="btn btn-default btn-sm oe_timeline_button_today">Today</button>
|
||||
<t t-name="TimelineView">
|
||||
<div class="oe_timeline_view">
|
||||
<div class="oe_timeline_buttons">
|
||||
<button class="btn btn-default btn-sm oe_timeline_button_today">Today</button>
|
||||
|
||||
<div class="btn-group btn-sm">
|
||||
<button class="btn btn-default oe_timeline_button_scale_day">Day</button>
|
||||
<button class="btn btn-default oe_timeline_button_scale_week">Week</button>
|
||||
<button class="btn btn-default oe_timeline_button_scale_month">Month</button>
|
||||
<button class="btn btn-default oe_timeline_button_scale_year">Year</button>
|
||||
<div class="btn-group btn-sm">
|
||||
<button class="btn btn-default oe_timeline_button_scale_day">Day</button>
|
||||
<button class="btn btn-default oe_timeline_button_scale_week">Week</button>
|
||||
<button class="btn btn-default oe_timeline_button_scale_month">Month</button>
|
||||
<button class="btn btn-default oe_timeline_button_scale_year">Year</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="oe_timeline_widget" />
|
||||
</div>
|
||||
</t>
|
||||
<div class="oe_timeline_widget"/>
|
||||
</div>
|
||||
</t>
|
||||
|
||||
<svg t-name="TimelineView.Canvas"
|
||||
class="oe_timeline_view_canvas">
|
||||
<defs>
|
||||
<marker id="arrowhead" markerWidth="10" markerHeight="7" refX="10" refY="3.5" orient="auto">
|
||||
<polygon points="10 0, 10 7, 0 3.5" />
|
||||
</marker>
|
||||
</defs>
|
||||
</svg>
|
||||
<svg t-name="TimelineView.Canvas" class="oe_timeline_view_canvas">
|
||||
<defs>
|
||||
<marker id="arrowhead" markerWidth="10" markerHeight="7" refX="10" refY="3.5" orient="auto">
|
||||
<polygon points="10 0, 10 7, 0 3.5"/>
|
||||
</marker>
|
||||
</defs>
|
||||
</svg>
|
||||
</template>
|
||||
|
|
|
@ -1,12 +1,11 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<odoo>
|
||||
|
||||
<template id="assets_backend" name="web_timeline assets" inherit_id="web.assets_backend">
|
||||
<xpath expr="." position="inside">
|
||||
<link rel="stylesheet" href="/web_timeline/static/lib/vis/vis-timeline-graph2d.min.css"/>
|
||||
<link rel="stylesheet" href="/web_timeline/static/src/css/web_timeline.css"/>
|
||||
<link rel="stylesheet" type="text/css" href="/web_timeline/static/lib/vis-timeline/vis-timeline-graph2d.min.css"/>
|
||||
<link rel="stylesheet" type="text/scss" href="/web_timeline/static/src/scss/web_timeline.scss"/>
|
||||
|
||||
<script type="text/javascript" src="/web_timeline/static/lib/vis/vis-timeline-graph2d.min.js"/>
|
||||
<script type="text/javascript" src="/web_timeline/static/lib/vis-timeline/vis-timeline-graph2d.min.js"/>
|
||||
<script type="text/javascript" src="/web_timeline/static/src/js/timeline_view.js"/>
|
||||
<script type="text/javascript" src="/web_timeline/static/src/js/timeline_renderer.js"/>
|
||||
<script type="text/javascript" src="/web_timeline/static/src/js/timeline_controller.js"/>
|
||||
|
@ -14,5 +13,4 @@
|
|||
<script type="text/javascript" src="/web_timeline/static/src/js/timeline_canvas.js"/>
|
||||
</xpath>
|
||||
</template>
|
||||
|
||||
</odoo>
|
||||
|
|
Loading…
Reference in New Issue