(core) Mappings improvements

Summary:
- Adding new icon for calendar view (the old one by just bigger)
- When there are no columns to map the select box is grayed out
- Optional mappings can be cleared now

Test Plan: Added

Reviewers: JakubSerafin

Reviewed By: JakubSerafin

Subscribers: JakubSerafin

Differential Revision: https://phab.getgrist.com/D4066
This commit is contained in:
Jarosław Sadziński 2023-10-06 11:11:37 +02:00
parent 34f366585d
commit 572279916e
8 changed files with 265 additions and 47 deletions

View File

@ -25,7 +25,9 @@ export class ColumnToMapImpl implements Required<ColumnToMap> {
this.description = typeof def === 'string' ? '' : (def.description ?? ''); this.description = typeof def === 'string' ? '' : (def.description ?? '');
this.optional = typeof def === 'string' ? false : (def.optional ?? false); this.optional = typeof def === 'string' ? false : (def.optional ?? false);
this.type = typeof def === 'string' ? 'Any' : (def.type ?? 'Any'); 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); this.allowMultiple = typeof def === 'string' ? false : (def.allowMultiple ?? false);
} }

View File

@ -1,30 +1,30 @@
import {allCommands} from 'app/client/components/commands'; import {allCommands} from 'app/client/components/commands';
import {GristDoc} from 'app/client/components/GristDoc'; import {GristDoc} from 'app/client/components/GristDoc';
import {makeTestId} from 'app/client/lib/domUtils';
import * as kf from 'app/client/lib/koForm'; import * as kf from 'app/client/lib/koForm';
import {makeT} from 'app/client/lib/localization'; import {makeT} from 'app/client/lib/localization';
import {ColumnToMapImpl} from 'app/client/models/ColumnToMap'; import {ColumnToMapImpl} from 'app/client/models/ColumnToMap';
import {ColumnRec, ViewSectionRec} from 'app/client/models/DocModel'; import {ColumnRec, ViewSectionRec} from 'app/client/models/DocModel';
import {reportError} from 'app/client/models/errors'; import {reportError} from 'app/client/models/errors';
import {cssHelp, cssLabel, cssRow, cssSeparator} from 'app/client/ui/RightPanelStyles'; 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 {cssDragRow, cssFieldEntry, cssFieldLabel} from 'app/client/ui/VisibleFieldsConfig';
import {basicButton, primaryButton, textButton} from 'app/client/ui2018/buttons'; import {basicButton, primaryButton, textButton} from 'app/client/ui2018/buttons';
import {theme, vars} from 'app/client/ui2018/cssVars'; import {theme, vars} from 'app/client/ui2018/cssVars';
import {cssDragger} from 'app/client/ui2018/draggableList'; import {cssDragger} from 'app/client/ui2018/draggableList';
import {textInput} from 'app/client/ui2018/editableLabel'; import {textInput} from 'app/client/ui2018/editableLabel';
import {IconName} from 'app/client/ui2018/IconList';
import {icon} from 'app/client/ui2018/icons'; import {icon} from 'app/client/ui2018/icons';
import {cssLink} from 'app/client/ui2018/links'; 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 {AccessLevel, ICustomWidget, isSatisfied} from 'app/common/CustomWidget';
import {GristLoadConfig} from 'app/common/gristUrls'; import {GristLoadConfig} from 'app/common/gristUrls';
import {unwrap} from 'app/common/gutil'; import {not, unwrap} from 'app/common/gutil';
import { import {
bundleChanges, bundleChanges,
Computed, Computed,
Disposable, Disposable,
dom, dom,
fromKo, fromKo,
makeTestId,
MultiHolder, MultiHolder,
Observable, Observable,
styled, styled,
@ -62,12 +62,50 @@ class ColumnPicker extends Disposable {
const value = use(this._value); const value = use(this._value);
return Array.isArray(value) ? null : value; return Array.isArray(value) ? null : value;
}); });
properValue.onWrite(value => this._value.set(value)); properValue.onWrite(value => this._value.set(value || null));
const options = Computed.create(this, use => {
const canBeMapped = Computed.create(this, use => {
return use(this._section.columns) return use(this._section.columns)
.filter(col => this._column.canByMapped(use(col.pureType))) .filter(col => this._column.canByMapped(use(col.pureType)));
.map((col) => ({value: col.getRowId(), label: use(col.label), icon: 'FieldColumn' as IconName}));
}); });
// 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<number|null>[] = 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 [ return [
cssLabel( cssLabel(
this._column.title, this._column.title,
@ -78,18 +116,49 @@ class ColumnPicker extends Disposable {
this._column.description, this._column.description,
testId('help-for-' + this._column.name), testId('help-for-' + this._column.name),
) : null, ) : null,
dom.maybe(not(isDisabled), () => [
cssRow( cssRow(
dom.update(
select( select(
properValue, properValue,
options, options,
{ {
defaultLabel: this._column.typeDesc != "any" defaultLabel,
? t("Pick a {{columnType}} column", {"columnType": this._column.typeDesc}) renderOptionArgs : (opt) => {
: t("Pick a column") // 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'),
];
}
} }
), ),
testId('mapping-for-' + this._column.name), 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() { 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 [ return [
cssRow( cssRow(
dom.autoDispose(owner),
cssAddMapping( cssAddMapping(
cssAddIcon('Plus'), t("Add") + ' ' + this._column.title, cssAddIcon('Plus'), t("Add") + ' ' + this._column.title,
dom.cls('disabled', use => use(notMapped).length === 0),
testId('disabled', use => use(notMapped).length === 0),
menu(() => { menu(() => {
const otherColumns = this._getNotMappedColumns(); const wrongTypeCount = notMapped.get().length - typedColumns.get().length;
const typedColumns = otherColumns.filter(this._typeFilter());
const wrongTypeCount = otherColumns.length - typedColumns.length;
return [ return [
...typedColumns ...typedColumns.get()
.map((col) => menuItem( .map((col) => menuItem(
() => this._addColumn(col), () => this._addColumn(col),
col.label.peek(), col.label.peek(),
@ -145,7 +228,8 @@ class ColumnListPicker extends Disposable {
} }
// Helper method for filtering columns that can be picked by the widget. // 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) { private _buildDraggableList(use: UseCBOwner) {
return dom.update(kf.draggableList( return dom.update(kf.draggableList(
@ -159,12 +243,7 @@ class ColumnListPicker extends Disposable {
} }
), testId('map-list-for-' + this._column.name)); ), 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[] { private _readItems(use: UseCBOwner): ColumnRec[] {
let selectedRefs = (use(this._value) || []) as number[]; let selectedRefs = (use(this._value) || []) as number[];
// Ignore if configuration was changed from what it was saved. // 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. // Remove any columns that are no longer there.
return selectedRefs.map(s => columnMap.get(s)!).filter(c => Boolean(c)); return selectedRefs.map(s => columnMap.get(s)!).filter(c => Boolean(c));
} }
private _renderItem(use: UseCBOwner, field: ColumnRec): any { private _renderItem(use: UseCBOwner, field: ColumnRec): any {
return cssFieldEntry( return cssFieldEntry(
cssFieldLabel( cssFieldLabel(
@ -199,7 +279,7 @@ class ColumnListPicker extends Disposable {
this._value.set(value); this._value.set(value);
} else { } else {
let current = (this._value.get() || []) as number[]; 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)) { if (!Array.isArray(current)) {
current = []; current = [];
} }
@ -224,8 +304,16 @@ class ColumnListPicker extends Disposable {
this._value.set(current.filter(c => c != column.id.peek())); this._value.set(current.filter(c => c != column.id.peek()));
} }
private _addColumn(col: ColumnRec): any { 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()); 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); this._value.set(current);
} }
} }
@ -551,8 +639,6 @@ export class CustomSectionConfig extends Disposable {
this._widgets.set(wigets); this._widgets.set(wigets);
} }
private _accept() { private _accept() {
if (this._desiredAccess.get()) { if (this._desiredAccess.get()) {
this._currentAccess.set(this._desiredAccess.get()!); this._currentAccess.set(this._desiredAccess.get()!);
@ -629,6 +715,11 @@ const cssAddMapping = styled('div', `
color: ${theme.controlHoverFg}; color: ${theme.controlHoverFg};
--icon-color: ${theme.controlHoverFg}; --icon-color: ${theme.controlHoverFg};
} }
&.disabled {
color: ${theme.lightText};
--icon-color: ${theme.lightText};
pointer-events: none;
}
`); `);
const cssTextInput = styled(textInput, ` const cssTextInput = styled(textInput, `
@ -647,3 +738,11 @@ const cssTextInput = styled(textInput, `
color: ${theme.inputPlaceholderFg}; color: ${theme.inputPlaceholderFg};
} }
`); `);
const cssDisabledSelect = styled(select, `
opacity: unset !important;
`);
const cssBlank = styled(cssOptionLabel, `
--grist-option-label-color: ${theme.lightText};
`);

View File

@ -8,7 +8,7 @@ export const widgetTypesMap = new Map<IWidgetType, IWidgetTypeInfo>([
['detail', {label: 'Card List', icon: 'TypeCardList'}], ['detail', {label: 'Card List', icon: 'TypeCardList'}],
['chart', {label: 'Chart', icon: 'TypeChart'}], ['chart', {label: 'Chart', icon: 'TypeChart'}],
['custom', {label: 'Custom', icon: 'TypeCustom'}], ['custom', {label: 'Custom', icon: 'TypeCustom'}],
['custom.calendar', {label: 'Calendar', icon: 'FieldDate'}] ['custom.calendar', {label: 'Calendar', icon: 'TypeCalendar'}],
]); ]);
// Widget type info. // Widget type info.

View File

@ -4,6 +4,7 @@ export type IconName = "ChartArea" |
"ChartKaplan" | "ChartKaplan" |
"ChartLine" | "ChartLine" |
"ChartPie" | "ChartPie" |
"TypeCalendar" |
"TypeCard" | "TypeCard" |
"TypeCardList" | "TypeCardList" |
"TypeCell" | "TypeCell" |
@ -152,6 +153,7 @@ export const IconList: IconName[] = ["ChartArea",
"ChartKaplan", "ChartKaplan",
"ChartLine", "ChartLine",
"ChartPie", "ChartPie",
"TypeCalendar",
"TypeCard", "TypeCard",
"TypeCardList", "TypeCardList",
"TypeCell", "TypeCell",

View File

@ -534,22 +534,25 @@ export const cssOptionRowIcon = styled(icon, `
} }
`); `);
const cssOptionLabel = styled('div', ` export const cssOptionLabel = styled('div', `
white-space: nowrap; white-space: nowrap;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; 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} & { .${weasel.cssMenuItem.className} & {
color: ${theme.menuItemFg}; color: var(--grist-option-label-color);
} }
.${weasel.cssMenuItem.className}-sel & { .${weasel.cssMenuItem.className}-sel & {
color: ${theme.menuItemSelectedFg}; color: var(--grist-option-label-color-sel);
background-color: ${theme.menuItemSelectedBg}; background-color: ${theme.menuItemSelectedBg};
} }
.${weasel.cssMenuItem.className}.disabled & { .${weasel.cssMenuItem.className}.disabled & {
color: ${theme.menuItemDisabledFg}; color: var(--grist-option-label-color-disabled);
} }
`); `);

View File

@ -5,6 +5,7 @@
--icon-ChartKaplan: url(''); --icon-ChartKaplan: url('');
--icon-ChartLine: url(''); --icon-ChartLine: url('');
--icon-ChartPie: url(''); --icon-ChartPie: url('');
--icon-TypeCalendar: url('');
--icon-TypeCard: url(''); --icon-TypeCard: url('');
--icon-TypeCardList: url(''); --icon-TypeCardList: url('');
--icon-TypeCell: url(''); --icon-TypeCell: url('');

View File

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg
xmlns:dc="http://purl.org/dc/elements/1.1/"
xmlns:cc="http://creativecommons.org/ns#"
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
xmlns:svg="http://www.w3.org/2000/svg"
xmlns="http://www.w3.org/2000/svg"
width="16px"
height="16px"
viewBox="0 0 16 16"
version="1.1">
<g
stroke="none"
stroke-width="1"
fill="none"
fill-rule="evenodd">
<path
d="M 5.1438183,3.2089992 H 11.143818 V 0.79814978 c 0,-0.29037145 0.243675,-0.49817134 0.5,-0.5 0.256325,-0.001829 0.5,0.21305073 0.5,0.5 V 3.2089992 h 2.129941 c 0.828427,0 1.5,0.6715729 1.5,1.5 v 9.5887298 c 0,0.828427 -0.671573,1.5 -1.5,1.5 H 1.8700589 c -0.8284271,0 -1.49999996,-0.671573 -1.49999996,-1.5 V 4.7089992 c 0,-0.8284271 0.67157286,-1.5 1.49999996,-1.5 H 4.1438183 V 0.79814978 c 0,-0.27614238 0.2044615,-0.50115994 0.5,-0.5 0.2955385,0.00116 0.5,0.2230347 0.5,0.5 z M 1.3700589,7.1198486 H 14.773759 V 4.7089992 c 0,-0.2761424 -0.223858,-0.5 -0.5,-0.5 H 1.8700589 c -0.2761423,0 -0.5,0.2238576 -0.5,0.5 z m 13.4037001,1 H 1.3700589 v 6.1778804 c 0,0.276142 0.2238577,0.5 0.5,0.5 H 14.273759 c 0.276142,0 0.5,-0.223858 0.5,-0.5 z"
id="Combined-Shape"
style="fill:#000000;fill-rule:nonzero"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 1.3 KiB

View File

@ -356,6 +356,97 @@ describe('CustomWidgetsConfig', function () {
await clickOption(TESTER_WIDGET); 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 () => { it('should render columns mapping', async () => {
const revert = await gu.begin(); const revert = await gu.begin();
assert.isTrue(await driver.find('.test-vfc-visible-fields-select-all').isPresent()); assert.isTrue(await driver.find('.test-vfc-visible-fields-select-all').isPresent());
@ -759,14 +850,12 @@ describe('CustomWidgetsConfig', function () {
await gu.selectSectionByTitle('Widget'); await gu.selectSectionByTitle('Widget');
await driver.find(".test-right-tab-pagewidget").click(); await driver.find(".test-right-tab-pagewidget").click();
// Drop should be empty, // 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")); assert.isEmpty(await getListItems("M2"));
// with no options // And drop is disabled.
await toggleDrop(pickerDrop("M1")); assert.isTrue(await driver.find(pickerDrop("M1")).matches(".test-config-widget-disabled"));
assert.isEmpty(await getOptions());
await gu.sendKeys(Key.ESCAPE);
// The same for M2 // 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.isEmpty(await getMenuOptions());
assert.deepEqual(await widget.onRecordsMappings(), {M1: null, M2: []}); assert.deepEqual(await widget.onRecordsMappings(), {M1: null, M2: []});
assert.deepEqual(await widget.onRecords(), [ assert.deepEqual(await widget.onRecords(), [