diff --git a/app/client/models/ColumnToMap.ts b/app/client/models/ColumnToMap.ts index d7432cf9..988d150d 100644 --- a/app/client/models/ColumnToMap.ts +++ b/app/client/models/ColumnToMap.ts @@ -25,7 +25,9 @@ export class ColumnToMapImpl implements Required { this.description = typeof def === 'string' ? '' : (def.description ?? ''); this.optional = typeof def === 'string' ? false : (def.optional ?? false); this.type = typeof def === 'string' ? 'Any' : (def.type ?? 'Any'); - this.typeDesc = String(UserType.typeDefs[this.type]?.label ?? "any").toLowerCase(); + this.type = this.type.split(',').map(t => t.trim()).filter(Boolean).join(','); + this.typeDesc = this.type.split(',') + .map(t => String(UserType.typeDefs[t]?.label ?? "any").toLowerCase()).join(', '); this.allowMultiple = typeof def === 'string' ? false : (def.allowMultiple ?? false); } diff --git a/app/client/ui/CustomSectionConfig.ts b/app/client/ui/CustomSectionConfig.ts index 8efff470..46aa9b80 100644 --- a/app/client/ui/CustomSectionConfig.ts +++ b/app/client/ui/CustomSectionConfig.ts @@ -1,30 +1,30 @@ import {allCommands} from 'app/client/components/commands'; import {GristDoc} from 'app/client/components/GristDoc'; +import {makeTestId} from 'app/client/lib/domUtils'; import * as kf from 'app/client/lib/koForm'; import {makeT} from 'app/client/lib/localization'; import {ColumnToMapImpl} from 'app/client/models/ColumnToMap'; import {ColumnRec, ViewSectionRec} from 'app/client/models/DocModel'; import {reportError} from 'app/client/models/errors'; import {cssHelp, cssLabel, cssRow, cssSeparator} from 'app/client/ui/RightPanelStyles'; +import {hoverTooltip} from 'app/client/ui/tooltips'; import {cssDragRow, cssFieldEntry, cssFieldLabel} from 'app/client/ui/VisibleFieldsConfig'; import {basicButton, primaryButton, textButton} from 'app/client/ui2018/buttons'; import {theme, vars} from 'app/client/ui2018/cssVars'; import {cssDragger} from 'app/client/ui2018/draggableList'; import {textInput} from 'app/client/ui2018/editableLabel'; -import {IconName} from 'app/client/ui2018/IconList'; import {icon} from 'app/client/ui2018/icons'; import {cssLink} from 'app/client/ui2018/links'; -import {IOptionFull, menu, menuItem, menuText, select} from 'app/client/ui2018/menus'; +import {cssOptionLabel, IOption, IOptionFull, menu, menuItem, menuText, select} from 'app/client/ui2018/menus'; import {AccessLevel, ICustomWidget, isSatisfied} from 'app/common/CustomWidget'; import {GristLoadConfig} from 'app/common/gristUrls'; -import {unwrap} from 'app/common/gutil'; +import {not, unwrap} from 'app/common/gutil'; import { bundleChanges, Computed, Disposable, dom, fromKo, - makeTestId, MultiHolder, Observable, styled, @@ -62,12 +62,50 @@ class ColumnPicker extends Disposable { const value = use(this._value); return Array.isArray(value) ? null : value; }); - properValue.onWrite(value => this._value.set(value)); - const options = Computed.create(this, use => { + properValue.onWrite(value => this._value.set(value || null)); + + const canBeMapped = Computed.create(this, use => { return use(this._section.columns) - .filter(col => this._column.canByMapped(use(col.pureType))) - .map((col) => ({value: col.getRowId(), label: use(col.label), icon: 'FieldColumn' as IconName})); + .filter(col => this._column.canByMapped(use(col.pureType))); + }); + + // This is a HACK, to refresh options only when the menu is opened (or closed) + // and not to track down all the dependencies. Otherwise the select menu won't + // be hidden when option is selected - there is a bug that prevents it from closing + // when list of options is changed. + const refreshTrigger = Observable.create(this, false); + + const options = Computed.create(this, use => { + void use(refreshTrigger); + + const columnsAsOptions: IOption[] = use(canBeMapped) + .map((col) => ({ + value: col.getRowId(), + label: col.label.peek(), + icon: 'FieldColumn', + })); + + // For optional mappings, add 'Blank' option but only if the value is set. + // This option will allow to clear the selection. + if (this._column.optional && properValue.get()) { + columnsAsOptions.push({ + value: 0, + // Another hack. Select doesn't allow to have different label for blank option and the default text. + // So we will render this label ourselves later using `renderOptionArgs`. + label: '', + }); + } + return columnsAsOptions; + }); + + const isDisabled = Computed.create(this, use => { + return use(canBeMapped).length === 0; }); + + const defaultLabel = this._column.typeDesc != "any" + ? t("Pick a {{columnType}} column", {"columnType": this._column.typeDesc}) + : t("Pick a column"); + return [ cssLabel( this._column.title, @@ -78,18 +116,49 @@ class ColumnPicker extends Disposable { this._column.description, testId('help-for-' + this._column.name), ) : null, - cssRow( - select( - properValue, - options, - { - defaultLabel: this._column.typeDesc != "any" - ? t("Pick a {{columnType}} column", {"columnType": this._column.typeDesc}) - : t("Pick a column") - } - ), - testId('mapping-for-' + this._column.name), - ), + dom.maybe(not(isDisabled), () => [ + cssRow( + dom.update( + select( + properValue, + options, + { + defaultLabel, + renderOptionArgs : (opt) => { + // If there is a label, render it. + // Otherwise show the 'Clear selection' label as a greyed out text. + // This is the continuation of the hack from above - were we added an option + // without a label. + return (opt.label) ? null : [ + cssBlank(t("Clear selection")), + testId('clear-selection'), + ]; + } + } + ), + dom.on('click', () => { + // When the menu is opened or closed, refresh the options. + refreshTrigger.set(!refreshTrigger.get()); + }) + ), + testId('mapping-for-' + this._column.name), + testId('enabled'), + ), + ]), + dom.maybe(isDisabled, () => [ + cssRow( + cssDisabledSelect( + Observable.create(this, null), + [], { + disabled: true, + defaultLabel: t("No {{columnType}} columns in table.", {"columnType": this._column.typeDesc}) + } + ), + hoverTooltip(t("No {{columnType}} columns in table.", {"columnType": this._column.typeDesc})), + testId('mapping-for-' + this._column.name), + testId('disabled'), + ), + ]), ]; } } @@ -114,16 +183,30 @@ class ColumnListPicker extends Disposable { }); } private _buildAddColumn() { + + const owner = MultiHolder.create(null); + + const notMapped = Computed.create(owner, use => { + const value = use(this._value) || []; + const mapped = !Array.isArray(value) ? [] : value; + return this._section.columns().filter(col => !mapped.includes(use(col.id))); + }); + + const typedColumns = Computed.create(owner, use => { + return use(notMapped).filter(this._typeFilter(use)); + }); + return [ cssRow( + dom.autoDispose(owner), cssAddMapping( cssAddIcon('Plus'), t("Add") + ' ' + this._column.title, + dom.cls('disabled', use => use(notMapped).length === 0), + testId('disabled', use => use(notMapped).length === 0), menu(() => { - const otherColumns = this._getNotMappedColumns(); - const typedColumns = otherColumns.filter(this._typeFilter()); - const wrongTypeCount = otherColumns.length - typedColumns.length; + const wrongTypeCount = notMapped.get().length - typedColumns.get().length; return [ - ...typedColumns + ...typedColumns.get() .map((col) => menuItem( () => this._addColumn(col), col.label.peek(), @@ -145,7 +228,8 @@ class ColumnListPicker extends Disposable { } // Helper method for filtering columns that can be picked by the widget. - private _typeFilter = (use = unwrap) => (col: ColumnRec) => this._column.canByMapped(use(col.pureType)); + private _typeFilter = (use = unwrap) => (col: ColumnRec|null) => + !col ? false : this._column.canByMapped(use(col.pureType)); private _buildDraggableList(use: UseCBOwner) { return dom.update(kf.draggableList( @@ -159,12 +243,7 @@ class ColumnListPicker extends Disposable { } ), testId('map-list-for-' + this._column.name)); } - private _getNotMappedColumns(): ColumnRec[] { - // Get all columns. - const all = this._section.columns.peek(); - const mapped = this._list(); - return all.filter(col => !mapped.includes(col.id.peek())); - } + private _readItems(use: UseCBOwner): ColumnRec[] { let selectedRefs = (use(this._value) || []) as number[]; // Ignore if configuration was changed from what it was saved. @@ -177,6 +256,7 @@ class ColumnListPicker extends Disposable { // Remove any columns that are no longer there. return selectedRefs.map(s => columnMap.get(s)!).filter(c => Boolean(c)); } + private _renderItem(use: UseCBOwner, field: ColumnRec): any { return cssFieldEntry( cssFieldLabel( @@ -199,7 +279,7 @@ class ColumnListPicker extends Disposable { this._value.set(value); } else { let current = (this._value.get() || []) as number[]; - // Ignore if the saved value is not a number. + // Ignore if the saved value is not a number list. if (!Array.isArray(current)) { current = []; } @@ -224,8 +304,16 @@ class ColumnListPicker extends Disposable { this._value.set(current.filter(c => c != column.id.peek())); } private _addColumn(col: ColumnRec): any { - const current = this._list(); + // Helper to find column model. + const model = (id: number) => this._section.columns().find(c => c.id.peek() === id) || null; + // Get the list of currently mapped columns. + let current = this._list(); + // Add new column. current.push(col.id.peek()); + // Remove those that don't exists anymore. + current = current.filter(c => model(c)); + // And those with wrong type. + current = current.filter(c => this._typeFilter()(model(c))); this._value.set(current); } } @@ -551,8 +639,6 @@ export class CustomSectionConfig extends Disposable { this._widgets.set(wigets); } - - private _accept() { if (this._desiredAccess.get()) { this._currentAccess.set(this._desiredAccess.get()!); @@ -629,6 +715,11 @@ const cssAddMapping = styled('div', ` color: ${theme.controlHoverFg}; --icon-color: ${theme.controlHoverFg}; } + &.disabled { + color: ${theme.lightText}; + --icon-color: ${theme.lightText}; + pointer-events: none; + } `); const cssTextInput = styled(textInput, ` @@ -647,3 +738,11 @@ const cssTextInput = styled(textInput, ` color: ${theme.inputPlaceholderFg}; } `); + +const cssDisabledSelect = styled(select, ` + opacity: unset !important; +`); + +const cssBlank = styled(cssOptionLabel, ` + --grist-option-label-color: ${theme.lightText}; +`); diff --git a/app/client/ui/widgetTypesMap.ts b/app/client/ui/widgetTypesMap.ts index 9e2bc614..4a2c33d6 100644 --- a/app/client/ui/widgetTypesMap.ts +++ b/app/client/ui/widgetTypesMap.ts @@ -8,7 +8,7 @@ export const widgetTypesMap = new Map([ ['detail', {label: 'Card List', icon: 'TypeCardList'}], ['chart', {label: 'Chart', icon: 'TypeChart'}], ['custom', {label: 'Custom', icon: 'TypeCustom'}], - ['custom.calendar', {label: 'Calendar', icon: 'FieldDate'}] + ['custom.calendar', {label: 'Calendar', icon: 'TypeCalendar'}], ]); // Widget type info. diff --git a/app/client/ui2018/IconList.ts b/app/client/ui2018/IconList.ts index 84735fa6..cd9735a1 100644 --- a/app/client/ui2018/IconList.ts +++ b/app/client/ui2018/IconList.ts @@ -4,6 +4,7 @@ export type IconName = "ChartArea" | "ChartKaplan" | "ChartLine" | "ChartPie" | + "TypeCalendar" | "TypeCard" | "TypeCardList" | "TypeCell" | @@ -152,6 +153,7 @@ export const IconList: IconName[] = ["ChartArea", "ChartKaplan", "ChartLine", "ChartPie", + "TypeCalendar", "TypeCard", "TypeCardList", "TypeCell", diff --git a/app/client/ui2018/menus.ts b/app/client/ui2018/menus.ts index 88d6f0c1..7ff22fba 100644 --- a/app/client/ui2018/menus.ts +++ b/app/client/ui2018/menus.ts @@ -534,22 +534,25 @@ export const cssOptionRowIcon = styled(icon, ` } `); -const cssOptionLabel = styled('div', ` +export const cssOptionLabel = styled('div', ` white-space: nowrap; overflow: hidden; text-overflow: ellipsis; + --grist-option-label-color: ${theme.menuItemFg}; + --grist-option-label-color-sel: ${theme.menuItemSelectedFg}; + --grist-option-label-color-disabled: ${theme.menuItemDisabledFg}; .${weasel.cssMenuItem.className} & { - color: ${theme.menuItemFg}; + color: var(--grist-option-label-color); } .${weasel.cssMenuItem.className}-sel & { - color: ${theme.menuItemSelectedFg}; + color: var(--grist-option-label-color-sel); background-color: ${theme.menuItemSelectedBg}; } .${weasel.cssMenuItem.className}.disabled & { - color: ${theme.menuItemDisabledFg}; + color: var(--grist-option-label-color-disabled); } `); diff --git a/static/icons/icons.css b/static/icons/icons.css index 85757bb3..d48fc32d 100644 --- a/static/icons/icons.css +++ b/static/icons/icons.css @@ -5,6 +5,7 @@ --icon-ChartKaplan: url(''); --icon-ChartLine: url(''); --icon-ChartPie: url(''); + --icon-TypeCalendar: url(''); --icon-TypeCard: url(''); --icon-TypeCardList: url(''); --icon-TypeCell: url(''); diff --git a/static/ui-icons/Data Types/TypeCalendar.svg b/static/ui-icons/Data Types/TypeCalendar.svg new file mode 100644 index 00000000..e99578d2 --- /dev/null +++ b/static/ui-icons/Data Types/TypeCalendar.svg @@ -0,0 +1,22 @@ + + + + + + diff --git a/test/nbrowser/CustomWidgetsConfig.ts b/test/nbrowser/CustomWidgetsConfig.ts index bfa4ea30..3e77f810 100644 --- a/test/nbrowser/CustomWidgetsConfig.ts +++ b/test/nbrowser/CustomWidgetsConfig.ts @@ -356,6 +356,97 @@ describe('CustomWidgetsConfig', function () { await clickOption(TESTER_WIDGET); }); + it('should hide mappings when there is no good column', async () => { + await gu.setWidgetUrl( + createConfigUrl({ + columns: [{name: 'M2', type: 'Date'}], + requiredAccess: 'read table', + }) + ); + + await accept(); + + // Get the drop for M2 mappings. + const mappingsForM2 = () => driver.find(pickerDrop('M2')); + + // Make sure it is disabled. + assert.isTrue(await mappingsForM2().matches('.test-config-widget-disabled')); + // And the text is: + assert.equal(await mappingsForM2().getText(), 'No date columns in table.'); + + // Now add Date column. + await gu.sendActions([['AddVisibleColumn', 'Table1', 'NewCol', {type: 'Date'}]]); + + // Now drop should be enabled. + assert.isFalse(await mappingsForM2().matches('.test-config-widget-disabled')); + assert.isTrue(await mappingsForM2().matches('.test-config-widget-enabled')); + + // And the text is: + assert.equal(await mappingsForM2().getText(), 'Pick a date column'); + + // Expand it and make sure we have NewCol there. + await toggleDrop(pickerDrop('M2')); + assert.deepEqual(await getOptions(), ['NewCol']); + + // Select that column. + await clickOption('NewCol'); + + // Now expand the drop again and make sure we can't clear it. + await toggleDrop(pickerDrop('M2')); + assert.deepEqual(await getOptions(), ['NewCol']); + + // Now remove the column, and make sure that the drop is disabled again. + await driver.sendKeys(Key.ESCAPE); + await gu.sendActions([['RemoveColumn', 'Table1', 'NewCol']]); + + // Make sure it is disabled. + assert.isTrue(await mappingsForM2().matches('.test-config-widget-disabled')); + assert.isFalse(await mappingsForM2().matches('.test-config-widget-enabled')); + assert.equal(await mappingsForM2().getText(), 'No date columns in table.'); + }); + + it('should clear optional mapping', async () => { + const revert = await gu.begin(); + await gu.setWidgetUrl( + createConfigUrl({ + columns: [{name: 'M2', type: 'Date', optional: true}], + requiredAccess: 'read table', + }) + ); + + await accept(); + + // Get the drop for M2 mappings. + const mappingsForM2 = () => driver.find(pickerDrop('M2')); + + // Make sure it is disabled. + assert.isTrue(await mappingsForM2().matches('.test-config-widget-disabled')); + // Now add Date column. + await gu.sendActions([['AddVisibleColumn', 'Table1', 'NewCol', {type: 'Date'}]]); + + // Expand it and make sure we have NewCol there. + await toggleDrop(pickerDrop('M2')); + assert.deepEqual(await getOptions(), ['NewCol']); + + // Select that column. + await clickOption('NewCol'); + + // Make sure widget sees the mapping. + assert.deepEqual(await widget.onRecordsMappings(), {M2: 'NewCol'}); + + // Now expand the drop again and make sure we can clear it. + await toggleDrop(pickerDrop('M2')); + assert.deepEqual(await getOptions(), ['NewCol', 'Clear selection']); + + // Now clear the mapping. + await clickOption('Clear selection'); + assert.equal(await mappingsForM2().getText(), 'Pick a date column'); + + // Make sure widget sees the mapping. + assert.deepEqual(await widget.onRecordsMappings(), {M2: null}); + await revert(); + }); + it('should render columns mapping', async () => { const revert = await gu.begin(); assert.isTrue(await driver.find('.test-vfc-visible-fields-select-all').isPresent()); @@ -759,14 +850,12 @@ describe('CustomWidgetsConfig', function () { await gu.selectSectionByTitle('Widget'); await driver.find(".test-right-tab-pagewidget").click(); // Drop should be empty, - assert.equal(await driver.find(pickerDrop("M1")).getText(), "Pick a text column"); + assert.equal(await driver.find(pickerDrop("M1")).getText(), "No text columns in table."); assert.isEmpty(await getListItems("M2")); - // with no options - await toggleDrop(pickerDrop("M1")); - assert.isEmpty(await getOptions()); - await gu.sendKeys(Key.ESCAPE); + // And drop is disabled. + assert.isTrue(await driver.find(pickerDrop("M1")).matches(".test-config-widget-disabled")); // The same for M2 - await click(pickerAdd("M2")); + assert.isTrue(await driver.find(pickerAdd("M2")).matches(".test-config-widget-disabled")); assert.isEmpty(await getMenuOptions()); assert.deepEqual(await widget.onRecordsMappings(), {M1: null, M2: []}); assert.deepEqual(await widget.onRecords(), [