mirror of
				https://github.com/gristlabs/grist-core.git
				synced 2025-06-13 20:53:59 +00:00 
			
		
		
		
	(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:
		
							parent
							
								
									007c4492dc
								
							
						
					
					
						commit
						95c0441d84
					
				| @ -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'}; | ||||||
| } | } | ||||||
|  | |||||||
| @ -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 => { | ||||||
|     const field = this.model.field; |     // Read choices from field.
 | ||||||
|     const choices: Computed<string[]> = Computed.create(this, use => { |     const list = use(use(use(this.model.field).origCol).widgetOptionsJson.prop('choices')) || []; | ||||||
|       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.
 |     // Make sure it is array of strings.
 | ||||||
|       if (!Array.isArray(value) || value.some((v) => typeof v !== 'string')) { |     if (!Array.isArray(list) || list.some((v) => typeof v !== 'string')) { | ||||||
|       return []; |       return []; | ||||||
|     } |     } | ||||||
|       return value; |     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; | ||||||
|     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])})), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -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'; | ||||||
|  | |||||||
| @ -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")), | ||||||
|         ]; |         ]; | ||||||
|       }; |       }; | ||||||
|  | |||||||
| @ -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()]))), | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -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(); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -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}; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -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); |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -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' | ||||||
| ]; | ]; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -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')), | ||||||
|       ]) |       ]) | ||||||
|     ); |     )))); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -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)), | ||||||
|  | |||||||
| @ -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", | ||||||
|  | |||||||
| @ -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, | ||||||
| }; | }; | ||||||
|  | |||||||
| @ -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
									
								
							
							
						
						
									
										19
									
								
								static/forms/README.md
									
									
									
									
									
										Normal 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 | ||||||
| @ -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"]'); | ||||||
|  | |||||||
| @ -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'); | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -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(''); | ||||||
|  | |||||||
							
								
								
									
										10
									
								
								static/ui-icons/UI/Headband.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								static/ui-icons/UI/Headband.svg
									
									
									
									
									
										Normal 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 | 
| @ -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(); | ||||||
|  | |||||||
| @ -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(); | ||||||
|  | |||||||
| @ -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' | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user