odoo.define('web.graph_view_tests', function (require) { "use strict"; var searchUtils = require('web.searchUtils'); var GraphView = require('web.GraphView'); var testUtils = require('web.test_utils'); const { sortBy } = require('web.utils'); const cpHelpers = testUtils.controlPanel; var createView = testUtils.createView; var patchDate = testUtils.mock.patchDate; const { INTERVAL_OPTIONS, PERIOD_OPTIONS, COMPARISON_OPTIONS } = searchUtils; var INTERVAL_OPTION_IDS = Object.keys(INTERVAL_OPTIONS); const yearIds = []; const otherIds = []; for (const id of Object.keys(PERIOD_OPTIONS)) { const option = PERIOD_OPTIONS[id]; if (option.granularity === 'year') { yearIds.push(id); } else { otherIds.push(id); } } const BASIC_DOMAIN_IDS = []; for (const yearId of yearIds) { BASIC_DOMAIN_IDS.push(yearId); for (const id of otherIds) { BASIC_DOMAIN_IDS.push(`${yearId}__${id}`); } } const GENERATOR_INDEXES = {}; let index = 0; for (const id of Object.keys(PERIOD_OPTIONS)) { GENERATOR_INDEXES[id] = index++; } const COMPARISON_OPTION_IDS = Object.keys(COMPARISON_OPTIONS); const COMPARISON_OPTION_INDEXES = {}; index = 0; for (const comparisonOptionId of COMPARISON_OPTION_IDS) { COMPARISON_OPTION_INDEXES[comparisonOptionId] = index++; } var f = (a, b) => [].concat(...a.map(d => b.map(e => [].concat(d, e)))); var cartesian = (a, b, ...c) => (b ? cartesian(f(a, b), ...c) : a); var COMBINATIONS = cartesian(COMPARISON_OPTION_IDS, BASIC_DOMAIN_IDS); var COMBINATIONS_WITH_DATE = cartesian(COMPARISON_OPTION_IDS, BASIC_DOMAIN_IDS, INTERVAL_OPTION_IDS); QUnit.assert.checkDatasets = function (graph, keys, expectedDatasets) { keys = keys instanceof Array ? keys : [keys]; expectedDatasets = expectedDatasets instanceof Array ? expectedDatasets : [expectedDatasets]; var datasets = graph.renderer.chart.data.datasets; var actualValues = datasets.map(dataset => _.pick(dataset, keys)); this.pushResult({ result: _.isEqual(actualValues, expectedDatasets), actual: actualValues, expected: expectedDatasets, }); }; QUnit.assert.checkLabels = function (graph, expectedLabels) { var labels = graph.renderer.chart.data.labels; this.pushResult({ result: _.isEqual(labels, expectedLabels), actual: labels, expected: expectedLabels, }); }; QUnit.assert.checkLegend = function (graph, expectedLegendLabels) { expectedLegendLabels = expectedLegendLabels instanceof Array ? expectedLegendLabels : [expectedLegendLabels]; var chart = graph.renderer.chart; var actualLegendLabels = chart.config.options.legend.labels.generateLabels(chart).map(o => o.text); this.pushResult({ result: _.isEqual(actualLegendLabels, expectedLegendLabels), actual: actualLegendLabels, expected: expectedLegendLabels, }); }; QUnit.module('Views', { beforeEach: function () { this.data = { foo: { fields: { foo: {string: "Foo", type: "integer", store: true}, bar: {string: "bar", type: "boolean"}, product_id: {string: "Product", type: "many2one", relation: 'product', store: true}, color_id: {string: "Color", type: "many2one", relation: 'color'}, date: {string: "Date", type: 'date', store: true, sortable: true}, revenue: {string: "Revenue", type: 'integer', store: true}, }, records: [ {id: 1, foo: 3, bar: true, product_id: 37, date: "2016-01-01", revenue: 1}, {id: 2, foo: 53, bar: true, product_id: 37, color_id: 7, date: "2016-01-03", revenue: 2}, {id: 3, foo: 2, bar: true, product_id: 37, date: "2016-03-04", revenue: 3}, {id: 4, foo: 24, bar: false, product_id: 37, date: "2016-03-07", revenue: 4}, {id: 5, foo: 4, bar: false, product_id: 41, date: "2016-05-01", revenue: 5}, {id: 6, foo: 63, bar: false, product_id: 41}, {id: 7, foo: 42, bar: false, product_id: 41}, {id: 8, foo: 48, bar: false, product_id: 41, date: "2016-04-01", revenue: 8}, ] }, product: { fields: { name: {string: "Product Name", type: "char"} }, records: [{ id: 37, display_name: "xphone", }, { id: 41, display_name: "xpad", }] }, color: { fields: { name: {string: "Color", type: "char"} }, records: [{ id: 7, display_name: "red", }, { id: 14, display_name: "black", }] }, }; } }, function () { QUnit.module('GraphView'); QUnit.test('simple graph rendering', async function (assert) { assert.expect(5); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.containsOnce(graph, 'div.o_graph_canvas_container canvas', "should contain a div with a canvas element"); assert.strictEqual(graph.renderer.state.mode, "bar", "should be in bar chart mode by default"); assert.checkLabels(graph, [[true], [false]]); assert.checkDatasets(graph, ['backgroundColor', 'data', 'label', 'originIndex', 'stack'], { backgroundColor: "#1f77b4", data: [3,5], label: "Count", originIndex: 0, stack: "", } ); assert.checkLegend(graph, 'Count'); graph.destroy(); }); QUnit.test('default type attribute', async function (assert) { assert.expect(1); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.strictEqual(graph.renderer.state.mode, "pie", "should be in pie chart mode by default"); graph.destroy(); }); QUnit.test('title attribute', async function (assert) { assert.expect(1); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.strictEqual(graph.$('.o_graph_renderer label').text(), "Partners", "should have 'Partners as title'"); graph.destroy(); }); QUnit.test('field id not in groupBy', async function (assert) { assert.expect(1); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', mockRPC: function (route, args) { if (args.method === 'read_group') { assert.deepEqual(args.kwargs.groupby, [], 'groupby should not contain id field'); } return this._super.apply(this, arguments); }, }); graph.destroy(); }); QUnit.test('switching mode', async function (assert) { assert.expect(6); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.strictEqual(graph.renderer.state.mode, "line", "should be in line chart mode by default"); assert.doesNotHaveClass(graph.$buttons.find('button[data-mode="bar"]'), 'active', 'bar type button should not be active'); assert.hasClass(graph.$buttons.find('button[data-mode="line"]'),'active', 'line type button should be active'); await testUtils.dom.click(graph.$buttons.find('button[data-mode="bar"]')); assert.strictEqual(graph.renderer.state.mode, "bar", "should be in bar chart mode by default"); assert.doesNotHaveClass(graph.$buttons.find('button[data-mode="line"]'), 'active', 'line type button should not be active'); assert.hasClass(graph.$buttons.find('button[data-mode="bar"]'),'active', 'bar type button should be active'); graph.destroy(); }); QUnit.test('displaying line chart with only 1 data point', async function (assert) { assert.expect(1); // this test makes sure the line chart does not crash when only one data // point is displayed. this.data.foo.records = this.data.foo.records.slice(0,1); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.containsOnce(graph, 'canvas', "should have a canvas"); graph.destroy(); }); QUnit.test('displaying chart data with multiple groupbys', async function (assert) { // this test makes sure the line chart shows all data labels (X axis) when // it is grouped by several fields assert.expect(6); var graph = await createView({ View: GraphView, model: 'foo', data: this.data, arch: '', groupBy: ['product_id', 'bar', 'color_id'], }); assert.checkLabels(graph, [['xphone'], ['xpad']]); assert.checkLegend(graph, ['true/Undefined', 'true/red', 'false/Undefined']); await testUtils.dom.click(graph.$buttons.find('button[data-mode="line"]')); assert.checkLabels(graph, [['xphone'], ['xpad']]); assert.checkLegend(graph, ['true/Undefined', 'true/red', 'false/Undefined']); await testUtils.dom.click(graph.$buttons.find('button[data-mode="pie"]')); assert.checkLabels(graph, [ ["xphone", true, "Undefined"], ["xphone", true,"red"], ["xphone", false, "Undefined"], ["xpad", false, "Undefined"] ]); assert.checkLegend(graph, [ 'xphone/true/Undefined', 'xphone/true/red', 'xphone/false/Undefined', 'xpad/false/Undefined' ]); graph.destroy(); }); QUnit.test('switching measures', async function (assert) { assert.expect(2); var rpcCount = 0; var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', mockRPC: function (route, args) { rpcCount++; return this._super(route, args); }, }); await cpHelpers.toggleMenu(graph, "Measures"); await cpHelpers.toggleMenuItem(graph, "Foo"); assert.checkLegend(graph, 'Foo'); assert.strictEqual(rpcCount, 2, "should have done 2 rpcs (2 readgroups)"); graph.destroy(); }); QUnit.test('no content helper (bar chart)', async function (assert) { assert.expect(3); this.data.foo.records = []; var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, viewOptions: { action: { help: '

This helper should not be displayed in graph views

' } }, }); assert.containsOnce(graph, 'div.o_graph_canvas_container canvas'); assert.containsNone(graph, 'div.o_view_nocontent'); assert.containsNone(graph, '.abc'); graph.destroy(); }); QUnit.test('no content helper (pie chart)', async function (assert) { assert.expect(3); this.data.foo.records = []; var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, viewOptions: { action: { help: '

This helper should not be displayed in graph views

' } }, }); assert.containsOnce(graph, 'div.o_graph_canvas_container canvas'); assert.containsNone(graph, 'div.o_view_nocontent'); assert.containsNone(graph, '.abc'); graph.destroy(); }); QUnit.test('render pie chart in comparison mode', async function (assert) { assert.expect(2); const unpatchDate = patchDate(2020, 4, 19, 1, 0, 0); var graph = await createView({ View: GraphView, model: "foo", data: this.data, context: { search_default_date_filter: 1, }, arch: '' + '' + '', archs: { 'foo,false,search': ` `, }, }); await cpHelpers.toggleComparisonMenu(graph); await cpHelpers.toggleMenuItem(graph, 'Date: Previous period'); assert.containsNone(graph, 'div.o_view_nocontent', "should not display the no content helper"); assert.checkLegend(graph, 'No data'); unpatchDate(); graph.destroy(); }); QUnit.test('no content helper after update', async function (assert) { assert.expect(6); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, viewOptions: { action: { help: '

This helper should not be displayed in graph views

' } }, }); assert.containsOnce(graph, 'div.o_graph_canvas_container canvas'); assert.containsNone(graph, 'div.o_view_nocontent'); assert.containsNone(graph, '.abc'); await testUtils.graph.reload(graph, {domain: [['product_id', '<', 0]]}); assert.containsOnce(graph, 'div.o_graph_canvas_container canvas'); assert.containsNone(graph, 'div.o_view_nocontent'); assert.containsNone(graph, '.abc'); graph.destroy(); }); QUnit.test('can reload with other group by', async function (assert) { assert.expect(2); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.checkLabels(graph, [['xphone'], ['xpad']]); await testUtils.graph.reload(graph, {groupBy: ['color_id']}); assert.checkLabels(graph, [['Undefined'], ['red']]); graph.destroy(); }); QUnit.test('getOwnedQueryParams correctly returns mode, measure, and groupbys', async function (assert) { assert.expect(4); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.deepEqual(graph.getOwnedQueryParams(), { context: { graph_mode: 'bar', graph_measure: '__count__', graph_groupbys: ['product_id'], } }, "context should be correct"); await cpHelpers.toggleMenu(graph, "Measures"); await cpHelpers.toggleMenuItem(graph, "Foo"); assert.deepEqual(graph.getOwnedQueryParams(), { context: { graph_mode: 'bar', graph_measure: 'foo', graph_groupbys: ['product_id'], }, }, "context should be correct"); await testUtils.dom.click(graph.$buttons.find('button[data-mode="line"]')); assert.deepEqual(graph.getOwnedQueryParams(), { context: { graph_mode: 'line', graph_measure: 'foo', graph_groupbys: ['product_id'], }, }, "context should be correct"); await testUtils.graph.reload(graph, {groupBy: ['product_id', 'color_id']}); // change groupbys assert.deepEqual(graph.getOwnedQueryParams(), { context: { graph_mode: 'line', graph_measure: 'foo', graph_groupbys: ['product_id', 'color_id'], }, }, "context should be correct"); graph.destroy(); }); QUnit.test('correctly uses graph_ keys from the context', async function (assert) { assert.expect(5); var lastOne = _.last(this.data.foo.records); lastOne.color_id = 14; var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '', viewOptions: { context: { graph_measure: 'foo', graph_mode: 'line', graph_groupbys: ['color_id'], }, }, }); // check measure name is present in legend assert.checkLegend(graph, 'Foo'); // check mode assert.strictEqual(graph.renderer.state.mode, "line", "should be in line chart mode"); assert.doesNotHaveClass(graph.$buttons.find('button[data-mode="bar"]'), 'active', 'bar chart button should not be active'); assert.hasClass(graph.$buttons.find('button[data-mode="line"]'),'active', 'line chart button should be active'); // check groupby values ('Undefined' is rejected in line chart) are in labels assert.checkLabels(graph, [['red'], ['black']]); graph.destroy(); }); QUnit.test('correctly use group_by key from the context', async function (assert) { assert.expect(1); var lastOne = _.last(this.data.foo.records); lastOne.color_id = 14; var graph = await createView({ View: GraphView, model: 'foo', data: this.data, arch: '', groupBy: ['color_id'], viewOptions: { context: { graph_measure: 'foo', graph_mode: 'line', }, }, }); // check groupby values ('Undefined' is rejected in line chart) are in labels assert.checkLabels(graph, [['red'], ['black']]); graph.destroy(); }); QUnit.test('correctly uses graph_ keys from the context (at reload)', async function (assert) { assert.expect(7); var lastOne = _.last(this.data.foo.records); lastOne.color_id = 14; var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '', }); assert.strictEqual(graph.renderer.state.mode, "bar", "should be in bar chart mode"); assert.hasClass(graph.$buttons.find('button[data-mode="bar"]'),'active', 'bar chart button should be active'); var reloadParams = { context: { graph_measure: 'foo', graph_mode: 'line', graph_groupbys: ['color_id'], }, }; await testUtils.graph.reload(graph, reloadParams); // check measure assert.checkLegend(graph, 'Foo'); // check mode assert.strictEqual(graph.renderer.state.mode, "line", "should be in line chart mode"); assert.doesNotHaveClass(graph.$buttons.find('button[data-mode="bar"]'), 'active', 'bar chart button should not be active'); assert.hasClass(graph.$buttons.find('button[data-mode="line"]'),'active', 'line chart button should be active'); // check groupby values ('Undefined' is rejected in line chart) are in labels assert.checkLabels(graph, [['red'], ['black']]); graph.destroy(); }); QUnit.test('reload graph with correct fields', async function (assert) { assert.expect(2); var graph = await createView({ View: GraphView, model: 'foo', data: this.data, arch: '' + '' + '' + '', mockRPC: function (route, args) { if (args.method === 'read_group') { assert.deepEqual(args.kwargs.fields, ['product_id', 'foo'], "should read the correct fields"); } return this._super.apply(this, arguments); }, }); await testUtils.graph.reload(graph, {groupBy: []}); graph.destroy(); }); QUnit.test('initial groupby is kept when reloading', async function (assert) { assert.expect(8); var graph = await createView({ View: GraphView, model: 'foo', data: this.data, arch: '' + '' + '' + '', mockRPC: function (route, args) { if (args.method === 'read_group') { assert.deepEqual(args.kwargs.groupby, ['product_id'], "should group by the correct field"); } return this._super.apply(this, arguments); }, }); assert.checkLabels(graph, [['xphone'], ['xpad']]); assert.checkLegend(graph, 'Foo'); assert.checkDatasets(graph, 'data', {data: [82, 157]}); await testUtils.graph.reload(graph, {groupBy: []}); assert.checkLabels(graph, [['xphone'], ['xpad']]); assert.checkLegend(graph, 'Foo'); assert.checkDatasets(graph, 'data', {data: [82, 157]}); graph.destroy(); }); QUnit.test('use a many2one as a measure should work (without groupBy)', async function (assert) { assert.expect(4); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.containsOnce(graph, 'div.o_graph_canvas_container canvas', "should contain a div with a canvas element"); assert.checkLabels(graph, [[]]); assert.checkLegend(graph, 'Product'); assert.checkDatasets(graph, 'data', {data: [2]}); graph.destroy(); }); QUnit.test('use a many2one as a measure should work (with groupBy)', async function (assert) { assert.expect(5); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '' + '', }); assert.containsOnce(graph, 'div.o_graph_canvas_container canvas', "should contain a div with a canvas element"); assert.strictEqual(graph.renderer.state.mode, "bar", "should be in bar chart mode by default"); assert.checkLabels(graph, [[true], [false]]); assert.checkLegend(graph, 'Product'); assert.checkDatasets(graph, 'data', {data: [1, 2]}); graph.destroy(); }); QUnit.test('use a many2one as a measure and as a groupby should work', async function (assert) { assert.expect(3); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', viewOptions: { additionalMeasures: ['product_id'], }, }); // need to set the measure this way because it cannot be set in the // arch. await cpHelpers.toggleMenu(graph, "Measures"); await cpHelpers.toggleMenuItem(graph, "Product"); assert.checkLabels(graph, [['xphone'], ['xpad']]); assert.checkLegend(graph, 'Product'); assert.checkDatasets(graph, 'data', {data: [1, 1]}); graph.destroy(); }); QUnit.test('not use a many2one as a measure by default', async function (assert) { assert.expect(1); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.notOk(graph.measures.product_id, "should not have product_id as measure"); graph.destroy(); }); QUnit.test('use a many2one as a measure if set as additional fields', async function (assert) { assert.expect(1); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', viewOptions: { additionalMeasures: ['product_id'], }, }); assert.ok(graph.measures.find(m => m.fieldName === 'product_id'), "should have product_id as measure"); graph.destroy(); }); QUnit.test('measure dropdown consistency', async function (assert) { assert.expect(2); const actionManager = await testUtils.createActionManager({ archs: { 'foo,false,graph': ` `, 'foo,false,search': ``, 'foo,false,kanban': `
`, }, data: this.data, }); await actionManager.doAction({ res_model: 'foo', type: 'ir.actions.act_window', views: [[false, 'graph'], [false, 'kanban']], flags: { graph: { additionalMeasures: ['product_id'], } }, }); assert.containsOnce(actionManager, '.o_control_panel .o_graph_measures_list', "Measures dropdown is present at init" ); await cpHelpers.switchView(actionManager, 'kanban'); await cpHelpers.switchView(actionManager, 'graph'); assert.containsOnce(actionManager, '.o_control_panel .o_graph_measures_list', "Measures dropdown is present after reload" ); actionManager.destroy(); }); QUnit.test('graph view crash when moving from search view using Down key', async function (assert) { assert.expect(1); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); graph._giveFocus(); assert.ok(true,"should not generate any error"); graph.destroy(); }); QUnit.test('graph measures should be alphabetically sorted', async function (assert) { assert.expect(2); var data = this.data; data.foo.fields.bouh = {string: "bouh", type: "integer"}; var graph = await createView({ View: GraphView, model: "foo", data: data, arch: '' + '' + '' + '', }); await cpHelpers.toggleMenu(graph, "Measures"); assert.strictEqual(graph.$buttons.find('.o_graph_measures_list .dropdown-item:first').text(), 'bouh', "Bouh should be the first measure"); assert.strictEqual(graph.$buttons.find('.o_graph_measures_list .dropdown-item:last').text(), 'Count', "Count should be the last measure"); graph.destroy(); }); QUnit.test('Undefined should appear in bar, pie graph but not in line graph', async function (assert) { assert.expect(3); var graph = await createView({ View: GraphView, model: "foo", groupBy:['date'], data: this.data, arch: '' + '' + '', }); function _indexOf (label) { return graph.renderer._indexOf(graph.renderer.chart.data.labels, label); } assert.strictEqual(_indexOf(['Undefined']), -1); await testUtils.dom.click(graph.$buttons.find('.o_graph_button[data-mode=bar]')); assert.ok(_indexOf(['Undefined']) >= 0); await testUtils.dom.click(graph.$buttons.find('.o_graph_button[data-mode=pie]')); assert.ok(_indexOf(['Undefined']) >= 0); graph.destroy(); }); QUnit.test('Undefined should appear in bar, pie graph but not in line graph with multiple groupbys', async function (assert) { assert.expect(4); var graph = await createView({ View: GraphView, model: "foo", groupBy:['date', 'color_id'], data: this.data, arch: '' + '' + '', }); function _indexOf (label) { return graph.renderer._indexOf(graph.renderer.chart.data.labels, label); } assert.strictEqual(_indexOf(['Undefined']), -1); await testUtils.dom.click(graph.$buttons.find('.o_graph_button[data-mode=bar]')); assert.ok(_indexOf(['Undefined']) >= 0); await testUtils.dom.click(graph.$buttons.find('.o_graph_button[data-mode=pie]')); var labels = graph.renderer.chart.data.labels; assert.ok(labels.filter(label => /Undefined/.test(label.join(''))).length >= 1); // Undefined should not appear after switching back to line chart await testUtils.dom.click(graph.$buttons.find('.o_graph_button[data-mode=line]')); assert.strictEqual(_indexOf(['Undefined']), -1); graph.destroy(); }); QUnit.test('no comparison and no groupby', async function (assert) { assert.expect(9); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', }); assert.checkLabels(graph, [[]]); assert.checkLegend(graph, 'Foo'); assert.checkDatasets(graph, 'data', {data: [239]}); await testUtils.dom.click(graph.$('.o_graph_button[data-mode=line]')); // the labels in line chart is translated in this case to avoid to have a single // point at the left of the screen and chart to seem empty. assert.checkLabels(graph, [[''], [], ['']]); assert.checkLegend(graph, 'Foo'); assert.checkDatasets(graph, 'data', {data: [undefined, 239]}); await testUtils.dom.click(graph.$('.o_graph_button[data-mode=pie]')); assert.checkLabels(graph, [[]]); assert.checkLegend(graph, 'Total'); assert.checkDatasets(graph, 'data', {data: [239]}); graph.destroy(); }); QUnit.test('no comparison and one groupby', async function (assert) { assert.expect(9); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '' + '', }); assert.checkLabels(graph, [[true], [false]]); assert.checkLegend(graph, 'Foo'); assert.checkDatasets(graph, 'data', {data: [58, 181]}); await testUtils.dom.click(graph.$('.o_graph_button[data-mode=line]')); assert.checkLabels(graph, [[true], [false]]); assert.checkLegend(graph, 'Foo'); assert.checkDatasets(graph, 'data', {data: [58, 181]}); await testUtils.dom.click(graph.$('.o_graph_button[data-mode=pie]')); assert.checkLabels(graph, [[true], [false]]); assert.checkLegend(graph, ['true', 'false']); assert.checkDatasets(graph, 'data', {data: [58, 181]}); graph.destroy(); }); QUnit.test('no comparison and two groupby', async function (assert) { assert.expect(9); var graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '' + '' + '', groupBy: ['product_id', 'color_id'], }); assert.checkLabels(graph, [['xphone'], ['xpad']]); assert.checkLegend(graph, ['Undefined', 'red']); assert.checkDatasets(graph, ['label', 'data'], [ { label: 'Undefined', data: [29, 157], }, { label: 'red', data: [53, 0], } ]); await testUtils.dom.click(graph.$('.o_graph_button[data-mode=line]')); assert.checkLabels(graph, [['xphone'], ['xpad']]); assert.checkLegend(graph, ['Undefined', 'red']); assert.checkDatasets(graph, ['label', 'data'], [ { label: 'Undefined', data: [29, 157], }, { label: 'red', data: [53, 0], } ]); await testUtils.dom.click(graph.$('.o_graph_button[data-mode=pie]')); assert.checkLabels(graph, [['xphone', 'Undefined'], ['xphone', 'red'], ['xpad', 'Undefined']]); assert.checkLegend(graph, ['xphone/Undefined', 'xphone/red', 'xpad/Undefined']); assert.checkDatasets(graph, ['label', 'data'], { label: '', data: [29, 53, 157], }); graph.destroy(); }); QUnit.test('graph view only keeps finer groupby filter option for a given groupby', async function (assert) { assert.expect(3); var graph = await createView({ View: GraphView, model: "foo", groupBy:['date:year','product_id', 'date', 'date:quarter'], data: this.data, arch: '' + '' + '', }); assert.checkLabels(graph, [["January 2016"], ["March 2016"], ["May 2016"], ["April 2016"]]); // mockReadGroup does not always sort groups -> May 2016 is before April 2016 for that reason. assert.checkLegend(graph, ["xphone","xpad"]); assert.checkDatasets(graph, ['label', 'data'], [ { label: 'xphone', data: [2, 2, 0, 0], }, { label: 'xpad', data: [0, 0, 1, 1], } ]); graph.destroy(); }); QUnit.test('clicking on bar and pie charts triggers a do_action', async function (assert) { assert.expect(5); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '', intercepts: { do_action: function (ev) { assert.deepEqual(ev.data.action, { context: {}, domain: [["bar", "=", true]], name: "Foo Analysis", res_model: "foo", target: 'current', type: 'ir.actions.act_window', view_mode: 'list', views: [[false, 'list'], [false, 'form']], }, "should trigger do_action with correct action parameter"); } }, }); await testUtils.nextTick(); // wait for the graph to be rendered // bar mode assert.strictEqual(graph.renderer.state.mode, "bar", "should be in bar chart mode"); assert.checkDatasets(graph, ['domain'], { domain: [[["bar", "=", true]], [["bar", "=", false]]], }); let myChart = graph.renderer.chart; let meta = myChart.getDatasetMeta(0); let rectangle = myChart.canvas.getBoundingClientRect(); let point = meta.data[0].getCenterPoint(); await testUtils.dom.triggerEvent(myChart.canvas, 'click', { pageX: rectangle.left + point.x, pageY: rectangle.top + point.y }); // pie mode await testUtils.dom.click(graph.$('.o_graph_button[data-mode=pie]')); assert.strictEqual(graph.renderer.state.mode, "pie", "should be in pie chart mode"); myChart = graph.renderer.chart; meta = myChart.getDatasetMeta(0); rectangle = myChart.canvas.getBoundingClientRect(); point = meta.data[0].getCenterPoint(); await testUtils.dom.triggerEvent(myChart.canvas, 'click', { pageX: rectangle.left + point.x, pageY: rectangle.top + point.y }); graph.destroy(); }); QUnit.test('clicking charts trigger a do_action with correct views', async function (assert) { assert.expect(3); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '', intercepts: { do_action: function (ev) { assert.deepEqual(ev.data.action, { context: {}, domain: [["bar", "=", true]], name: "Foo Analysis", res_model: "foo", target: 'current', type: 'ir.actions.act_window', view_mode: 'list', views: [[364, 'list'], [29, 'form']], }, "should trigger do_action with correct action parameter"); } }, viewOptions: { actionViews: [{ type: 'list', viewID: 364, }, { type: 'form', viewID: 29, }], }, }); await testUtils.nextTick(); // wait for the graph to be rendered assert.strictEqual(graph.renderer.state.mode, "bar", "should be in bar chart mode"); assert.checkDatasets(graph, ['domain'], { domain: [[["bar", "=", true]], [["bar", "=", false]]], }); let myChart = graph.renderer.chart; let meta = myChart.getDatasetMeta(0); let rectangle = myChart.canvas.getBoundingClientRect(); let point = meta.data[0].getCenterPoint(); await testUtils.dom.triggerEvent(myChart.canvas, 'click', { pageX: rectangle.left + point.x, pageY: rectangle.top + point.y }); graph.destroy(); }); QUnit.test('graph view with attribute disable_linking="True"', async function (assert) { assert.expect(2); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: '', intercepts: { do_action: function () { throw new Error('Should not perform a do_action'); }, }, }); await testUtils.nextTick(); // wait for the graph to be rendered assert.strictEqual(graph.renderer.state.mode, "bar", "should be in bar chart mode"); assert.checkDatasets(graph, ['domain'], { domain: [[["bar", "=", true]], [["bar", "=", false]]], }); let myChart = graph.renderer.chart; let meta = myChart.getDatasetMeta(0); let rectangle = myChart.canvas.getBoundingClientRect(); let point = meta.data[0].getCenterPoint(); await testUtils.dom.triggerEvent(myChart.canvas, 'click', { pageX: rectangle.left + point.x, pageY: rectangle.top + point.y }); graph.destroy(); }); QUnit.test('graph view without invisible attribute on field', async function (assert) { assert.expect(4); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ``, }); await testUtils.dom.click(graph.$('.btn-group:first button')); assert.containsN(graph, 'li.o_menu_item', 3, "there should be three menu item in the measures dropdown (count, revenue and foo)"); assert.containsOnce(graph, 'li.o_menu_item a:contains("Revenue")'); assert.containsOnce(graph, 'li.o_menu_item a:contains("Foo")'); assert.containsOnce(graph, 'li.o_menu_item a:contains("Count")'); graph.destroy(); }); QUnit.test('graph view with invisible attribute on field', async function (assert) { assert.expect(2); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, }); await testUtils.dom.click(graph.$('.btn-group:first button')); assert.containsN(graph, 'li.o_menu_item', 2, "there should be only two menu item in the measures dropdown (count and foo)"); assert.containsNone(graph, 'li.o_menu_item a:contains("Revenue")'); graph.destroy(); }); QUnit.test('graph view sort by measure', async function (assert) { assert.expect(18); // change first record from foo as there are 4 records count for each product this.data.product.records.push({ id: 38, display_name: "zphone"}); this.data.foo.records[7].product_id = 38; const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, }); assert.containsN(graph, 'button[data-order]', 2, "there should be two order buttons for sorting axis labels in bar mode"); assert.checkLegend(graph, 'Count', 'measure should be by count'); assert.hasClass(graph.$('button[data-order="desc"]'), 'active', 'sorting should be applie on descending order by default when sorting="desc"'); assert.checkDatasets(graph, 'data', {data: [4, 3, 1]}); await testUtils.dom.click(graph.$buttons.find('button[data-order="asc"]')); assert.hasClass(graph.$('button[data-order="asc"]'), 'active', "ascending order should be applied"); assert.checkDatasets(graph, 'data', {data: [1, 3, 4]}); await testUtils.dom.click(graph.$buttons.find('button[data-order="desc"]')); assert.hasClass(graph.$('button[data-order="desc"]'), 'active', "descending order button should be active"); assert.checkDatasets(graph, 'data', { data: [4, 3, 1] }); // again click on descending button to deactivate order button await testUtils.dom.click(graph.$buttons.find('button[data-order="desc"]')); assert.doesNotHaveClass(graph.$('button[data-order="desc"]'), 'active', "descending order button should not be active"); assert.checkDatasets(graph, 'data', {data: [4, 3, 1]}); // set line mode await testUtils.dom.click(graph.$buttons.find('button[data-mode="line"]')); assert.containsN(graph, 'button[data-order]', 2, "there should be two order buttons for sorting axis labels in line mode"); assert.checkLegend(graph, 'Count', 'measure should be by count'); assert.doesNotHaveClass(graph.$('button[data-order="desc"]'), 'active', "descending order should be applied"); assert.checkDatasets(graph, 'data', {data: [4, 3, 1]}); await testUtils.dom.click(graph.$buttons.find('button[data-order="asc"]')); assert.hasClass(graph.$('button[data-order="asc"]'), 'active', "ascending order button should be active"); assert.checkDatasets(graph, 'data', { data: [1, 3, 4] }); await testUtils.dom.click(graph.$buttons.find('button[data-order="desc"]')); assert.hasClass(graph.$('button[data-order="desc"]'), 'active', "descending order button should be active"); assert.checkDatasets(graph, 'data', { data: [4, 3, 1] }); graph.destroy(); }); QUnit.test('graph view sort by measure for grouped data', async function (assert) { assert.expect(9); // change first record from foo as there are 4 records count for each product this.data.product.records.push({ id: 38, display_name: "zphone", }); this.data.foo.records[7].product_id = 38; const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, }); assert.checkLegend(graph, ["true","false"], 'measure should be by count'); assert.containsN(graph, 'button[data-order]', 2, "there should be two order buttons for sorting axis labels"); assert.checkDatasets(graph, 'data', [{data: [3, 0, 0]}, {data: [1, 3, 1]}]); await testUtils.dom.click(graph.$buttons.find('button[data-order="asc"]')); assert.hasClass(graph.$('button[data-order="asc"]'), 'active', "ascending order should be applied by default"); assert.checkDatasets(graph, 'data', [{ data: [1, 3, 1] }, { data: [0, 0, 3] }]); await testUtils.dom.click(graph.$buttons.find('button[data-order="desc"]')); assert.hasClass(graph.$('button[data-order="desc"]'), 'active', "ascending order button should be active"); assert.checkDatasets(graph, 'data', [{data: [1, 3, 1]}, {data: [3, 0, 0]}]); // again click on descending button to deactivate order button await testUtils.dom.click(graph.$buttons.find('button[data-order="desc"]')); assert.doesNotHaveClass(graph.$('button[data-order="desc"]'), 'active', "descending order button should not be active"); assert.checkDatasets(graph, 'data', [{ data: [3, 0, 0] }, { data: [1, 3, 1] }]); graph.destroy(); }); QUnit.test('graph view sort by measure for multiple grouped data', async function (assert) { assert.expect(9); // change first record from foo as there are 4 records count for each product this.data.product.records.push({ id: 38, display_name: "zphone" }); this.data.foo.records[7].product_id = 38; // add few more records to data to have grouped data date wise const data = [ {id: 9, foo: 48, bar: false, product_id: 41, date: "2016-04-01"}, {id: 10, foo: 49, bar: false, product_id: 41, date: "2016-04-01"}, {id: 11, foo: 50, bar: true, product_id: 37, date: "2016-01-03"}, {id: 12, foo: 50, bar: true, product_id: 41, date: "2016-01-03"}, ]; Object.assign(this.data.foo.records, data); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, groupBy: ['date', 'product_id'] }); assert.checkLegend(graph, ["xpad","xphone","zphone"], 'measure should be by count'); assert.containsN(graph, 'button[data-order]', 2, "there should be two order buttons for sorting axis labels"); assert.checkDatasets(graph, 'data', [{data: [2, 1, 1, 2]}, {data: [0, 1, 0, 0]}, {data: [1, 0, 0, 0]}]); await testUtils.dom.click(graph.$buttons.find('button[data-order="asc"]')); assert.hasClass(graph.$('button[data-order="asc"]'), 'active', "ascending order should be applied by default"); assert.checkDatasets(graph, 'data', [{ data: [1, 1, 2, 2] }, { data: [0, 1, 0, 0] }, { data: [0, 0, 0, 1] }]); await testUtils.dom.click(graph.$buttons.find('button[data-order="desc"]')); assert.hasClass(graph.$('button[data-order="desc"]'), 'active', "descending order button should be active"); assert.checkDatasets(graph, 'data', [{data: [1, 0, 0, 0]}, {data: [2, 2, 1, 1]}, {data: [0, 0, 1, 0]}]); // again click on descending button to deactivate order button await testUtils.dom.click(graph.$buttons.find('button[data-order="desc"]')); assert.doesNotHaveClass(graph.$('button[data-order="desc"]'), 'active', "descending order button should not be active"); assert.checkDatasets(graph, 'data', [{ data: [2, 1, 1, 2] }, { data: [0, 1, 0, 0] }, { data: [1, 0, 0, 0] }]); graph.destroy(); }); QUnit.test('empty graph view with sample data', async function (assert) { assert.expect(8); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, domain: [['id', '<', 0]], viewOptions: { action: { help: '

click to add a foo

' } }, }); assert.hasClass(graph.el, 'o_view_sample_data'); assert.containsOnce(graph, '.o_view_nocontent'); assert.containsOnce(graph, '.o_graph_canvas_container canvas'); assert.hasClass(graph.$('.o_graph_canvas_container'), 'o_sample_data_disabled'); await graph.reload({ domain: [] }); assert.doesNotHaveClass(graph.el, 'o_view_sample_data'); assert.containsNone(graph, '.o_view_nocontent'); assert.containsOnce(graph, '.o_graph_canvas_container canvas'); assert.doesNotHaveClass(graph.$('.o_graph_canvas_container'), 'o_sample_data_disabled'); graph.destroy(); }); QUnit.test('non empty graph view with sample data', async function (assert) { assert.expect(8); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, viewOptions: { action: { help: '

click to add a foo

' } }, }) assert.doesNotHaveClass(graph.el, 'o_view_sample_data'); assert.containsNone(graph, '.o_view_nocontent'); assert.containsOnce(graph, '.o_graph_canvas_container canvas'); assert.doesNotHaveClass(graph.$('.o_graph_canvas_container'), 'o_sample_data_disabled'); await graph.reload({ domain: [['id', '<', 0]] }); assert.doesNotHaveClass(graph.el, 'o_view_sample_data'); assert.containsOnce(graph, '.o_graph_canvas_container canvas'); assert.doesNotHaveClass(graph.$('.o_graph_canvas_container'), 'o_sample_data_disabled'); assert.containsNone(graph, '.o_view_nocontent'); graph.destroy(); }); QUnit.module('GraphView: comparison mode', { beforeEach: async function () { this.data.foo.records[0].date = '2016-12-15'; this.data.foo.records[1].date = '2016-12-17'; this.data.foo.records[2].date = '2016-11-22'; this.data.foo.records[3].date = '2016-11-03'; this.data.foo.records[4].date = '2016-12-20'; this.data.foo.records[5].date = '2016-12-19'; this.data.foo.records[6].date = '2016-12-15'; this.data.foo.records[7].date = undefined; this.data.foo.records.push({id: 9, foo: 48, bar: false, product_id: 41, color_id: 7, date: "2016-12-01"}); this.data.foo.records.push({id: 10, foo: 17, bar: true, product_id: 41, color_id: 7, date: "2016-12-01"}); this.data.foo.records.push({id: 11, foo: 45, bar: true, product_id: 37, color_id: 14, date: "2016-12-01"}); this.data.foo.records.push({id: 12, foo: 48, bar: false, product_id: 37, color_id: 7, date: "2016-12-10"}); this.data.foo.records.push({id: 13, foo: 48, bar: false, product_id: undefined, color_id: 14, date: "2016-11-30"}); this.data.foo.records.push({id: 14, foo: -50, bar: true, product_id: 41, color_id: 14, date: "2016-12-15"}); this.data.foo.records.push({id: 15, foo: 53, bar: false, product_id: 41, color_id: 14, date: "2016-11-01"}); this.data.foo.records.push({id: 16, foo: 53, bar: true, product_id: undefined, color_id: 7, date: "2016-09-01"}); this.data.foo.records.push({id: 17, foo: 48, bar: false, product_id: 41, color_id: undefined, date: "2016-08-01"}); this.data.foo.records.push({id: 18, foo: -156, bar: false, product_id: 37, color_id: undefined, date: "2016-07-15"}); this.data.foo.records.push({id: 19, foo: 31, bar: false, product_id: 41, color_id: 14, date: "2016-12-15"}); this.data.foo.records.push({id: 20, foo: 109, bar: true, product_id: 41, color_id: 7, date: "2015-06-01"}); this.data.foo.records = sortBy(this.data.foo.records, 'date'); this.unpatchDate = patchDate(2016, 11, 20, 1, 0, 0); const graph = await createView({ View: GraphView, model: "foo", data: this.data, arch: ` `, archs: { 'foo,false,search': ` `, }, viewOptions: { additionalMeasures: ['product_id'], }, }); this.graph = graph; var checkOnlyToCheck = true; var exhaustiveTest = false || checkOnlyToCheck; var self = this; async function* graphGenerator(combinations) { var i = 0; while (i < combinations.length) { var combination = combinations[i]; if (!checkOnlyToCheck || combination.toString() in self.combinationsToCheck) { await self.setConfig(combination); } if (exhaustiveTest) { i++; } else { i += Math.floor(1 + Math.random() * 20); } yield combination; } } this.combinationsToCheck = {}; this.testCombinations = async function (combinations, assert) { for await (var combination of graphGenerator(combinations)) { // we can check particular combinations here if (combination.toString() in self.combinationsToCheck) { if (self.combinationsToCheck[combination].errorMessage) { assert.strictEqual( graph.$('.o_nocontent_help p').eq(1).text().trim(), self.combinationsToCheck[combination].errorMessage ); } else { assert.checkLabels(graph, self.combinationsToCheck[combination].labels); assert.checkLegend(graph, self.combinationsToCheck[combination].legend); assert.checkDatasets(graph, ['label', 'data'], self.combinationsToCheck[combination].datasets); } } } }; const GROUPBY_NAMES = ['Date', 'Bar', 'Product', 'Color']; this.selectTimeRanges = async function (comparisonOptionId, basicDomainId) { const facetEls = graph.el.querySelectorAll('.o_searchview_facet'); const facetIndex = [...facetEls].findIndex(el => !!el.querySelector('span.fa-filter')); if (facetIndex > -1) { await cpHelpers.removeFacet(graph, facetIndex); } const [yearId, otherId] = basicDomainId.split('__'); await cpHelpers.toggleFilterMenu(graph); await cpHelpers.toggleMenuItem(graph, 'Date Filter'); await cpHelpers.toggleMenuItemOption(graph, 'Date Filter', GENERATOR_INDEXES[yearId]); if (otherId) { await cpHelpers.toggleMenuItemOption(graph, 'Date Filter', GENERATOR_INDEXES[otherId]); } const itemIndex = COMPARISON_OPTION_INDEXES[comparisonOptionId]; await cpHelpers.toggleComparisonMenu(graph); await cpHelpers.toggleMenuItem(graph, itemIndex); }; // groupby menu is assumed to be closed this.selectDateIntervalOption = async function (intervalOption) { intervalOption = intervalOption || 'month'; const optionIndex = INTERVAL_OPTION_IDS.indexOf(intervalOption); await cpHelpers.toggleGroupByMenu(graph); let wasSelected = false; if (this.keepFirst) { if (cpHelpers.isItemSelected(graph, 2)) { wasSelected = true; await cpHelpers.toggleMenuItem(graph, 2); } } await cpHelpers.toggleMenuItem(graph, 0); if (!cpHelpers.isOptionSelected(graph, 0, optionIndex)) { await cpHelpers.toggleMenuItemOption(graph, 0, optionIndex); } for (let i = 0; i < INTERVAL_OPTION_IDS.length; i++) { const oId = INTERVAL_OPTION_IDS[i]; if (oId !== intervalOption && cpHelpers.isOptionSelected(graph, 0, i)) { await cpHelpers.toggleMenuItemOption(graph, 0, i); } } if (this.keepFirst) { if (wasSelected && !cpHelpers.isItemSelected(graph, 2)) { await cpHelpers.toggleMenuItem(graph, 2); } } await cpHelpers.toggleGroupByMenu(graph); }; // groupby menu is assumed to be closed this.selectGroupBy = async function (groupByName) { await cpHelpers.toggleGroupByMenu(graph); const index = GROUPBY_NAMES.indexOf(groupByName); if (!cpHelpers.isItemSelected(graph, index)) { await cpHelpers.toggleMenuItem(graph, index); } await cpHelpers.toggleGroupByMenu(graph); }; this.setConfig = async function (combination) { await this.selectTimeRanges(combination[0], combination[1]); if (combination.length === 3) { await self.selectDateIntervalOption(combination[2]); } }; this.setMode = async function (mode) { await testUtils.dom.click($(`.o_control_panel .o_graph_button[data-mode="${mode}"]`)); }; }, afterEach: function () { this.unpatchDate(); this.graph.destroy(); }, }, function () { QUnit.test('comparison with one groupby equal to comparison date field', async function (assert) { assert.expect(11); this.combinationsToCheck = { 'previous_period,this_year__this_month,day': { labels: [...Array(6).keys()].map(x => [x]), legend: ["December 2016", "November 2016"], datasets: [ { data: [110, 48, 26, 53, 63, 4], label: "December 2016", }, { data: [53, 24, 2, 48], label: "November 2016", } ], } }; var combinations = COMBINATIONS_WITH_DATE; await this.testCombinations(combinations, assert); await this.setMode('line'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year__this_month,day'] = { labels: [...Array(6).keys()].map(x => [x]), legend: [ "2016-12-01,2016-11-01", "2016-12-10,2016-11-03", "2016-12-15,2016-11-22", "2016-12-17,2016-11-30", "2016-12-19", "2016-12-20" ], datasets: [ { data: [ 110, 48, 26, 53, 63, 4], label: "December 2016", }, { data: [ 53, 24, 2, 48, 0, 0], label: "November 2016", } ], }; await this.setMode('pie'); await this.testCombinations(combinations, assert); // isNotVisible can not have two elements so checking visibility of first element assert.isNotVisible(this.graph.$('button[data-order]:first'), "there should not be order button in comparison mode"); assert.ok(true, "No combination causes a crash"); }); QUnit.test('comparison with no groupby', async function (assert) { assert.expect(10); this.combinationsToCheck = { 'previous_period,this_year__this_month': { labels: [[]], legend: ["December 2016", "November 2016"], datasets: [ { data: [304], label: "December 2016", }, { data: [127], label: "November 2016", } ], } }; var combinations = COMBINATIONS; await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year__this_month'] = { labels: [[''], [], ['']], legend: ["December 2016", "November 2016"], datasets: [ { data: [undefined, 304], label: "December 2016", }, { data: [undefined, 127], label: "November 2016", } ], }; await this.setMode('line'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year__this_month'] = { labels: [[]], legend: ["Total"], datasets: [ { data: [304], label: "December 2016", }, { data: [127], label: "November 2016", } ], }; await this.setMode('pie'); await this.testCombinations(combinations, assert); assert.ok(true, "No combination causes a crash"); }); QUnit.test('comparison with one groupby different from comparison date field', async function (assert) { assert.expect(10); this.combinationsToCheck = { 'previous_period,this_year__this_month': { labels: [["xpad"], ["xphone"],["Undefined"]], legend: ["December 2016", "November 2016"], datasets: [ { data: [ 155, 149, 0], label: "December 2016", }, { data: [ 53, 26, 48], label: "November 2016", } ], } }; var combinations = COMBINATIONS; await this.selectGroupBy('Product'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year__this_month'] = { labels: [["xpad"], ["xphone"]], legend: ["December 2016", "November 2016"], datasets: [ { data: [155, 149], label: "December 2016", }, { data: [53, 26], label: "November 2016", } ], }; await this.setMode('line'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year__this_month'] = { labels: [["xpad"], ["xphone"], ["Undefined"]], legend: ["xpad", "xphone", "Undefined"], datasets: [ { data: [ 155, 149, 0], label: "December 2016", }, { data: [ 53, 26, 48], label: "November 2016", } ], }; await this.setMode('pie'); await this.testCombinations(combinations, assert); assert.ok(true, "No combination causes a crash"); }); QUnit.test('comparison with two groupby with first groupby equal to comparison date field', async function (assert) { assert.expect(10); this.keepFirst = true; this.combinationsToCheck = { 'previous_period,this_year__this_month,day': { labels: [...Array(6).keys()].map(x => [x]), legend: [ "December 2016/xpad", "December 2016/xphone", "November 2016/xpad", "November 2016/xphone", "November 2016/Undefined" ], datasets: [ { data: [ 65, 0, 23, 0, 63, 4], label: "December 2016/xpad" }, { data: [ 45, 48, 3, 53, 0, 0], label: "December 2016/xphone" }, { data: [ 53, 0, 0, 0], label: "November 2016/xpad" }, { data: [ 0, 24, 2, 0], label: "November 2016/xphone" }, { data: [ 0, 0, 0, 48], label: "November 2016/Undefined" } ] } }; var combinations = COMBINATIONS_WITH_DATE; await this.selectGroupBy('Product'); await this.testCombinations(combinations, assert); await this.setMode('line'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year__this_month,day'] = { labels: [[0, "xpad"], [0, "xphone"], [1, "xphone"], [2, "xphone"], [2, "xpad"], [3, "xphone"], [4, "xpad"], [5, "xpad"], [3, "Undefined"]], legend: [ "2016-12-01,2016-11-01/xpad", "2016-12-01,2016-11-01/xphone", "2016-12-10,2016-11-03/xphone", "2016-12-15,2016-11-22/xphone", "2016-12-15,2016-11-22/xpad", "2016-12-17,2016-11-30/xphone", "2016-12-19/xpad", "2016-12-20/xpad", "2016-12-17,2016-11-30/Undefine..." ], datasets: [ { "data": [ 65, 45, 48, 3, 23, 53, 63, 4, 0], "label": "December 2016" }, { "data": [ 53, 0, 24, 2, 0, 0, 0, 0, 48], "label": "November 2016" } ], }; await this.setMode('pie'); await this.testCombinations(combinations, assert); assert.ok(true, "No combination causes a crash"); this.keepFirst = false; }); QUnit.test('comparison with two groupby with second groupby equal to comparison date field', async function (assert) { assert.expect(8); this.combinationsToCheck = { 'previous_period,this_year,quarter': { labels: [["xphone"], ["xpad"],["Undefined"]], legend: [ "2016/Q3 2016", "2016/Q4 2016", "2015/Q2 2015" ], datasets: [ { data: [-156, 48, 53], label: "2016/Q3 2016", }, { data: [175, 208, 48], label: "2016/Q4 2016", }, { data: [0, 109, 0], label: "2015/Q2 2015", }, ] } }; const combinations = COMBINATIONS_WITH_DATE; await this.selectGroupBy('Product'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year,quarter'] = { labels: [["xphone"], ["xpad"]], legend: [ "2016/Q3 2016", "2016/Q4 2016", "2015/Q2 2015" ], datasets: [ { data: [-156, 48], label: "2016/Q3 2016", }, { data: [175, 208], label: "2016/Q4 2016", }, { data: [0, 109], label: "2015/Q2 2015", }, ] }; await this.setMode('line'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_period,this_year,quarter'] = { errorMessage: 'Pie chart cannot mix positive and negative numbers. ' + 'Try to change your domain to only display positive results' }; await this.setMode('pie'); await this.testCombinations(combinations, assert); assert.ok(true, "No combination causes a crash"); }); QUnit.test('comparison with two groupby with no groupby equal to comparison date field', async function (assert) { assert.expect(10); this.combinationsToCheck = { 'previous_year,this_year__last_month': { labels: [["xpad"], ["xphone"],["Undefined"] ], legend: ["November 2016/false", "November 2016/true"], datasets: [ { data: [53, 24, 48], label: "November 2016/false", }, { data: [0, 2, 0], label: "November 2016/true", } ], } }; var combinations = COMBINATIONS; await this.selectGroupBy('Product'); await this.selectGroupBy('Bar'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_year,this_year__last_month'] = { labels: [["xpad"], ["xphone"] ], legend: ["November 2016/false", "November 2016/true"], datasets: [ { data: [53, 24], label: "November 2016/false", }, { data: [0, 2], label: "November 2016/true", } ], }; await this.setMode('line'); await this.testCombinations(combinations, assert); this.combinationsToCheck['previous_year,this_year__last_month'] = { labels: [["xpad", false], ["xphone", false], ["xphone", true], ["Undefined", false], ["No data"]], legend: [ "xpad/false", "xphone/false", "xphone/true", "Undefined/false", "No data" ], datasets: [ { "data": [ 53, 24, 2, 48], "label": "November 2016" }, { "data": [ undefined, undefined, undefined, undefined, 1], "label": "November 2015" } ], }; await this.setMode('pie'); await this.testCombinations(combinations, assert); assert.ok(true, "No combination causes a crash"); }); }); }); });