summaryrefslogtreecommitdiff
path: root/addons/board/static/tests
diff options
context:
space:
mode:
authorstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
committerstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
commit3751379f1e9a4c215fb6eb898b4ccc67659b9ace (patch)
treea44932296ef4a9b71d5f010906253d8c53727726 /addons/board/static/tests
parent0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (diff)
initial commit 2
Diffstat (limited to 'addons/board/static/tests')
-rw-r--r--addons/board/static/tests/dashboard_tests.js1153
1 files changed, 1153 insertions, 0 deletions
diff --git a/addons/board/static/tests/dashboard_tests.js b/addons/board/static/tests/dashboard_tests.js
new file mode 100644
index 00000000..8ba4be76
--- /dev/null
+++ b/addons/board/static/tests/dashboard_tests.js
@@ -0,0 +1,1153 @@
+odoo.define('board.dashboard_tests', function (require) {
+"use strict";
+
+var BoardView = require('board.BoardView');
+
+var ListController = require('web.ListController');
+var testUtils = require('web.test_utils');
+var ListRenderer = require('web.ListRenderer');
+var pyUtils = require('web.py_utils');
+
+const cpHelpers = testUtils.controlPanel;
+var createActionManager = testUtils.createActionManager;
+var createView = testUtils.createView;
+
+const patchDate = testUtils.mock.patchDate;
+
+QUnit.module('Dashboard', {
+ beforeEach: function () {
+ this.data = {
+ board: {
+ fields: {
+ },
+ records: [
+ ]
+ },
+ partner: {
+ fields: {
+ display_name: {string: "Displayed name", type: "char", searchable: true},
+ foo: {string: "Foo", type: "char", default: "My little Foo Value", searchable: true},
+ bar: {string: "Bar", type: "boolean"},
+ int_field: {string: "Integer field", type: "integer", group_operator: 'sum'},
+ },
+ records: [{
+ id: 1,
+ display_name: "first record",
+ foo: "yop",
+ int_field: 3,
+ }, {
+ id: 2,
+ display_name: "second record",
+ foo: "lalala",
+ int_field: 5,
+ }, {
+ id: 4,
+ display_name: "aaa",
+ foo: "abc",
+ int_field: 2,
+ }],
+ },
+ };
+ },
+});
+
+QUnit.test('dashboard basic rendering', async function (assert) {
+ assert.expect(4);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '</form>',
+ });
+
+ assert.doesNotHaveClass(form.renderer.$el, 'o_dashboard',
+ "should not have the o_dashboard css class");
+
+ form.destroy();
+
+ form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column></column>' +
+ '</board>' +
+ '</form>',
+ });
+
+ assert.hasClass(form.renderer.$el,'o_dashboard',
+ "with a dashboard, the renderer should have the proper css class");
+ assert.containsOnce(form, '.o_dashboard .o_view_nocontent',
+ "should have a no content helper");
+ assert.strictEqual(form.$('.o_control_panel .breadcrumb-item').text(), "My Dashboard",
+ "should have the correct title");
+ form.destroy();
+});
+
+QUnit.test('display the no content helper', async function (assert) {
+ assert.expect(1);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column></column>' +
+ '</board>' +
+ '</form>',
+ viewOptions: {
+ action: {
+ help: '<p class="hello">click to add a partner</p>'
+ }
+ },
+ });
+
+ assert.containsOnce(form, '.o_dashboard .o_view_nocontent',
+ "should have a no content helper with action help");
+ form.destroy();
+});
+
+QUnit.test('basic functionality, with one sub action', async function (assert) {
+ assert.expect(26);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{&quot;orderedBy&quot;: [{&quot;name&quot;: &quot;foo&quot;, &quot;asc&quot;: True}]}" view_mode="list" string="ABC" name="51" domain="[[\'foo\', \'!=\', \'False\']]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route, args) {
+ if (route === '/web/action/load') {
+ assert.step('load action');
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list']],
+ });
+ }
+ if (route === '/web/dataset/search_read') {
+ assert.deepEqual(args.domain, [['foo', '!=', 'False']], "the domain should be passed");
+ assert.deepEqual(args.context.orderedBy, [{
+ 'name': 'foo',
+ 'asc': true,
+ }],
+ 'orderedBy is present in the search read when specified on the custom action'
+ );
+ }
+ if (route === '/web/view/edit_custom') {
+ assert.step('edit custom');
+ return Promise.resolve(true);
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ },
+ });
+
+ assert.containsOnce(form, '.oe_dashboard_links',
+ "should have rendered a link div");
+ assert.containsOnce(form, 'table.oe_dashboard[data-layout="2-1"]',
+ "should have rendered a table");
+ assert.containsNone(form, 'td.o_list_record_selector',
+ "td should not have a list selector");
+ assert.strictEqual(form.$('h2 span.oe_header_txt:contains(ABC)').length, 1,
+ "should have rendered a header with action string");
+ assert.containsN(form, 'tr.o_data_row', 3,
+ "should have rendered 3 data rows");
+
+ assert.ok(form.$('.oe_content').is(':visible'), "content is visible");
+
+ await testUtils.dom.click(form.$('.oe_fold'));
+
+ assert.notOk(form.$('.oe_content').is(':visible'), "content is no longer visible");
+
+ await testUtils.dom.click(form.$('.oe_fold'));
+
+ assert.ok(form.$('.oe_content').is(':visible'), "content is visible again");
+ assert.verifySteps(['load action', 'edit custom', 'edit custom']);
+
+ assert.strictEqual($('.modal').length, 0, "should have no modal open");
+
+ await testUtils.dom.click(form.$('button.oe_dashboard_link_change_layout'));
+
+ assert.strictEqual($('.modal').length, 1, "should have opened a modal");
+ assert.strictEqual($('.modal li[data-layout="2-1"] i.oe_dashboard_selected_layout').length, 1,
+ "should mark currently selected layout");
+
+ await testUtils.dom.click($('.modal .oe_dashboard_layout_selector li[data-layout="1-1"]'));
+
+ assert.strictEqual($('.modal').length, 0, "should have no modal open");
+ assert.containsOnce(form, 'table.oe_dashboard[data-layout="1-1"]',
+ "should have rendered a table with correct layout");
+
+
+ assert.containsOnce(form, '.oe_action', "should have one displayed action");
+ await testUtils.dom.click(form.$('span.oe_close'));
+
+ assert.strictEqual($('.modal').length, 1, "should have opened a modal");
+
+ // confirm the close operation
+ await testUtils.dom.click($('.modal button.btn-primary'));
+
+ assert.strictEqual($('.modal').length, 0, "should have no modal open");
+ assert.containsNone(form, '.oe_action', "should have no displayed action");
+
+ assert.verifySteps(['edit custom', 'edit custom']);
+ form.destroy();
+});
+
+QUnit.test('views in the dashboard do not have a control panel', async function (assert) {
+ assert.expect(2);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form>' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list'], [5, 'form']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ },
+ });
+
+ assert.containsOnce(form, '.o_action .o_list_view');
+ assert.containsNone(form, '.o_action .o_control_panel');
+
+ form.destroy();
+});
+
+QUnit.test('can render an action without view_mode attribute', async function (assert) {
+ // The view_mode attribute is automatically set to the 'action' nodes when
+ // the action is added to the dashboard using the 'Add to dashboard' button
+ // in the searchview. However, other dashboard views can be written by hand
+ // (see openacademy tutorial), and in this case, we don't want hardcode
+ // action's params (like context or domain), as the dashboard can directly
+ // retrieve them from the action. Same applies for the view_type, as the
+ // first view of the action can be used, by default.
+ assert.expect(3);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action string="ABC" name="51" context="{\'a\': 1}"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ },
+ mockRPC: function (route, args) {
+ if (route === '/board/static/src/img/layout_1-1-1.png') {
+ return Promise.resolve();
+ }
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ context: '{"b": 2}',
+ domain: '[["foo", "=", "yop"]]',
+ res_model: 'partner',
+ views: [[4, 'list'], [false, 'form']],
+ });
+ }
+ if (args.method === 'load_views') {
+ assert.deepEqual(args.kwargs.context, {a: 1, b: 2},
+ "should have mixed both contexts");
+ }
+ if (route === '/web/dataset/search_read') {
+ assert.deepEqual(args.domain, [['foo', '=', 'yop']],
+ "should use the domain of the action");
+ }
+ return this._super.apply(this, arguments);
+ },
+ });
+
+ assert.strictEqual(form.$('.oe_action:contains(ABC) .o_list_view').length, 1,
+ "the list view (first view of action) should have been rendered correctly");
+
+ form.destroy();
+});
+
+QUnit.test('can sort a sub list', async function (assert) {
+ assert.expect(2);
+
+ this.data.partner.fields.foo.sortable = true;
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ },
+ });
+
+ assert.strictEqual($('tr.o_data_row').text(), 'yoplalalaabc',
+ "should have correct initial data");
+
+ await testUtils.dom.click(form.$('th.o_column_sortable:contains(Foo)'));
+
+ assert.strictEqual($('tr.o_data_row').text(), 'abclalalayop',
+ "data should have been sorted");
+ form.destroy();
+});
+
+QUnit.test('can open a record', async function (assert) {
+ assert.expect(1);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ },
+ intercepts: {
+ do_action: function (event) {
+ assert.deepEqual(event.data.action, {
+ res_id: 1,
+ res_model: 'partner',
+ type: 'ir.actions.act_window',
+ views: [[false, 'form']],
+ }, "should do a do_action with correct parameters");
+ },
+ },
+ });
+
+ await testUtils.dom.click(form.$('tr.o_data_row td:contains(yop)'));
+ form.destroy();
+});
+
+QUnit.test('can open record using action form view', async function (assert) {
+ assert.expect(1);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list'], [5, 'form']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ 'partner,5,form':
+ '<form string="Partner"><field name="display_name"/></form>',
+ },
+ intercepts: {
+ do_action: function (event) {
+ assert.deepEqual(event.data.action, {
+ res_id: 1,
+ res_model: 'partner',
+ type: 'ir.actions.act_window',
+ views: [[5, 'form']],
+ }, "should do a do_action with correct parameters");
+ },
+ },
+ });
+
+ await testUtils.dom.click(form.$('tr.o_data_row td:contains(yop)'));
+ form.destroy();
+});
+
+QUnit.test('can drag and drop a view', async function (assert) {
+ assert.expect(5);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list']],
+ });
+ }
+ if (route === '/web/view/edit_custom') {
+ assert.step('edit custom');
+ return Promise.resolve(true);
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ },
+ });
+
+ assert.containsOnce(form, 'td.index_0 .oe_action',
+ "initial action is in column 0");
+
+ await testUtils.dom.dragAndDrop(form.$('.oe_dashboard_column.index_0 .oe_header'),
+ form.$('.oe_dashboard_column.index_1'));
+ assert.containsNone(form, 'td.index_0 .oe_action',
+ "initial action is not in column 0");
+ assert.containsOnce(form, 'td.index_1 .oe_action',
+ "initial action is in in column 1");
+ assert.verifySteps(['edit custom']);
+
+ form.destroy();
+});
+
+QUnit.test('twice the same action in a dashboard', async function (assert) {
+ assert.expect(2);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '<action context="{}" view_mode="kanban" string="DEF" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list'],[5, 'kanban']],
+ });
+ }
+ if (route === '/web/view/edit_custom') {
+ assert.step('edit custom');
+ return Promise.resolve(true);
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<tree string="Partner"><field name="foo"/></tree>',
+ 'partner,5,kanban':
+ '<kanban><templates><t t-name="kanban-box">' +
+ '<div><field name="foo"/></div>' +
+ '</t></templates></kanban>',
+ },
+ });
+
+ var $firstAction = form.$('.oe_action:contains(ABC)');
+ assert.strictEqual($firstAction.find('.o_list_view').length, 1,
+ "list view should be displayed in 'ABC' block");
+ var $secondAction = form.$('.oe_action:contains(DEF)');
+ assert.strictEqual($secondAction.find('.o_kanban_view').length, 1,
+ "kanban view should be displayed in 'DEF' block");
+
+ form.destroy();
+});
+
+QUnit.test('non-existing action in a dashboard', async function (assert) {
+ assert.expect(1);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="kanban" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ intercepts: {
+ load_views: function () {
+ throw new Error('load_views should not be called');
+ }
+ },
+ mockRPC: function (route) {
+ if (route === '/board/static/src/img/layout_1-1-1.png') {
+ return Promise.resolve();
+ }
+ if (route === '/web/action/load') {
+ // server answer if the action doesn't exist anymore
+ return Promise.resolve(false);
+ }
+ return this._super.apply(this, arguments);
+ },
+ });
+
+ assert.strictEqual(form.$('.oe_action:contains(ABC)').length, 1,
+ "there should be a box for the non-existing action");
+
+ form.destroy();
+});
+
+QUnit.test('clicking on a kanban\'s button should trigger the action', async function (assert) {
+ assert.expect(2);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action name="149" string="Partner" view_mode="kanban" id="action_0_1"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ archs: {
+ 'partner,false,kanban':
+ '<kanban class="o_kanban_test"><templates><t t-name="kanban-box">' +
+ '<div>' +
+ '<field name="foo"/>' +
+ '</div>' +
+ '<div><button name="sitting_on_a_park_bench" type="object">Eying little girls with bad intent</button>' +
+ '</div>' +
+ '</t></templates></kanban>',
+ },
+ intercepts: {
+ execute_action: function (event) {
+ var data = event.data;
+ assert.strictEqual(data.env.model, 'partner', "should have correct model");
+ assert.strictEqual(data.action_data.name, 'sitting_on_a_park_bench',
+ "should call correct method");
+ }
+ },
+
+ mockRPC: function (route) {
+ if (route === '/board/static/src/img/layout_1-1-1.png') {
+ return Promise.resolve();
+ }
+ if (route === '/web/action/load') {
+ return Promise.resolve({res_model: 'partner', view_mode: 'kanban', views: [[false, 'kanban']]});
+ }
+ if (route === '/web/dataset/search_read') {
+ return Promise.resolve({records: [{foo: 'aqualung'}]});
+ }
+ return this._super.apply(this, arguments);
+ }
+ });
+
+ await testUtils.dom.click(form.$('.o_kanban_test').find('button:first'));
+
+ form.destroy();
+});
+
+QUnit.test('subviews are aware of attach in or detach from the DOM', async function (assert) {
+ assert.expect(2);
+
+ // patch list renderer `on_attach_callback` for the test only
+ testUtils.mock.patch(ListRenderer, {
+ on_attach_callback: function () {
+ assert.step('subview on_attach_callback');
+ }
+ });
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<list string="Partner"><field name="foo"/></list>',
+ },
+ });
+
+ assert.verifySteps(['subview on_attach_callback']);
+
+ // restore on_attach_callback of ListRenderer
+ testUtils.mock.unpatch(ListRenderer);
+
+ form.destroy();
+});
+
+QUnit.test('dashboard intercepts custom events triggered by sub controllers', async function (assert) {
+ assert.expect(1);
+
+ // we patch the ListController to force it to trigger the custom events that
+ // we want the dashboard to intercept (to stop them or to tweak their data)
+ testUtils.mock.patch(ListController, {
+ start: function () {
+ this.trigger_up('update_filters');
+ return this._super.apply(this, arguments);
+ },
+ });
+
+ var board = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({res_model: 'partner', views: [[false, 'list']]});
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,false,list': '<tree string="Partner"/>',
+ },
+ intercepts: {
+ update_filters: assert.step.bind(assert, 'update_filters'),
+ },
+ });
+
+ assert.verifySteps([]);
+
+ testUtils.mock.unpatch(ListController);
+ board.destroy();
+});
+
+QUnit.test('save actions to dashboard', async function (assert) {
+ assert.expect(6);
+
+ testUtils.patch(ListController, {
+ getOwnedQueryParams: function () {
+ var result = this._super.apply(this, arguments);
+ result.context = {
+ 'fire': 'on the bayou',
+ };
+ return result;
+ }
+ });
+
+ this.data['partner'].fields.foo.sortable = true;
+
+ var actionManager = await createActionManager({
+ data: this.data,
+ archs: {
+ 'partner,false,list': '<list><field name="foo"/></list>',
+ 'partner,false,search': '<search></search>',
+ },
+ mockRPC: function (route, args) {
+ if (route === '/board/add_to_dashboard') {
+ assert.deepEqual(args.context_to_save.group_by, ['foo'],
+ 'The group_by should have been saved');
+ assert.deepEqual(args.context_to_save.orderedBy,
+ [{
+ name: 'foo',
+ asc: true,
+ }],
+ 'The orderedBy should have been saved');
+ assert.strictEqual(args.context_to_save.fire, 'on the bayou',
+ 'The context of a controller should be passed and flattened');
+ assert.strictEqual(args.action_id, 1,
+ "should save the correct action");
+ assert.strictEqual(args.view_mode, 'list',
+ "should save the correct view type");
+ return Promise.resolve(true);
+ }
+ return this._super.apply(this, arguments);
+ }
+ });
+
+ await actionManager.doAction({
+ id: 1,
+ res_model: 'partner',
+ type: 'ir.actions.act_window',
+ views: [[false, 'list']],
+ });
+
+ assert.containsOnce(actionManager, '.o_list_view',
+ "should display the list view");
+
+ // Sort the list
+ await testUtils.dom.click($('.o_column_sortable'));
+
+ // Group It
+ await cpHelpers.toggleGroupByMenu(actionManager);
+ await cpHelpers.toggleAddCustomGroup(actionManager);
+ await cpHelpers.applyGroup(actionManager);
+
+ // add this action to dashboard
+ await cpHelpers.toggleFavoriteMenu(actionManager);
+
+ await testUtils.dom.click($('.o_add_to_board > button'));
+ await testUtils.fields.editInput($('.o_add_to_board input'), 'a name');
+ await testUtils.dom.click($('.o_add_to_board div button'));
+
+ testUtils.unpatch(ListController);
+
+ actionManager.destroy();
+});
+
+QUnit.test('save two searches to dashboard', async function (assert) {
+ // the second search saved should not be influenced by the first
+ assert.expect(2);
+
+ var actionManager = await createActionManager({
+ data: this.data,
+ archs: {
+ 'partner,false,list': '<list><field name="foo"/></list>',
+ 'partner,false,search': '<search></search>',
+ },
+ mockRPC: function (route, args) {
+ if (route === '/board/add_to_dashboard') {
+ if (filter_count === 0) {
+ assert.deepEqual(args.domain, [["display_name", "ilike", "a"]],
+ "the correct domain should be sent");
+ }
+ if (filter_count === 1) {
+ assert.deepEqual(args.domain, [["display_name", "ilike", "b"]],
+ "the correct domain should be sent");
+ }
+
+ filter_count += 1;
+ return Promise.resolve(true);
+ }
+ return this._super.apply(this, arguments);
+ },
+ });
+
+ await actionManager.doAction({
+ id: 1,
+ res_model: 'partner',
+ type: 'ir.actions.act_window',
+ views: [[false, 'list']],
+ });
+
+ var filter_count = 0;
+ // Add a first filter
+ await cpHelpers.toggleFilterMenu(actionManager);
+ await cpHelpers.toggleAddCustomFilter(actionManager);
+ await testUtils.fields.editInput(actionManager.el.querySelector('.o_generator_menu_value .o_input'), 'a');
+ await cpHelpers.applyFilter(actionManager);
+
+ // Add it to dashboard
+ await cpHelpers.toggleFavoriteMenu(actionManager);
+ await testUtils.dom.click($('.o_add_to_board > button'));
+ await testUtils.dom.click($('.o_add_to_board div button'));
+
+ // Remove it
+ await testUtils.dom.click(actionManager.el.querySelector('.o_facet_remove'));
+
+ // Add the second filter
+ await cpHelpers.toggleFilterMenu(actionManager);
+ await cpHelpers.toggleAddCustomFilter(actionManager);
+ await testUtils.fields.editInput(actionManager.el.querySelector('.o_generator_menu_value .o_input'), "b");
+ await cpHelpers.applyFilter(actionManager);
+ // Add it to dashboard
+ await cpHelpers.toggleFavoriteMenu(actionManager);
+ await testUtils.dom.click(actionManager.el.querySelector('.o_add_to_board > button'));
+ await testUtils.dom.click(actionManager.el.querySelector('.o_add_to_board div button'));
+
+ actionManager.destroy();
+});
+
+QUnit.test('save a action domain to dashboard', async function (assert) {
+ // View domains are to be added to the dashboard domain
+ assert.expect(1);
+
+ var view_domain = ["display_name", "ilike", "a"];
+ var filter_domain = ["display_name", "ilike", "b"];
+
+ // The filter domain already contains the view domain, but is always added by dashboard..,
+ var expected_domain = ['&', view_domain, '&', view_domain, filter_domain];
+
+ var actionManager = await createActionManager({
+ data: this.data,
+ archs: {
+ 'partner,false,list': '<list><field name="foo"/></list>',
+ 'partner,false,search': '<search></search>',
+ },
+ mockRPC: function (route, args) {
+ if (route === '/board/add_to_dashboard') {
+ assert.deepEqual(args.domain, expected_domain,
+ "the correct domain should be sent");
+ return Promise.resolve(true);
+ }
+ return this._super.apply(this, arguments);
+ },
+ });
+
+ await actionManager.doAction({
+ id: 1,
+ res_model: 'partner',
+ type: 'ir.actions.act_window',
+ views: [[false, 'list']],
+ domain: [view_domain],
+ });
+
+ // Add a filter
+ await cpHelpers.toggleFilterMenu(actionManager);
+ await cpHelpers.toggleAddCustomFilter(actionManager);
+ await testUtils.fields.editInput(
+ actionManager.el.querySelector('.o_generator_menu_value .o_input'),
+ "b"
+ );
+ await cpHelpers.applyFilter(actionManager);
+ // Add it to dashboard
+ await cpHelpers.toggleFavoriteMenu(actionManager);
+ await testUtils.dom.click(actionManager.el.querySelector('.o_add_to_board > button'));
+ // add
+ await testUtils.dom.click(actionManager.el.querySelector('.o_add_to_board div button'));
+
+ actionManager.destroy();
+});
+
+QUnit.test("Views should be loaded in the user's language", async function (assert) {
+ assert.expect(2);
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ session: {user_context: {lang: 'fr_FR'}},
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{\'lang\': \'en_US\'}" view_mode="list" string="ABC" name="51" domain="[]"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route, args) {
+ if (args.method === 'load_views') {
+ assert.deepEqual(pyUtils.eval('context', args.kwargs.context), {lang: 'fr_FR'},
+ 'The views should be loaded with the correct context');
+ }
+ if (route === "/web/dataset/search_read") {
+ assert.equal(args.context.lang, 'fr_FR',
+ 'The data should be loaded with the correct context');
+ }
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'list']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<list string="Partner"><field name="foo"/></list>',
+ },
+ });
+
+ form.destroy();
+});
+
+QUnit.test("Dashboard should use correct groupby", async function (assert) {
+ assert.expect(1);
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{\'group_by\': [\'bar\']}" string="ABC" name="51"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route, args) {
+ if (args.method === 'web_read_group') {
+ assert.deepEqual(args.kwargs.groupby, ['bar'],
+ 'user defined groupby should have precedence on action groupby');
+ }
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ context: {
+ group_by: 'some_field',
+ },
+ views: [[4, 'list']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<list string="Partner"><field name="foo"/></list>',
+ },
+ });
+
+ form.destroy();
+});
+
+QUnit.test("Dashboard should use correct groupby when defined as a string of one field", async function (assert) {
+ assert.expect(1);
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form string="My Dashboard">' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action context="{\'group_by\': \'bar\'}" string="ABC" name="51"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route, args) {
+ if (args.method === 'web_read_group') {
+ assert.deepEqual(args.kwargs.groupby, ['bar'],
+ 'user defined groupby should have precedence on action groupby');
+ }
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ context: {
+ group_by: 'some_field',
+ },
+ views: [[4, 'list']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,list':
+ '<list string="Partner"><field name="foo"/></list>',
+ },
+ });
+
+ form.destroy();
+});
+
+QUnit.test('click on a cell of pivot view inside dashboard', async function (assert) {
+ assert.expect(3);
+
+ var form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: '<form>' +
+ '<board style="2-1">' +
+ '<column>' +
+ '<action view_mode="pivot" string="ABC" name="51"></action>' +
+ '</column>' +
+ '</board>' +
+ '</form>',
+ mockRPC: function (route) {
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ res_model: 'partner',
+ views: [[4, 'pivot']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ archs: {
+ 'partner,4,pivot': '<pivot><field name="int_field" type="measure"/></pivot>',
+ },
+ intercepts: {
+ do_action: function () {
+ assert.step('do action');
+ },
+ },
+ });
+
+ assert.verifySteps([]);
+
+ await testUtils.dom.click(form.$('.o_pivot .o_pivot_cell_value'));
+
+ assert.verifySteps(['do action']);
+
+ form.destroy();
+});
+
+QUnit.test('correctly save the time ranges of a reporting view in comparison mode', async function (assert) {
+ assert.expect(1);
+
+ const unpatchDate = patchDate(2020, 6, 1, 11, 0, 0);
+
+ this.data.partner.fields.date = { string: 'Date', type: 'date', sortable: true };
+
+ const actionManager = await createActionManager({
+ data: this.data,
+ archs: {
+ 'partner,false,pivot': '<pivot><field name="foo"/></pivot>',
+ 'partner,false,search': '<search><filter name="Date" date="date"/></search>',
+ },
+ mockRPC: function (route, args) {
+ if (route === '/board/add_to_dashboard') {
+ assert.deepEqual(args.context_to_save.comparison, {
+ comparisonId: "previous_period",
+ fieldName: "date",
+ fieldDescription: "Date",
+ rangeDescription: "July 2020",
+ range: ["&",["date", ">=", "2020-07-01"], ["date", "<=", "2020-07-31"]],
+ comparisonRange: ["&", ["date", ">=", "2020-06-01"], ["date", "<=", "2020-06-30"]],
+ comparisonRangeDescription: "June 2020",
+ });
+ return Promise.resolve(true);
+ }
+ return this._super.apply(this, arguments);
+ },
+ });
+
+ await actionManager.doAction({
+ id: 1,
+ res_model: 'partner',
+ type: 'ir.actions.act_window',
+ views: [[false, 'pivot']],
+ });
+
+ // filter on July 2020
+ await cpHelpers.toggleFilterMenu(actionManager);
+ await cpHelpers.toggleMenuItem(actionManager, 'Date');
+ await cpHelpers.toggleMenuItemOption(actionManager, 'Date', 'July');
+
+ // compare July 2020 to June 2020
+ await cpHelpers.toggleComparisonMenu(actionManager);
+ await cpHelpers.toggleMenuItem(actionManager, 0);
+
+ // add the view to the dashboard
+ await cpHelpers.toggleFavoriteMenu(actionManager);
+
+ await testUtils.dom.click($('.o_add_to_board > button'));
+ await testUtils.fields.editInput($('.o_add_to_board input'), 'a name');
+ await testUtils.dom.click($('.o_add_to_board div button'));
+
+ unpatchDate();
+ actionManager.destroy();
+});
+
+QUnit.test('correctly display the time range descriptions of a reporting view in comparison mode', async function (assert) {
+ assert.expect(1);
+
+ this.data.partner.fields.date = { string: 'Date', type: 'date', sortable: true };
+ this.data.partner.records[0].date = '2020-07-15';
+
+ const form = await createView({
+ View: BoardView,
+ model: 'board',
+ data: this.data,
+ arch: `<form string="My Dashboard">
+ <board style="2-1">
+ <column>
+ <action string="ABC" name="51"></action>
+ </column>
+ </board>
+ </form>`,
+ archs: {
+ 'partner,1,pivot':
+ '<pivot string="Partner"></pivot>',
+ },
+ mockRPC: function (route, args) {
+ if (route === '/board/static/src/img/layout_1-1-1.png') {
+ return Promise.resolve();
+ }
+ if (route === '/web/action/load') {
+ return Promise.resolve({
+ context: JSON.stringify({ comparison: {
+ comparisonId: "previous_period",
+ fieldName: "date",
+ fieldDescription: "Date",
+ rangeDescription: "July 2020",
+ range: ["&",["date", ">=", "2020-07-01"], ["date", "<=", "2020-07-31"]],
+ comparisonRange: ["&", ["date", ">=", "2020-06-01"], ["date", "<=", "2020-06-30"]],
+ comparisonRangeDescription: "June 2020",
+ }}),
+ domain: '[]',
+ res_model: 'partner',
+ views: [[1, 'pivot']],
+ });
+ }
+ return this._super.apply(this, arguments);
+ },
+ });
+
+ assert.deepEqual(
+ [...form.el.querySelectorAll('div.o_pivot th.o_pivot_origin_row')].map(el => el.innerText),
+ ['June 2020', 'July 2020', 'Variation']
+ );
+
+ form.destroy();
+});
+});