import mapValues = require('lodash/mapValues'); import { assert, driver, Key } from 'mocha-webdriver'; import * as gu from 'test/nbrowser/gristUtils'; import { server, setupTestSuite } from 'test/nbrowser/testUtils'; describe("SelectBy", function() { this.timeout(20000); setupTestSuite(); let doc: any; function formatOption(main: string, srcColumn?: string, tgtColumn?: string) { let ret = main; ret += srcColumn ? ' \u2022 ' + srcColumn : ''; ret += tgtColumn ? ' \u2192 ' + tgtColumn : ''; return ret; } it("should offer correct options", async () => { await server.simulateLogin("Chimpy", "chimpy@getgrist.com", "nasa"); doc = await gu.importFixturesDoc('chimpy', 'nasa', 'Horizon', 'selectBy.grist', false); // check tables const api = gu.createHomeApi('Chimpy', 'nasa'); assert.deepInclude(await api.getTable(doc.id, '_grist_Tables'), { id: [1, 2, 3, 4], tableId: ['Table1', 'Table2', 'Table3', 'Table3_summary_A'], summarySourceTable: [0, 0, 0, 3], }); // check visible columns (no manualSort columns) const allColumns = await api.getTable(doc.id, '_grist_Tables_column'); const visibleColumns = mapValues(allColumns, (vals) => vals.filter((v, i) => allColumns.colId[i] !== 'manualSort')); assert.deepInclude(visibleColumns, { id: [2, 3, 6, 10, 12, 13, 14, 15, 16], colId: ['table2_ref', 'table3_ref', 'table3_ref', 'A', 'A', 'table3_ref_2', 'A', 'group', 'count'], parentId: [1, 1, 2, 3, 1, 1, 4, 4, 4], type: ['Ref:Table2', 'Ref:Table3', 'Ref:Table3', 'Numeric', 'Text', 'Ref:Table3', 'Numeric', 'RefList:Table3', 'Int'], label: ['table2_ref', 'table3_ref', 'table3_ref', 'A', 'A', 'table3_ref_2', 'A', 'group', 'count'], }); // open document await driver.get(`${server.getHost()}/o/nasa/doc/${doc.id}`); // create a new page with table1 and table2 as 2 tables await gu.addNewPage(/Table/, /Table1/); await gu.addNewSection(/Table/, /Table2/); // beginning adding a new widget to page await driver.findWait('.test-dp-add-new', 2000).doClick(); await driver.findWait('.test-dp-add-widget-to-page', 2000).doClick(); // select /Table/ /Table1/ and check options of `SELECT BY` drop down await driver.findContent('.test-wselect-table', /Table1/).doClick(); await driver.findContent('.test-wselect-type', /Table/).doClick(); await driver.find('.test-wselect-selectby').doClick(); assert.deepEqual( // let's ignore the first option which is an internal added by grainjs await driver.findAll('.test-wselect-selectby option:not(:first-of-type)', (e) => e.getText()), [ // note: this is a very contrived example to test various possible links. Real world use // cases are expected to be simpler, resulting in simpler list of options that are easier to // navigate for the user than this one (in particular the `->` separator might rarely show // up). formatOption('Select Widget'), formatOption("TABLE1"), formatOption("TABLE1", 'table2_ref'), formatOption('TABLE1', 'table3_ref', 'table3_ref'), formatOption('TABLE1', 'table3_ref', 'table3_ref_2'), formatOption('TABLE1', 'table3_ref_2', 'table3_ref'), formatOption('TABLE1', 'table3_ref_2', 'table3_ref_2'), formatOption('TABLE2'), formatOption('TABLE2', 'table3_ref', 'table3_ref'), formatOption('TABLE2', 'table3_ref', 'table3_ref_2'), ] ); // select Table2 and check options of `SELECT BY` drop down await driver.findContent('.test-wselect-table', /Table2/).doClick(); await driver.find('.test-wselect-selectby').doClick(); assert.deepEqual( await driver.findAll('.test-wselect-selectby option:not(:first-of-type)', (e) => e.getText()), [ formatOption('Select Widget'), formatOption('TABLE1', 'table2_ref'), formatOption('TABLE1', 'table3_ref'), formatOption('TABLE1', 'table3_ref_2'), formatOption('TABLE2'), formatOption('TABLE2', 'table3_ref') ] ); // Selecting "New Table" should show no options. await driver.findContent('.test-wselect-table', /New Table/).doClick(); assert.equal(await driver.find('.test-wselect-selectby').isPresent(), false); assert.lengthOf(await driver.findAll('.test-wselect-selectby option'), 0); // Selecting a regular table should show options again. await driver.findContent('.test-wselect-table', /Table2/).doClick(); assert.equal(await driver.find('.test-wselect-selectby').isPresent(), true); assert.lengthOf(await driver.findAll('.test-wselect-selectby option'), 7); // Create a page with with charts and custom widget and then check that no linking is offered await gu.addNewPage(/Chart/, /Table1/); await gu.addNewSection(/Custom/, /Table2/); // open add widget to page await driver.findWait('.test-dp-add-new', 2000).doClick(); await driver.findWait('.test-dp-add-widget-to-page', 2000).doClick(); // select /Table/ /Table1/ and check no options are available await driver.findContent('.test-wselect-table', /Table1/).doClick(); await driver.findContent('.test-wselect-type', /Table/).doClick(); assert.equal(await driver.find('.test-wselect-selectby').isPresent(), false); // select Table2 and check no options are available await driver.findContent('.test-wselect-table', /Table2/).doClick(); assert.equal(await driver.find('.test-wselect-selectby').isPresent(), false); }); it('should handle summary table correctly', async () => { // Notice that table of view 4 is a summary of Table3 const api = gu.createHomeApi('Chimpy', 'nasa'); assert.deepInclude((await api.getTable(doc.id, '_grist_Tables')), { id: [1, 2, 3, 4], tableId: ['Table1', 'Table2', 'Table3', 'Table3_summary_A'], summarySourceTable: [0, 0, 0, 3], }); // open Summary page await driver.get(`${server.getHost()}/o/nasa/doc/${doc.id}/p/4`); // add new widget to page await driver.findWait('.test-dp-add-new', 2000).doClick(); await driver.findWait('.test-dp-add-widget-to-page', 2000).doClick(); // select Table3 and summarize await driver.findContent('.test-wselect-table', /Table3/).find('.test-wselect-pivot').doClick(); // check selectBy options assert.deepEqual( await driver.findAll('.test-wselect-selectby option:not(:first-of-type)', (e) => e.getText()), [], ); await driver.sendKeys(Key.ESCAPE); }); it('should show nav buttons for card view linked to its summary', async function() { // Still on the page with summary of Table3, add a new Card widget linked to the summary await gu.addNewSection(/Card$/, /Table3/, {selectBy: /TABLE3.*by A/}); // Check that we have a card view. await gu.getCell({section: 'TABLE3 [by A]', rowNum: 1, col: 'A'}).click(); const section = await gu.getSection('TABLE3 Card'); assert.equal(await gu.getDetailCell({section, rowNum: 1, col: 'A'}).getText(), '1'); // Check there are nav buttons in the card view. assert.equal(await section.find('.btn.detail-left').isPresent(), true); assert.equal(await section.find('.btn.detail-right').isPresent(), true); assert.equal(await section.find('.grist-single-record__menu__count').getText(), '1 OF 1'); // Now add a record to the source table using the card view. await section.find('.btn.detail-add-btn').click(); assert.equal(await gu.getDetailCell({section, rowNum: 1, col: 'A'}).getText(), ''); await gu.getDetailCell({section, rowNum: 1, col: 'A'}).click(); await gu.sendKeys('1', Key.ENTER); await gu.waitForServer(); // Check that this group now has 2 records. assert.equal(await section.find('.grist-single-record__menu__count').getText(), '2 OF 2'); // There is another group that still has one record. await gu.getCell({section: 'TABLE3 [by A]', rowNum: 2, col: 'A'}).click(); assert.equal(await section.find('.grist-single-record__menu__count').getText(), '1 OF 1'); }); it('should save link correctly', async () => { // create new page with table2 as a table await gu.addNewPage(/Table/, /Table2/); // begin adding table1 as a table to page await driver.findWait('.test-dp-add-new', 2000).doClick(); await driver.findWait('.test-dp-add-widget-to-page', 2000).doClick(); await driver.findContent('.test-wselect-table', /Table1/).doClick(); // select link await driver.find('.test-wselect-selectby').doClick(); await driver.findContent('.test-wselect-selectby option', /Table2/i).doClick(); // click `add to page` btn await driver.find('.test-wselect-addBtn').doClick(); await gu.waitForServer(); // check new section added and check content assert.deepEqual(await gu.getVisibleGridCells(1, [1, 2]), ['a', 'b']); // select other row in selector section await gu.getSection('Table2').doClick(); await gu.getCell({col: 0, rowNum: 2}).doClick(); // check that linked section was filterd await gu.getSection('Table1').doClick(); assert.deepEqual(await gu.getVisibleGridCells(1, [1, 2]), ['c', 'd']); // check that an single undo remove the section await gu.undo(); assert.equal(await gu.getSection('Table1').isPresent(), false); // check that a single redo add and link the section await gu.redo(); await gu.getSection('Table1').doClick(); assert.deepEqual(await gu.getVisibleGridCells(1, [1, 2]), ['a', 'b']); }); });