(core) Form kanban tasks

Summary:
- Open all links in a new tab
- Excluding not filled columns (to fix trigger formulas)
- Fixed Ref/RefList submission
- Removing redundant type definitions for Box
- Adding header menu item
- Default empty values in select control

Test Plan: Updated

Reviewers: georgegevoian

Reviewed By: georgegevoian

Differential Revision: https://phab.getgrist.com/D4166
This commit is contained in:
Jarosław Sadziński 2024-01-23 21:52:57 +01:00
parent 007c4492dc
commit 95c0441d84
22 changed files with 363 additions and 124 deletions

View File

@ -1,10 +1,11 @@
import {buildEditor} from 'app/client/components/Forms/Editor'; import {buildEditor} from 'app/client/components/Forms/Editor';
import {buildMenu} from 'app/client/components/Forms/Menu'; import {buildMenu} from 'app/client/components/Forms/Menu';
import {Box, BoxModel} from 'app/client/components/Forms/Model'; import {BoxModel} from 'app/client/components/Forms/Model';
import * as style from 'app/client/components/Forms/styles'; import * as style from 'app/client/components/Forms/styles';
import {makeTestId} from 'app/client/lib/domUtils'; import {makeTestId} from 'app/client/lib/domUtils';
import {icon} from 'app/client/ui2018/icons'; import {icon} from 'app/client/ui2018/icons';
import * as menus from 'app/client/ui2018/menus'; import * as menus from 'app/client/ui2018/menus';
import {Box} from 'app/common/Forms';
import {inlineStyle, not} from 'app/common/gutil'; import {inlineStyle, not} from 'app/common/gutil';
import {bundleChanges, Computed, dom, IDomArgs, MultiHolder, Observable, styled} from 'grainjs'; import {bundleChanges, Computed, dom, IDomArgs, MultiHolder, Observable, styled} from 'grainjs';
@ -30,8 +31,7 @@ export class ColumnsModel extends BoxModel {
if (!this.parent) { throw new Error('No parent'); } if (!this.parent) { throw new Error('No parent'); }
// We need to remove it from the parent, so find it first. // We need to remove it from the parent, so find it first.
const droppedId = dropped.id; const droppedRef = dropped.id ? this.root().get(dropped.id) : null;
const droppedRef = this.root().get(droppedId);
// Now we simply insert it after this box. // Now we simply insert it after this box.
droppedRef?.removeSelf(); droppedRef?.removeSelf();
@ -165,6 +165,10 @@ export class PlaceholderModel extends BoxModel {
} }
} }
export function Paragraph(text: string, alignment?: 'left'|'right'|'center'): Box {
return {type: 'Paragraph', text, alignment};
}
export function Placeholder(): Box { export function Placeholder(): Box {
return {type: 'Placeholder'}; return {type: 'Placeholder'};
} }

View File

@ -1,12 +1,13 @@
import {buildEditor} from 'app/client/components/Forms/Editor'; import {buildEditor} from 'app/client/components/Forms/Editor';
import {FormView} from 'app/client/components/Forms/FormView'; import {FormView} from 'app/client/components/Forms/FormView';
import {Box, BoxModel, ignoreClick} from 'app/client/components/Forms/Model'; import {BoxModel, ignoreClick} from 'app/client/components/Forms/Model';
import * as css from 'app/client/components/Forms/styles'; import * as css from 'app/client/components/Forms/styles';
import {stopEvent} from 'app/client/lib/domUtils'; import {stopEvent} from 'app/client/lib/domUtils';
import {refRecord} from 'app/client/models/DocModel'; import {refRecord} from 'app/client/models/DocModel';
import {autoGrow} from 'app/client/ui/forms'; import {autoGrow} from 'app/client/ui/forms';
import {squareCheckbox} from 'app/client/ui2018/checkbox'; import {squareCheckbox} from 'app/client/ui2018/checkbox';
import {colors} from 'app/client/ui2018/cssVars'; import {colors} from 'app/client/ui2018/cssVars';
import {Box} from 'app/common/Forms';
import {Constructor} from 'app/common/gutil'; import {Constructor} from 'app/common/gutil';
import { import {
BindableValue, BindableValue,
@ -63,7 +64,11 @@ export class FieldModel extends BoxModel {
* Field row id. * Field row id.
*/ */
public get leaf() { public get leaf() {
return this.props['leaf'] as Observable<number>; return this.prop('leaf') as Observable<number>;
}
public get required() {
return this.prop('formRequired', false) as Observable<boolean|undefined>;
} }
/** /**
@ -260,41 +265,47 @@ class TextModel extends Question {
} }
class ChoiceModel extends Question { class ChoiceModel extends Question {
public renderInput() { protected choices: Computed<string[]> = Computed.create(this, use => {
// Read choices from field.
const list = use(use(use(this.model.field).origCol).widgetOptionsJson.prop('choices')) || [];
// Make sure it is array of strings.
if (!Array.isArray(list) || list.some((v) => typeof v !== 'string')) {
return [];
}
return list;
});
protected choicesWithEmpty = Computed.create(this, use => {
const list = Array.from(use(this.choices));
// Add empty choice if not present.
if (list.length === 0 || list[0] !== '') {
list.unshift('');
}
return list;
});
public renderInput(): HTMLElement {
const field = this.model.field; const field = this.model.field;
const choices: Computed<string[]> = Computed.create(this, use => {
return use(use(use(field).origCol).widgetOptionsJson.prop('choices')) || [];
});
const typedChoices = Computed.create(this, use => {
const value = use(choices);
// Make sure it is array of strings.
if (!Array.isArray(value) || value.some((v) => typeof v !== 'string')) {
return [];
}
return value;
});
return css.cssSelect( return css.cssSelect(
{tabIndex: "-1"}, {tabIndex: "-1"},
ignoreClick, ignoreClick,
dom.prop('name', use => use(use(field).colId)), dom.prop('name', use => use(use(field).colId)),
dom.forEach(typedChoices, (choice) => dom('option', choice, {value: choice})), dom.forEach(this.choicesWithEmpty, (choice) => dom('option', choice, {value: choice})),
); );
} }
} }
class ChoiceListModel extends Question { class ChoiceListModel extends ChoiceModel {
public renderInput() { public renderInput() {
const field = this.model.field; const field = this.model.field;
const choices: Computed<string[]> = Computed.create(this, use => {
return use(use(use(field).origCol).widgetOptionsJson.prop('choices')) || [];
});
return dom('div', return dom('div',
dom.prop('name', use => use(use(field).colId)), dom.prop('name', use => use(use(field).colId)),
dom.forEach(choices, (choice) => css.cssCheckboxLabel( dom.forEach(this.choices, (choice) => css.cssCheckboxLabel(
squareCheckbox(observable(false)), squareCheckbox(observable(false)),
choice choice
)), )),
dom.maybe(use => use(choices).length === 0, () => [ dom.maybe(use => use(this.choices).length === 0, () => [
dom('div', 'No choices defined'), dom('div', 'No choices defined'),
]), ]),
); );
@ -393,12 +404,19 @@ class RefListModel extends Question {
} }
class RefModel extends RefListModel { class RefModel extends RefListModel {
protected withEmpty = Computed.create(this, use => {
const list = Array.from(use(this.choices));
// Add empty choice if not present.
list.unshift([0, '']);
return list;
});
public renderInput() { public renderInput() {
return css.cssSelect( return css.cssSelect(
{tabIndex: "-1"}, {tabIndex: "-1"},
ignoreClick, ignoreClick,
dom.prop('name', this.model.colId), dom.prop('name', this.model.colId),
dom.forEach(this.choices, (choice) => dom('option', String(choice[1] ?? ''), {value: String(choice[0])})), dom.forEach(this.withEmpty, (choice) => dom('option', String(choice[1] ?? ''), {value: String(choice[0])})),
); );
} }
} }

View File

@ -3,7 +3,7 @@ import * as commands from 'app/client/components/commands';
import {Cursor} from 'app/client/components/Cursor'; import {Cursor} from 'app/client/components/Cursor';
import * as components from 'app/client/components/Forms/elements'; import * as components from 'app/client/components/Forms/elements';
import {NewBox} from 'app/client/components/Forms/Menu'; import {NewBox} from 'app/client/components/Forms/Menu';
import {Box, BoxModel, BoxType, LayoutModel, parseBox, Place} from 'app/client/components/Forms/Model'; import {BoxModel, LayoutModel, parseBox, Place} from 'app/client/components/Forms/Model';
import * as style from 'app/client/components/Forms/styles'; import * as style from 'app/client/components/Forms/styles';
import {GristDoc} from 'app/client/components/GristDoc'; import {GristDoc} from 'app/client/components/GristDoc';
import {copyToClipboard} from 'app/client/lib/clipboardUtils'; import {copyToClipboard} from 'app/client/lib/clipboardUtils';
@ -20,7 +20,7 @@ import {showTransientTooltip} from 'app/client/ui/tooltips';
import {cssButton} from 'app/client/ui2018/buttons'; import {cssButton} from 'app/client/ui2018/buttons';
import {icon} from 'app/client/ui2018/icons'; import {icon} from 'app/client/ui2018/icons';
import {confirmModal} from 'app/client/ui2018/modals'; import {confirmModal} from 'app/client/ui2018/modals';
import {INITIAL_FIELDS_COUNT} from "app/common/Forms"; import {Box, BoxType, INITIAL_FIELDS_COUNT} from "app/common/Forms";
import {Events as BackboneEvents} from 'backbone'; import {Events as BackboneEvents} from 'backbone';
import {Computed, dom, Holder, IDomArgs, Observable} from 'grainjs'; import {Computed, dom, Holder, IDomArgs, Observable} from 'grainjs';
import defaults from 'lodash/defaults'; import defaults from 'lodash/defaults';

View File

@ -1,12 +1,13 @@
import {allCommands} from 'app/client/components/commands'; import {allCommands} from 'app/client/components/commands';
import * as components from 'app/client/components/Forms/elements';
import {FormView} from 'app/client/components/Forms/FormView'; import {FormView} from 'app/client/components/Forms/FormView';
import {BoxModel, BoxType, Place} from 'app/client/components/Forms/Model'; import {BoxModel, Place} from 'app/client/components/Forms/Model';
import {makeTestId, stopEvent} from 'app/client/lib/domUtils'; import {makeTestId, stopEvent} from 'app/client/lib/domUtils';
import {FocusLayer} from 'app/client/lib/FocusLayer'; import {FocusLayer} from 'app/client/lib/FocusLayer';
import {makeT} from 'app/client/lib/localization'; import {makeT} from 'app/client/lib/localization';
import {getColumnTypes as getNewColumnTypes} from 'app/client/ui/GridViewMenus'; import {getColumnTypes as getNewColumnTypes} from 'app/client/ui/GridViewMenus';
import * as menus from 'app/client/ui2018/menus'; import * as menus from 'app/client/ui2018/menus';
import * as components from 'app/client/components/Forms/elements'; import {BoxType} from 'app/common/Forms';
import {Computed, dom, IDomArgs, MultiHolder} from 'grainjs'; import {Computed, dom, IDomArgs, MultiHolder} from 'grainjs';
const t = makeT('FormView'); const t = makeT('FormView');
@ -140,8 +141,9 @@ export function buildMenu(props: Props, ...args: IDomArgs<HTMLElement>): IDomArg
]), ]),
menus.menuDivider(), menus.menuDivider(),
menus.menuSubHeader(t('Building blocks')), menus.menuSubHeader(t('Building blocks')),
menus.menuItem(where(struct('Columns')), menus.menuIcon('Columns'), t("Columns")), menus.menuItem(where(struct('Header')), menus.menuIcon('Headband'), t("Header")),
menus.menuItem(where(struct('Paragraph')), menus.menuIcon('Paragraph'), t("Paragraph")), menus.menuItem(where(struct('Paragraph')), menus.menuIcon('Paragraph'), t("Paragraph")),
menus.menuItem(where(struct('Columns')), menus.menuIcon('Columns'), t("Columns")),
menus.menuItem(where(struct('Separator')), menus.menuIcon('Separator'), t("Separator")), menus.menuItem(where(struct('Separator')), menus.menuIcon('Separator'), t("Separator")),
]; ];
}; };

View File

@ -1,22 +1,10 @@
import * as elements from 'app/client/components/Forms/elements'; import * as elements from 'app/client/components/Forms/elements';
import {FormView} from 'app/client/components/Forms/FormView'; import {FormView} from 'app/client/components/Forms/FormView';
import {Box, BoxType} from 'app/common/Forms';
import {bundleChanges, Computed, Disposable, dom, IDomArgs, MutableObsArray, obsArray, Observable} from 'grainjs'; import {bundleChanges, Computed, Disposable, dom, IDomArgs, MutableObsArray, obsArray, Observable} from 'grainjs';
import {v4 as uuidv4} from 'uuid'; import {v4 as uuidv4} from 'uuid';
type Callback = () => Promise<void>; type Callback = () => Promise<void>;
export type BoxType = 'Paragraph' | 'Section' | 'Columns' | 'Submit'
| 'Placeholder' | 'Layout' | 'Field' | 'Label'
| 'Separator'
;
/**
* Box model is a JSON that represents a form element. Every element can be converted to this element and every
* ViewModel should be able to read it and built itself from it.
*/
export interface Box extends Record<string, any> {
type: BoxType,
children?: Array<Box>,
}
/** /**
* A place where to insert a box. * A place where to insert a box.
@ -59,10 +47,6 @@ export abstract class BoxModel extends Disposable {
* List of children boxes. * List of children boxes.
*/ */
public children: MutableObsArray<BoxModel>; public children: MutableObsArray<BoxModel>;
/**
* Any other dynamically added properties (that are not concrete fields in the derived classes)
*/
public props: Record<string, Observable<any>> = {};
/** /**
* Publicly exposed state if the element was just cut. * Publicly exposed state if the element was just cut.
* TODO: this should be moved to FormView, as this model doesn't care about that. * TODO: this should be moved to FormView, as this model doesn't care about that.
@ -70,6 +54,11 @@ export abstract class BoxModel extends Disposable {
public cut = Observable.create(this, false); public cut = Observable.create(this, false);
public selected: Observable<boolean>; public selected: Observable<boolean>;
/**
* Any other dynamically added properties (that are not concrete fields in the derived classes)
*/
private _props: Record<string, Observable<any>> = {};
/** /**
* Don't use it directly, use the BoxModel.new factory method instead. * Don't use it directly, use the BoxModel.new factory method instead.
*/ */
@ -163,7 +152,7 @@ export abstract class BoxModel extends Disposable {
} }
// We need to remove it from the parent, so find it first. // We need to remove it from the parent, so find it first.
const droppedId = dropped.id; const droppedId = dropped.id;
const droppedRef = this.root().get(droppedId); const droppedRef = droppedId ? this.root().get(droppedId) : null;
if (droppedRef) { if (droppedRef) {
droppedRef.removeSelf(); droppedRef.removeSelf();
} }
@ -171,14 +160,14 @@ export abstract class BoxModel extends Disposable {
} }
public prop(name: string, defaultValue?: any) { public prop(name: string, defaultValue?: any) {
if (!this.props[name]) { if (!this._props[name]) {
this.props[name] = Observable.create(this, defaultValue ?? null); this._props[name] = Observable.create(this, defaultValue ?? null);
} }
return this.props[name]; return this._props[name];
} }
public hasProp(name: string) { public hasProp(name: string) {
return this.props.hasOwnProperty(name); return this._props.hasOwnProperty(name);
} }
public async save(before?: () => Promise<void>): Promise<void> { public async save(before?: () => Promise<void>): Promise<void> {
@ -306,7 +295,8 @@ export abstract class BoxModel extends Disposable {
} }
// Update all properties of self. // Update all properties of self.
for (const key in boxDef) { for (const someKey in boxDef) {
const key = someKey as keyof Box;
// Skip some keys. // Skip some keys.
if (key === 'id' || key === 'type' || key === 'children') { continue; } if (key === 'id' || key === 'type' || key === 'children') { continue; }
// Skip any inherited properties. // Skip any inherited properties.
@ -347,7 +337,7 @@ export abstract class BoxModel extends Disposable {
id: this.id, id: this.id,
type: this.type, type: this.type,
children: this.children.get().map(child => child?.toJSON() || null), children: this.children.get().map(child => child?.toJSON() || null),
...(Object.fromEntries(Object.entries(this.props).map(([key, val]) => [key, val.get()]))), ...(Object.fromEntries(Object.entries(this._props).map(([key, val]) => [key, val.get()]))),
}; };
} }

View File

@ -1,9 +1,10 @@
import * as style from './styles'; import * as style from './styles';
import {buildEditor} from 'app/client/components/Forms/Editor'; import {buildEditor} from 'app/client/components/Forms/Editor';
import {buildMenu} from 'app/client/components/Forms/Menu'; import {buildMenu} from 'app/client/components/Forms/Menu';
import {Box, BoxModel} from 'app/client/components/Forms/Model'; import {BoxModel} from 'app/client/components/Forms/Model';
import {dom, styled} from 'grainjs';
import {makeTestId} from 'app/client/lib/domUtils'; import {makeTestId} from 'app/client/lib/domUtils';
import {Box} from 'app/common/Forms';
import {dom, styled} from 'grainjs';
const testId = makeTestId('test-forms-'); const testId = makeTestId('test-forms-');
@ -53,8 +54,7 @@ export class SectionModel extends BoxModel {
return null; return null;
} }
// We need to remove it from the parent, so find it first. // We need to remove it from the parent, so find it first.
const droppedId = dropped.id; const droppedRef = dropped.id ? this.root().get(dropped.id) : null;
const droppedRef = this.root().get(droppedId);
if (droppedRef) { if (droppedRef) {
droppedRef.removeSelf(); droppedRef.removeSelf();
} }

View File

@ -1,5 +1,5 @@
import {Columns, Placeholder} from 'app/client/components/Forms/Columns'; import {Columns, Paragraph, Placeholder} from 'app/client/components/Forms/Columns';
import {Box, BoxType} from 'app/client/components/Forms/Model'; import {Box, BoxType} from 'app/common/Forms';
/** /**
* Add any other element you whish to use in the form here. * Add any other element you whish to use in the form here.
* FormView will look for any exported BoxModel derived class in format `type` + `Model`, and use It * FormView will look for any exported BoxModel derived class in format `type` + `Model`, and use It
@ -16,10 +16,8 @@ export function defaultElement(type: BoxType): Box {
switch(type) { switch(type) {
case 'Columns': return Columns(); case 'Columns': return Columns();
case 'Placeholder': return Placeholder(); case 'Placeholder': return Placeholder();
case 'Separator': return { case 'Separator': return Paragraph('---');
type: 'Paragraph', case 'Header': return Paragraph('## **Header**', 'center');
text: '---',
};
default: return {type}; default: return {type};
} }
} }

View File

@ -33,7 +33,3 @@ export function PERMITTED_CUSTOM_WIDGETS(): Observable<string[]> {
} }
return G.window.PERMITTED_CUSTOM_WIDGETS; return G.window.PERMITTED_CUSTOM_WIDGETS;
} }
export function GRIST_FORMS_FEATURE() {
return Boolean(getGristConfig().experimentalPlugins);
}

View File

@ -3,7 +3,7 @@ import {GristDoc} from 'app/client/components/GristDoc';
import {makeT} from 'app/client/lib/localization'; import {makeT} from 'app/client/lib/localization';
import {reportError} from 'app/client/models/AppModel'; import {reportError} from 'app/client/models/AppModel';
import {ColumnRec, TableRec, ViewSectionRec} from 'app/client/models/DocModel'; import {ColumnRec, TableRec, ViewSectionRec} from 'app/client/models/DocModel';
import {GRIST_FORMS_FEATURE, PERMITTED_CUSTOM_WIDGETS} from "app/client/models/features"; import {PERMITTED_CUSTOM_WIDGETS} from "app/client/models/features";
import {GristTooltips} from 'app/client/ui/GristTooltips'; import {GristTooltips} from 'app/client/ui/GristTooltips';
import {linkId, NoLink} from 'app/client/ui/selectBy'; import {linkId, NoLink} from 'app/client/ui/selectBy';
import {overflowTooltip, withInfoTooltip} from 'app/client/ui/tooltips'; import {overflowTooltip, withInfoTooltip} from 'app/client/ui/tooltips';
@ -98,21 +98,17 @@ export interface IOptions extends ISelectOptions {
const testId = makeTestId('test-wselect-'); const testId = makeTestId('test-wselect-');
function maybeForms(): Array<'form'> {
return GRIST_FORMS_FEATURE() ? ['form'] : [];
}
// The picker disables some choices that do not make much sense. This function return the list of // The picker disables some choices that do not make much sense. This function return the list of
// compatible types given the tableId and whether user is creating a new page or not. // compatible types given the tableId and whether user is creating a new page or not.
function getCompatibleTypes(tableId: TableRef, isNewPage: boolean|undefined): IWidgetType[] { function getCompatibleTypes(tableId: TableRef, isNewPage: boolean|undefined): IWidgetType[] {
if (tableId !== 'New Table') { if (tableId !== 'New Table') {
return ['record', 'single', 'detail', 'chart', 'custom', 'custom.calendar', ...maybeForms()]; return ['record', 'single', 'detail', 'chart', 'custom', 'custom.calendar', 'form'];
} else if (isNewPage) { } else if (isNewPage) {
// New view + new table means we'll be switching to the primary view. // New view + new table means we'll be switching to the primary view.
return ['record', ...maybeForms()]; return ['record', 'form'];
} else { } else {
// The type 'chart' makes little sense when creating a new table. // The type 'chart' makes little sense when creating a new table.
return ['record', 'single', 'detail', ...maybeForms()]; return ['record', 'single', 'detail', 'form'];
} }
} }
@ -275,7 +271,7 @@ const permittedCustomWidgets: IAttachedCustomWidget[] = PERMITTED_CUSTOM_WIDGETS
const finalListOfCustomWidgetToShow = permittedCustomWidgets.filter(a=> const finalListOfCustomWidgetToShow = permittedCustomWidgets.filter(a=>
registeredCustomWidgets.includes(a)); registeredCustomWidgets.includes(a));
const sectionTypes: IWidgetType[] = [ const sectionTypes: IWidgetType[] = [
'record', 'single', 'detail', ...maybeForms(), 'chart', ...finalListOfCustomWidgetToShow, 'custom' 'record', 'single', 'detail', 'form', 'chart', ...finalListOfCustomWidgetToShow, 'custom'
]; ];

View File

@ -67,7 +67,8 @@ import {
MultiHolder, MultiHolder,
Observable, Observable,
styled, styled,
subscribe subscribe,
toKo
} from 'grainjs'; } from 'grainjs';
import * as ko from 'knockout'; import * as ko from 'knockout';
@ -955,12 +956,25 @@ export class RightPanel extends Disposable {
return vsi && vsi.activeFieldBuilder(); return vsi && vsi.activeFieldBuilder();
})); }));
const formView = owner.autoDispose(ko.computed(() => { // Sorry for the acrobatics below, but grainjs are not reentred when the active section changes.
const viewInstance = owner.autoDispose(ko.computed(() => {
const vsi = this._gristDoc.viewModel.activeSection?.().viewInstance(); const vsi = this._gristDoc.viewModel.activeSection?.().viewInstance();
return (vsi ?? null) as FormView|null; if (!vsi || vsi.isDisposed() || !toKo(ko, this._isForm)) { return null; }
return vsi;
})); }));
const selectedBox = Computed.create(owner, (use) => use(formView) && use(use(formView)!.selectedBox)); const formView = owner.autoDispose(ko.computed(() => {
const view = viewInstance() as unknown as FormView;
if (!view || !view.selectedBox) { return null; }
return view;
}));
const selectedBox = owner.autoDispose(ko.pureComputed(() => {
const view = formView();
if (!view) { return null; }
const box = toKo(ko, view.selectedBox)();
return box;
}));
const selectedField = Computed.create(owner, (use) => { const selectedField = Computed.create(owner, (use) => {
const box = use(selectedBox); const box = use(selectedBox);
if (!box) { return null; } if (!box) { return null; }
@ -983,33 +997,38 @@ export class RightPanel extends Disposable {
} }
}); });
return cssSection( return domAsync(imports.loadViewPane().then(() => buildConfigContainer(cssSection(
// Field config. // Field config.
dom.maybe(selectedField, (field) => { dom.maybeOwned(selectedField, (scope, field) => {
const requiredField = field.widgetOptionsJson.prop('formRequired'); const requiredField = field.widgetOptionsJson.prop('formRequired');
// V2 thing. // V2 thing.
// const hiddenField = field.widgetOptionsJson.prop('formHidden'); // const hiddenField = field.widgetOptionsJson.prop('formHidden');
const defaultField = field.widgetOptionsJson.prop('formDefault'); const defaultField = field.widgetOptionsJson.prop('formDefault');
const toComputed = (obs: typeof defaultField) => { const toComputed = (obs: typeof defaultField) => {
const result = Computed.create(null, (use) => use(obs)); const result = Computed.create(scope, (use) => use(obs));
result.onWrite(val => obs.setAndSave(val)); result.onWrite(val => obs.setAndSave(val));
return result; return result;
}; };
const fieldTitle = field.widgetOptionsJson.prop('question');
return [ return [
cssLabel(t("Field title")), cssLabel(t("Field title")),
cssRow( cssRow(
cssTextInput( cssTextInput(
fromKo(field.label), fromKo(fieldTitle),
(val) => field.displayLabel.saveOnly(val), (val) => fieldTitle.saveOnly(val).catch(reportError),
dom.prop('readonly', use => use(field.disableModify)), dom.prop('readonly', use => use(field.disableModify)),
dom.prop('placeholder', use => use(field.displayLabel) || use(field.colId)),
testId('field-title'),
), ),
), ),
cssLabel(t("Table column name")), cssLabel(t("Table column name")),
cssRow( cssRow(
cssTextInput( cssTextInput(
fromKo(field.colId), fromKo(field.displayLabel),
(val) => field.column().colId.saveOnly(val), (val) => field.displayLabel.saveOnly(val).catch(reportError),
dom.prop('readonly', use => use(field.disableModify)), dom.prop('readonly', use => use(field.disableModify)),
testId('field-label'),
), ),
), ),
// TODO: this is for V1 as it requires full cell editor here. // TODO: this is for V1 as it requires full cell editor here.
@ -1038,7 +1057,11 @@ export class RightPanel extends Disposable {
]), ]),
cssSeparator(), cssSeparator(),
cssLabel(t("Field rules")), cssLabel(t("Field rules")),
cssRow(labeledSquareCheckbox(toComputed(requiredField), t("Required field")),), cssRow(labeledSquareCheckbox(
toComputed(requiredField),
t("Required field"),
testId('field-required'),
)),
// V2 thing // V2 thing
// cssRow(labeledSquareCheckbox(toComputed(hiddenField), t("Hidden field")),), // cssRow(labeledSquareCheckbox(toComputed(hiddenField), t("Hidden field")),),
]; ];
@ -1071,7 +1094,7 @@ export class RightPanel extends Disposable {
dom.maybe(u => !u(selectedColumn) && !u(selectedBox), () => [ dom.maybe(u => !u(selectedColumn) && !u(selectedBox), () => [
cssLabel(t('Layout')), cssLabel(t('Layout')),
]) ])
); ))));
} }
} }

View File

@ -2,7 +2,6 @@ import {hooks} from 'app/client/Hooks';
import {makeT} from 'app/client/lib/localization'; import {makeT} from 'app/client/lib/localization';
import {allCommands} from 'app/client/components/commands'; import {allCommands} from 'app/client/components/commands';
import {ViewSectionRec} from 'app/client/models/DocModel'; import {ViewSectionRec} from 'app/client/models/DocModel';
import {GRIST_FORMS_FEATURE} from 'app/client/models/features';
import {urlState} from 'app/client/models/gristUrlState'; import {urlState} from 'app/client/models/gristUrlState';
import {testId} from 'app/client/ui2018/cssVars'; import {testId} from 'app/client/ui2018/cssVars';
import {menuDivider, menuItemCmd, menuItemLink} from 'app/client/ui2018/menus'; import {menuDivider, menuItemCmd, menuItemLink} from 'app/client/ui2018/menus';
@ -96,7 +95,7 @@ export function makeViewLayoutMenu(viewSection: ViewSectionRec, isReadonly: bool
menuItemCmd(allCommands.viewTabOpen, t("Widget options"), testId('widget-options')), menuItemCmd(allCommands.viewTabOpen, t("Widget options"), testId('widget-options')),
menuItemCmd(allCommands.sortFilterTabOpen, t("Advanced Sort & Filter"), dom.hide(viewSection.isRecordCard)), menuItemCmd(allCommands.sortFilterTabOpen, t("Advanced Sort & Filter"), dom.hide(viewSection.isRecordCard)),
menuItemCmd(allCommands.dataSelectionTabOpen, t("Data selection"), dom.hide(viewSection.isRecordCard)), menuItemCmd(allCommands.dataSelectionTabOpen, t("Data selection"), dom.hide(viewSection.isRecordCard)),
!GRIST_FORMS_FEATURE() ? null : menuItemCmd(allCommands.createForm, t("Create a form"), dom.show(isTable)), menuItemCmd(allCommands.createForm, t("Create a form"), dom.show(isTable)),
]), ]),
menuDivider(dom.hide(viewSection.isRecordCard)), menuDivider(dom.hide(viewSection.isRecordCard)),

View File

@ -79,6 +79,7 @@ export type IconName = "ChartArea" |
"FunctionResult" | "FunctionResult" |
"GreenArrow" | "GreenArrow" |
"Grow" | "Grow" |
"Headband" |
"Heart" | "Heart" |
"Help" | "Help" |
"Home" | "Home" |
@ -234,6 +235,7 @@ export const IconList: IconName[] = ["ChartArea",
"FunctionResult", "FunctionResult",
"GreenArrow", "GreenArrow",
"Grow", "Grow",
"Headband",
"Heart", "Heart",
"Help", "Help",
"Home", "Home",

View File

@ -12,8 +12,10 @@ import {marked} from 'marked';
/** /**
* All allowed boxes. * All allowed boxes.
*/ */
export type BoxType = 'Paragraph' | 'Section' | 'Columns' | 'Submit' | 'Placeholder' | 'Layout' | 'Field' | export type BoxType = 'Paragraph' | 'Section' | 'Columns' | 'Submit'
'Label'; | 'Placeholder' | 'Layout' | 'Field' | 'Label'
| 'Separator' | 'Header'
;
/** /**
* Number of fields to show in the form by default. * Number of fields to show in the form by default.
@ -24,7 +26,7 @@ export const INITIAL_FIELDS_COUNT = 9;
* Box model is a JSON that represents a form element. Every element can be converted to this element and every * Box model is a JSON that represents a form element. Every element can be converted to this element and every
* ViewModel should be able to read it and built itself from it. * ViewModel should be able to read it and built itself from it.
*/ */
export interface Box extends Record<string, any> { export interface Box {
type: BoxType, type: BoxType,
children?: Array<Box>, children?: Array<Box>,
@ -33,6 +35,18 @@ export interface Box extends Record<string, any> {
successURL?: string, successURL?: string,
successText?: string, successText?: string,
anotherResponse?: boolean, anotherResponse?: boolean,
// Unique ID of the field, used only in UI.
id?: string,
// Some properties used by fields and stored in the column/field.
formRequired?: boolean,
// Used by Label and Paragraph.
text?: string,
// Used by Paragraph.
alignment?: string,
// Used by Field.
leaf?: number,
} }
/** /**
@ -83,10 +97,9 @@ export class RenderBox {
class Label extends RenderBox { class Label extends RenderBox {
public override async toHTML() { public override async toHTML() {
const text = this.box['text']; const text = this.box.text || '';
const cssClass = this.box['cssClass'] || '';
return ` return `
<div class="grist-label ${cssClass}">${text || ''}</div> <div class="grist-label">${text || ''}</div>
`; `;
} }
} }
@ -160,7 +173,7 @@ class Field extends RenderBox {
} }
public async toHTML() { public async toHTML() {
const field = this.ctx.field(this.box['leaf']); const field = this.box.leaf ? this.ctx.field(this.box.leaf) : null;
if (!field) { if (!field) {
return `<div class="grist-field">Field not found</div>`; return `<div class="grist-field">Field not found</div>`;
} }
@ -232,6 +245,8 @@ class Choice extends BaseQuestion {
public input(field: FieldModel, context: RenderContext): string { public input(field: FieldModel, context: RenderContext): string {
const required = field.options.formRequired ? 'required' : ''; const required = field.options.formRequired ? 'required' : '';
const choices: string[] = field.options.choices || []; const choices: string[] = field.options.choices || [];
// Insert empty option.
choices.unshift('');
return ` return `
<select name='${field.colId}' ${required} > <select name='${field.colId}' ${required} >
${choices.map((choice) => `<option value='${choice}'>${choice}</option>`).join('')} ${choices.map((choice) => `<option value='${choice}'>${choice}</option>`).join('')}
@ -272,7 +287,7 @@ class ChoiceList extends BaseQuestion {
const required = field.options.formRequired ? 'required' : ''; const required = field.options.formRequired ? 'required' : '';
const choices: string[] = field.options.choices || []; const choices: string[] = field.options.choices || [];
return ` return `
<div name='${field.colId}' class='grist-choice-list ${required}'> <div name='${field.colId}' class='grist-choice-list grist-checkbox-list ${required}'>
${choices.map((choice) => ` ${choices.map((choice) => `
<label> <label>
<input type='checkbox' name='${field.colId}[]' value='${choice}' /> <input type='checkbox' name='${field.colId}[]' value='${choice}' />
@ -288,16 +303,20 @@ class ChoiceList extends BaseQuestion {
class RefList extends BaseQuestion { class RefList extends BaseQuestion {
public async input(field: FieldModel, context: RenderContext) { public async input(field: FieldModel, context: RenderContext) {
const required = field.options.formRequired ? 'required' : '';
const choices: [number, CellValue][] = (await field.values()) ?? []; const choices: [number, CellValue][] = (await field.values()) ?? [];
// Sort by the second value, which is the display value. // Sort by the second value, which is the display value.
choices.sort((a, b) => String(a[1]).localeCompare(String(b[1]))); choices.sort((a, b) => String(a[1]).localeCompare(String(b[1])));
// Support for 20 choices, TODO: make it dynamic. // Support for 30 choices, TODO: make it dynamic.
choices.splice(20); choices.splice(30);
return ` return `
<div name='${field.colId}' class='grist-ref-list'> <div name='${field.colId}' class='grist-ref-list grist-checkbox-list ${required}'>
${choices.map((choice) => ` ${choices.map((choice) => `
<label class='grist-checkbox'> <label class='grist-checkbox'>
<input type='checkbox' name='${field.colId}[]' value='${String(choice[0])}' /> <input type='checkbox'
data-grist-type='${field.type}'
name='${field.colId}[]'
value='${String(choice[0])}' />
<span> <span>
${String(choice[1] ?? '')} ${String(choice[1] ?? '')}
</span> </span>
@ -310,14 +329,17 @@ class RefList extends BaseQuestion {
class Ref extends BaseQuestion { class Ref extends BaseQuestion {
public async input(field: FieldModel) { public async input(field: FieldModel) {
const choices: [number, CellValue][] = (await field.values()) ?? []; const choices: [number|string, CellValue][] = (await field.values()) ?? [];
// Sort by the second value, which is the display value. // Sort by the second value, which is the display value.
choices.sort((a, b) => String(a[1]).localeCompare(String(b[1]))); choices.sort((a, b) => String(a[1]).localeCompare(String(b[1])));
// Support for 1000 choices, TODO: make it dynamic. // Support for 1000 choices, TODO: make it dynamic.
choices.splice(1000); choices.splice(1000);
// Insert empty option.
choices.unshift(['', '']);
// <option type='number' is not standard, we parse it ourselves. // <option type='number' is not standard, we parse it ourselves.
const required = field.options.formRequired ? 'required' : '';
return ` return `
<select name='${field.colId}' class='grist-ref' data-grist-type='${field.type}'> <select name='${field.colId}' class='grist-ref' data-grist-type='${field.type}' ${required}>
${choices.map((choice) => `<option value='${String(choice[0])}'>${String(choice[1] ?? '')}</option>`).join('')} ${choices.map((choice) => `<option value='${String(choice[0])}'>${String(choice[1] ?? '')}</option>`).join('')}
</select> </select>
`; `;
@ -351,4 +373,8 @@ const elements = {
'Layout': Layout, 'Layout': Layout,
'Field': Field, 'Field': Field,
'Label': Label, 'Label': Label,
// Those are just aliases for Paragraph.
'Separator': Paragraph,
'Header': Paragraph,
}; };

View File

@ -166,6 +166,7 @@ handlebars.registerHelper('dompurify', (html: string) => {
return new handlebars.SafeString(` return new handlebars.SafeString(`
<script data-html="${handlebars.escapeExpression(html)}"> <script data-html="${handlebars.escapeExpression(html)}">
document.write(DOMPurify.sanitize(document.currentScript.getAttribute('data-html'))); document.write(DOMPurify.sanitize(document.currentScript.getAttribute('data-html')));
document.currentScript.remove(); // remove the script tag so it is easier to inspect the DOM
</script> </script>
`); `);
}); });

19
static/forms/README.md Normal file
View File

@ -0,0 +1,19 @@
## grist-form-submit.js
File is taken from https://github.com/gristlabs/grist-form-submit. But it is modified to work with
forms, especially for:
- Ref and RefList columns, as by default it sends numbers as strings (FormData issue), and Grist
doesn't know how to convert them back to numbers.
- Empty strings are not sent at all - otherwise Grist won't be able to fire trigger formulas
correctly and provide default values for columns.
- By default it requires a redirect URL, now it is optional.
## purify.min.js
File taken from https://www.npmjs.com/package/dompurify. It is used to sanitize HTML. It wasn't
modified at all.
## form.html
This is handlebars template filled by DocApi.ts

View File

@ -9,6 +9,15 @@
</style> </style>
<script src="forms/grist-form-submit.js"></script> <script src="forms/grist-form-submit.js"></script>
<script src="forms/purify.min.js"></script> <script src="forms/purify.min.js"></script>
<script>
// Make all links open in a new tab.
DOMPurify.addHook('uponSanitizeAttribute', (node) => {
if (!('target' in node)) { return; }
node.setAttribute('target', '_blank');
// Make sure that this is set explicitly, as it's often set by the browser.
node.setAttribute('rel', 'noopener');
});
</script>
<link rel="stylesheet" href="forms/form.css"> <link rel="stylesheet" href="forms/form.css">
<meta name="viewport" content="width=device-width, initial-scale=1"> <meta name="viewport" content="width=device-width, initial-scale=1">
</head> </head>
@ -43,7 +52,7 @@
document.querySelector('.grist-form input[type="submit"]').addEventListener('click', function(event) { document.querySelector('.grist-form input[type="submit"]').addEventListener('click', function(event) {
// When submit is pressed make sure that all choice lists that are required // When submit is pressed make sure that all choice lists that are required
// have at least one option selected // have at least one option selected
const choiceLists = document.querySelectorAll('.grist-choice-list.required:not(:has(input:checked))'); const choiceLists = document.querySelectorAll('.grist-checkbox-list.required:not(:has(input:checked))');
Array.from(choiceLists).forEach(function(choiceList) { Array.from(choiceLists).forEach(function(choiceList) {
// If the form has at least one checkbox make it required // If the form has at least one checkbox make it required
const firstCheckbox = choiceList.querySelector('input[type="checkbox"]'); const firstCheckbox = choiceList.querySelector('input[type="checkbox"]');
@ -51,7 +60,7 @@
}); });
// All other required choice lists with at least one option selected are no longer required // All other required choice lists with at least one option selected are no longer required
const choiceListsRequired = document.querySelectorAll('.grist-choice-list.required:has(input:checked)'); const choiceListsRequired = document.querySelectorAll('.grist-checkbox-list.required:has(input:checked)');
Array.from(choiceListsRequired).forEach(function(choiceList) { Array.from(choiceListsRequired).forEach(function(choiceList) {
// If the form has at least one checkbox make it required // If the form has at least one checkbox make it required
const firstCheckbox = choiceList.querySelector('input[type="checkbox"]'); const firstCheckbox = choiceList.querySelector('input[type="checkbox"]');

View File

@ -69,7 +69,19 @@ class TypedFormData {
this._formData = formData ?? new FormData(formElement); this._formData = formData ?? new FormData(formElement);
this._formElement = formElement; this._formElement = formElement;
} }
keys() { return this._formData.keys(); } keys() {
const keys = Array.from(this._formData.keys());
// Don't return keys for scalar values which just return empty string.
// Otherwise Grist won't fire trigger formulas.
return keys.filter(key => {
// If there are multiple values, return this key as it is.
if (this._formData.getAll(key).length !== 1) { return true; }
// If the value is empty string or null, don't return the key.
const value = this._formData.get(key);
return value !== '' && value !== null;
});
}
type(key) { type(key) {
return this._formElement?.querySelector(`[name="${key}"]`)?.getAttribute('data-grist-type'); return this._formElement?.querySelector(`[name="${key}"]`)?.getAttribute('data-grist-type');
} }

View File

@ -80,6 +80,7 @@
--icon-FunctionResult: url(''); --icon-FunctionResult: url('');
--icon-GreenArrow: url(''); --icon-GreenArrow: url('');
--icon-Grow: url(''); --icon-Grow: url('');
--icon-Headband: url('');
--icon-Heart: url(''); --icon-Heart: url('');
--icon-Help: url(''); --icon-Help: url('');
--icon-Home: url(''); --icon-Home: url('');

View File

@ -0,0 +1,10 @@
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1061_7585)">
<path d="M14 16L8 11L2 16V0H14V16Z" fill="black"/>
</g>
<defs>
<clipPath id="clip0_1061_7585">
<rect width="16" height="16" fill="white"/>
</clipPath>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 300 B

View File

@ -60,6 +60,7 @@ describe('FormView', function() {
async function createFormWith(type: string, more = false) { async function createFormWith(type: string, more = false) {
await gu.addNewSection('Form', 'Table1'); await gu.addNewSection('Form', 'Table1');
// Make sure column D is not there.
assert.isUndefined(await api.getTable(docId, 'Table1').then(t => t.D)); assert.isUndefined(await api.getTable(docId, 'Table1').then(t => t.D));
// Add a text question // Add a text question
@ -117,10 +118,62 @@ describe('FormView', function() {
assert.deepEqual(await api.getTable(docId, 'Table1').then(t => t.D), [value]); assert.deepEqual(await api.getTable(docId, 'Table1').then(t => t.D), [value]);
} }
async function expect(values: any[]) { async function expectInD(values: any[]) {
assert.deepEqual(await api.getTable(docId, 'Table1').then(t => t.D), values); assert.deepEqual(await api.getTable(docId, 'Table1').then(t => t.D), values);
} }
it('updates creator panel when navigated away', async function() {
// Add 2 new pages.
await gu.addNewPage('Form', 'New Table', {tableName: 'TabA'});
await gu.renamePage('TabA');
await gu.addNewPage('Form', 'New Table', {tableName: 'TabB'});
// Open the creator panel on field tab
await gu.openColumnPanel();
// Select A column
await question('A').click();
// Make sure it is selected.
assert.equal(await selectedLabel(), 'A');
// And creator panel reflects it.
assert.equal(await driver.find('.test-field-label').value(), "A");
// Now switch to page TabA.
await gu.openPage('TabA');
// And select B column.
await question('B').click();
assert.equal(await selectedLabel(), 'B');
// Make sure creator panel reflects it (it didn't).
assert.equal(await driver.find('.test-field-label').value(), "B");
await gu.undo(2); // There was a bug with second undo.
await gu.undo();
});
it('triggers trigger formulas', async function() {
const formUrl = await createFormWith('Text');
// Add a trigger formula for this column.
await gu.showRawData();
await gu.getCell('D', 1).click();
await gu.openColumnPanel();
await driver.find(".test-field-set-trigger").click();
await gu.waitAppFocus(false);
await gu.sendKeys('"Hello from trigger"', Key.ENTER);
await gu.waitForServer();
await gu.closeRawTable();
await gu.onNewTab(async () => {
await driver.get(formUrl);
await driver.find('input[type="submit"]').click();
await waitForConfirm();
});
await expectSingle('Hello from trigger');
await removeForm();
});
it('can submit a form with Text field', async function() { it('can submit a form with Text field', async function() {
const formUrl = await createFormWith('Text'); const formUrl = await createFormWith('Text');
// We are in a new window. // We are in a new window.
@ -189,7 +242,7 @@ describe('FormView', function() {
await gu.onNewTab(async () => { await gu.onNewTab(async () => {
await driver.get(formUrl); await driver.get(formUrl);
// Make sure options are there. // Make sure options are there.
assert.deepEqual(await driver.findAll('select[name="D"] option', e => e.getText()), ['Foo', 'Bar', 'Baz']); assert.deepEqual(await driver.findAll('select[name="D"] option', e => e.getText()), ['', 'Foo', 'Bar', 'Baz']);
await driver.findWait('select[name="D"]', 1000).click(); await driver.findWait('select[name="D"]', 1000).click();
await driver.find("option[value='Bar']").click(); await driver.find("option[value='Bar']").click();
await driver.find('input[type="submit"]').click(); await driver.find('input[type="submit"]').click();
@ -229,7 +282,7 @@ describe('FormView', function() {
await driver.find('input[type="submit"]').click(); await driver.find('input[type="submit"]').click();
await waitForConfirm(); await waitForConfirm();
}); });
await expect([true, false]); await expectInD([true, false]);
// Remove the additional record added just now. // Remove the additional record added just now.
await gu.sendActions([ await gu.sendActions([
@ -262,6 +315,78 @@ describe('FormView', function() {
await removeForm(); await removeForm();
}); });
it('can submit a form with Ref field', async function() {
const formUrl = await createFormWith('Reference', true);
// Add some options.
await gu.openColumnPanel();
await gu.setRefShowColumn('A');
// Add 3 records to this table (it is now empty).
await gu.sendActions([
['AddRecord', 'Table1', null, {A: 'Foo'}], // id 1
['AddRecord', 'Table1', null, {A: 'Bar'}], // id 2
['AddRecord', 'Table1', null, {A: 'Baz'}], // id 3
]);
await gu.toggleSidePanel('right', 'close');
// We are in a new window.
await gu.onNewTab(async () => {
await driver.get(formUrl);
assert.deepEqual(
await driver.findAll('select[name="D"] option', e => e.getText()),
['', ...['Bar', 'Baz', 'Foo']]
);
assert.deepEqual(
await driver.findAll('select[name="D"] option', e => e.value()),
['', ...['2', '3', '1']]
);
await driver.findWait('select[name="D"]', 1000).click();
await driver.find('option[value="2"]').click();
await driver.find('input[type="submit"]').click();
await waitForConfirm();
});
await expectInD([0, 0, 0, 2]);
// Remove 3 records.
await gu.sendActions([
['BulkRemoveRecord', 'Table1', [1, 2, 3, 4]],
]);
await removeForm();
});
it('can submit a form with RefList field', async function() {
const formUrl = await createFormWith('Reference List', true);
// Add some options.
await gu.openColumnPanel();
await gu.setRefShowColumn('A');
// Add 3 records to this table (it is now empty).
await gu.sendActions([
['AddRecord', 'Table1', null, {A: 'Foo'}], // id 1
['AddRecord', 'Table1', null, {A: 'Bar'}], // id 2
['AddRecord', 'Table1', null, {A: 'Baz'}], // id 3
]);
await gu.toggleSidePanel('right', 'close');
// We are in a new window.
await gu.onNewTab(async () => {
await driver.get(formUrl);
await driver.findWait('input[name="D[]"][value="1"]', 1000).click();
await driver.findWait('input[name="D[]"][value="2"]', 1000).click();
assert.equal(await driver.find('.grist-checkbox:has(input[name="D[]"][value="1"])').getText(), 'Foo');
assert.equal(await driver.find('.grist-checkbox:has(input[name="D[]"][value="2"])').getText(), 'Bar');
assert.equal(await driver.find('.grist-checkbox:has(input[name="D[]"][value="3"])').getText(), 'Baz');
await driver.find('input[type="submit"]').click();
await waitForConfirm();
});
await expectInD([null, null, null, ['L', 2, 1]]);
// Remove 3 records.
await gu.sendActions([
['BulkRemoveRecord', 'Table1', [1, 2, 3, 4]],
]);
await removeForm();
});
it('can unpublish forms', async function() { it('can unpublish forms', async function() {
const formUrl = await createFormWith('Text'); const formUrl = await createFormWith('Text');
await driver.find('.test-forms-unpublish').click(); await driver.find('.test-forms-unpublish').click();

View File

@ -508,7 +508,7 @@ describe('RawData', function () {
await gu.sendKeys("abc"); await gu.sendKeys("abc");
await gu.checkTextEditor("abc"); await gu.checkTextEditor("abc");
await gu.sendKeys(Key.ESCAPE); await gu.sendKeys(Key.ESCAPE);
await showRawData(); await gu.showRawData();
assert.equal(await gu.getActiveSectionTitle(), 'City'); assert.equal(await gu.getActiveSectionTitle(), 'City');
assert.deepEqual(await gu.getCursorPosition(), {rowNum: 20, col: 0}); // raw popup is not sorted assert.deepEqual(await gu.getCursorPosition(), {rowNum: 20, col: 0}); // raw popup is not sorted
await gu.sendKeys("abc"); await gu.sendKeys("abc");
@ -530,7 +530,7 @@ describe('RawData', function () {
await gu.sendKeys(Key.ESCAPE); await gu.sendKeys(Key.ESCAPE);
// Now open popup again, but close it by clicking on the close button. // Now open popup again, but close it by clicking on the close button.
await showRawData(); await gu.showRawData();
await gu.closeRawTable(); await gu.closeRawTable();
await assertNoPopup(); await assertNoPopup();
assert.equal(await gu.getActiveSectionTitle(), 'CITY'); assert.equal(await gu.getActiveSectionTitle(), 'CITY');
@ -540,7 +540,7 @@ describe('RawData', function () {
await gu.sendKeys(Key.ESCAPE); await gu.sendKeys(Key.ESCAPE);
// Now do the same, but close by clicking on a diffrent page // Now do the same, but close by clicking on a diffrent page
await showRawData(); await gu.showRawData();
await gu.getPageItem('Country').click(); await gu.getPageItem('Country').click();
await assertNoPopup(); await assertNoPopup();
assert.equal(await gu.getActiveSectionTitle(), 'COUNTRY'); assert.equal(await gu.getActiveSectionTitle(), 'COUNTRY');
@ -552,7 +552,7 @@ describe('RawData', function () {
// Now make sure that raw data is available for card view. // Now make sure that raw data is available for card view.
await gu.selectSectionByTitle("COUNTRY Card List"); await gu.selectSectionByTitle("COUNTRY Card List");
assert.equal(await gu.getActiveSectionTitle(), 'COUNTRY Card List'); assert.equal(await gu.getActiveSectionTitle(), 'COUNTRY Card List');
await showRawData(); await gu.showRawData();
assert.equal(await gu.getActiveSectionTitle(), 'Country'); assert.equal(await gu.getActiveSectionTitle(), 'Country');
assert.deepEqual(await gu.getCursorPosition(), {rowNum: 1, col: 1}); assert.deepEqual(await gu.getCursorPosition(), {rowNum: 1, col: 1});
await gu.sendKeys("abc"); await gu.sendKeys("abc");
@ -623,7 +623,7 @@ describe('RawData', function () {
// Now open plain raw data for City table. // Now open plain raw data for City table.
await gu.selectSectionByTitle("CITY"); await gu.selectSectionByTitle("CITY");
assert.equal(await gu.getActiveSectionTitle(), 'CITY'); // CITY is viewSection title assert.equal(await gu.getActiveSectionTitle(), 'CITY'); // CITY is viewSection title
await showRawData(); await gu.showRawData();
assert.equal(await gu.getActiveSectionTitle(), 'City'); // City is now a table title assert.equal(await gu.getActiveSectionTitle(), 'City'); // City is now a table title
// Now remove the table. // Now remove the table.
await api.applyUserActions(doc, [[ await api.applyUserActions(doc, [[
@ -787,12 +787,6 @@ function replaceAnchor(link: string, values: {
return link.replace(anchorRegex, `#a${values.a || a}.s${values.s || s}.r${values.r || r}.c${values.c || c}`); return link.replace(anchorRegex, `#a${values.a || a}.s${values.s || s}.r${values.r || r}.c${values.c || c}`);
} }
async function showRawData() {
await gu.openSectionMenu('viewLayout');
await driver.find('.test-show-raw-data').click();
await waitForPopup();
}
async function openRawData() { async function openRawData() {
await driver.find('.test-tools-raw').click(); await driver.find('.test-tools-raw').click();
await waitForRawData(); await waitForRawData();

View File

@ -1188,7 +1188,12 @@ export async function changeWidget(type: WidgetType) {
/** /**
* Rename the given page to a new name. The oldName can be a full string name or a RegExp. * Rename the given page to a new name. The oldName can be a full string name or a RegExp.
*/ */
export async function renamePage(oldName: string|RegExp, newName: string) { export async function renamePage(oldName: string|RegExp, newName?: string) {
if (!newName && typeof oldName === 'string') {
newName = oldName;
oldName = await getCurrentPageName();
}
if (newName === undefined) { throw new Error('newName must be specified'); }
await openPageMenu(oldName); await openPageMenu(oldName);
await driver.find('.test-docpage-rename').click(); await driver.find('.test-docpage-rename').click();
await driver.find('.test-docpage-editor').sendKeys(newName, Key.ENTER); await driver.find('.test-docpage-editor').sendKeys(newName, Key.ENTER);
@ -1570,6 +1575,15 @@ export async function openSectionMenu(which: 'sortAndFilter'|'viewLayout', secti
return await driver.findWait('.grist-floating-menu', 100); return await driver.findWait('.grist-floating-menu', 100);
} }
/**
* Opens Raw data view for current section.
*/
export async function showRawData(section?: string|WebElement) {
await openSectionMenu('viewLayout', section);
await driver.find('.test-show-raw-data').click();
assert.isTrue(await driver.findWait('.test-raw-data-overlay', 100).isDisplayed());
}
// Mapping from column menu option name to dom element selector to wait for, or null if no need to wait. // Mapping from column menu option name to dom element selector to wait for, or null if no need to wait.
const ColumnMenuOption: { [id: string]: string; } = { const ColumnMenuOption: { [id: string]: string; } = {
Filter: '.test-filter-menu-wrapper' Filter: '.test-filter-menu-wrapper'