@gravity-ui/page-constructor 8.5.0-alpha.1 → 8.5.0-alpha.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/cjs/containers/PageConstructor/PageConstructor.d.ts +0 -1
- package/build/cjs/containers/PageConstructor/PageConstructor.js +0 -1
- package/build/cjs/containers/PageConstructor/PageConstructor.js.map +1 -1
- package/build/cjs/form-builder-v2/CanvasContentContext.d.ts +18 -0
- package/build/cjs/form-builder-v2/CanvasContentContext.js +21 -0
- package/build/cjs/form-builder-v2/CanvasContentContext.js.map +1 -0
- package/build/cjs/form-builder-v2/FormBuilderV2.css +93 -0
- package/build/cjs/form-builder-v2/FormBuilderV2.d.ts +10 -0
- package/build/cjs/form-builder-v2/FormBuilderV2.js +123 -0
- package/build/cjs/form-builder-v2/FormBuilderV2.js.map +1 -0
- package/build/cjs/form-builder-v2/components/Canvas/Canvas.css +15 -0
- package/build/cjs/form-builder-v2/components/Canvas/Canvas.d.ts +8 -0
- package/build/cjs/form-builder-v2/components/Canvas/Canvas.js +17 -0
- package/build/cjs/form-builder-v2/components/Canvas/Canvas.js.map +1 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/CanvasCard.css +133 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/CanvasCard.d.ts +9 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/CanvasCard.js +72 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/CanvasCard.js.map +1 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/components/FieldPreview.d.ts +6 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/components/FieldPreview.js +44 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/components/FieldPreview.js.map +1 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/components/SectionChildrenDropZone.d.ts +9 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/components/SectionChildrenDropZone.js +20 -0
- package/build/cjs/form-builder-v2/components/CanvasCard/components/SectionChildrenDropZone.js.map +1 -0
- package/build/cjs/form-builder-v2/components/ContentTab/ContentTab.css +22 -0
- package/build/cjs/form-builder-v2/components/ContentTab/ContentTab.d.ts +1 -0
- package/build/cjs/form-builder-v2/components/ContentTab/ContentTab.js +21 -0
- package/build/cjs/form-builder-v2/components/ContentTab/ContentTab.js.map +1 -0
- package/build/cjs/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.css +13 -0
- package/build/cjs/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.d.ts +7 -0
- package/build/cjs/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.js +24 -0
- package/build/cjs/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/FieldSettings.css +30 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/FieldSettings.d.ts +6 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/FieldSettings.js +49 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/FieldSettings.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/ColorInputSettings.d.ts +12 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/ColorInputSettings.js +11 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/ColorInputSettings.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/OptionsSettings.d.ts +12 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/OptionsSettings.js +38 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/OptionsSettings.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/Row.d.ts +7 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/Row.js +9 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/Row.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/SectionSettings.d.ts +9 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/SectionSettings.js +40 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/SectionSettings.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/SwitchSettings.d.ts +12 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/SwitchSettings.js +11 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/SwitchSettings.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/TextFieldSettings.d.ts +12 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/TextFieldSettings.js +11 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/TextFieldSettings.js.map +1 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/TextSettings.d.ts +11 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/TextSettings.js +53 -0
- package/build/cjs/form-builder-v2/components/FieldSettings/fields/TextSettings.js.map +1 -0
- package/build/cjs/form-builder-v2/components/Inspector/Inspector.css +20 -0
- package/build/cjs/form-builder-v2/components/Inspector/Inspector.d.ts +1 -0
- package/build/cjs/form-builder-v2/components/Inspector/Inspector.js +20 -0
- package/build/cjs/form-builder-v2/components/Inspector/Inspector.js.map +1 -0
- package/build/cjs/form-builder-v2/components/Palette/Palette.css +56 -0
- package/build/cjs/form-builder-v2/components/Palette/Palette.d.ts +4 -0
- package/build/cjs/form-builder-v2/components/Palette/Palette.js +62 -0
- package/build/cjs/form-builder-v2/components/Palette/Palette.js.map +1 -0
- package/build/cjs/form-builder-v2/components/ResizeHandle/ResizeHandle.css +29 -0
- package/build/cjs/form-builder-v2/components/ResizeHandle/ResizeHandle.d.ts +9 -0
- package/build/cjs/form-builder-v2/components/ResizeHandle/ResizeHandle.js +98 -0
- package/build/cjs/form-builder-v2/components/ResizeHandle/ResizeHandle.js.map +1 -0
- package/build/cjs/form-builder-v2/components/SchemaPopup/SchemaPopup.css +49 -0
- package/build/cjs/form-builder-v2/components/SchemaPopup/SchemaPopup.d.ts +11 -0
- package/build/cjs/form-builder-v2/components/SchemaPopup/SchemaPopup.js +82 -0
- package/build/cjs/form-builder-v2/components/SchemaPopup/SchemaPopup.js.map +1 -0
- package/build/cjs/form-builder-v2/components/WhenEditor/WhenEditor.css +24 -0
- package/build/cjs/form-builder-v2/components/WhenEditor/WhenEditor.d.ts +8 -0
- package/build/cjs/form-builder-v2/components/WhenEditor/WhenEditor.js +50 -0
- package/build/cjs/form-builder-v2/components/WhenEditor/WhenEditor.js.map +1 -0
- package/build/cjs/form-builder-v2/components/WhenEditor/utils.d.ts +24 -0
- package/build/cjs/form-builder-v2/components/WhenEditor/utils.js +83 -0
- package/build/cjs/form-builder-v2/components/WhenEditor/utils.js.map +1 -0
- package/build/cjs/form-builder-v2/hooks/FormContext.d.ts +11 -0
- package/build/cjs/form-builder-v2/hooks/FormContext.js +22 -0
- package/build/cjs/form-builder-v2/hooks/FormContext.js.map +1 -0
- package/build/cjs/form-builder-v2/hooks/useFormFields.d.ts +23 -0
- package/build/cjs/form-builder-v2/hooks/useFormFields.js +144 -0
- package/build/cjs/form-builder-v2/hooks/useFormFields.js.map +1 -0
- package/build/cjs/form-builder-v2/index.d.ts +2 -0
- package/build/cjs/form-builder-v2/index.js +6 -0
- package/build/cjs/form-builder-v2/index.js.map +1 -0
- package/build/cjs/form-builder-v2/styles/variables.css +0 -0
- package/build/cjs/form-builder-v2/types.d.ts +32 -0
- package/build/cjs/form-builder-v2/types.js +3 -0
- package/build/cjs/form-builder-v2/types.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/cn.d.ts +2 -0
- package/build/cjs/form-builder-v2/utils/cn.js +7 -0
- package/build/cjs/form-builder-v2/utils/cn.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/dragData.d.ts +18 -0
- package/build/cjs/form-builder-v2/utils/dragData.js +18 -0
- package/build/cjs/form-builder-v2/utils/dragData.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/fieldDefaults.d.ts +2 -0
- package/build/cjs/form-builder-v2/utils/fieldDefaults.js +51 -0
- package/build/cjs/form-builder-v2/utils/fieldDefaults.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/fieldGroups.d.ts +3 -0
- package/build/cjs/form-builder-v2/utils/fieldGroups.js +26 -0
- package/build/cjs/form-builder-v2/utils/fieldGroups.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/fieldMeta.d.ts +6 -0
- package/build/cjs/form-builder-v2/utils/fieldMeta.js +45 -0
- package/build/cjs/form-builder-v2/utils/fieldMeta.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/fieldNames.d.ts +8 -0
- package/build/cjs/form-builder-v2/utils/fieldNames.js +75 -0
- package/build/cjs/form-builder-v2/utils/fieldNames.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/fieldTree.d.ts +2 -0
- package/build/cjs/form-builder-v2/utils/fieldTree.js +7 -0
- package/build/cjs/form-builder-v2/utils/fieldTree.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/fieldTreeOps.d.ts +13 -0
- package/build/cjs/form-builder-v2/utils/fieldTreeOps.js +80 -0
- package/build/cjs/form-builder-v2/utils/fieldTreeOps.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/parseSchema.d.ts +9 -0
- package/build/cjs/form-builder-v2/utils/parseSchema.js +102 -0
- package/build/cjs/form-builder-v2/utils/parseSchema.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/stripIds.d.ts +3 -0
- package/build/cjs/form-builder-v2/utils/stripIds.js +12 -0
- package/build/cjs/form-builder-v2/utils/stripIds.js.map +1 -0
- package/build/cjs/form-builder-v2/utils/treeWalk.d.ts +4 -0
- package/build/cjs/form-builder-v2/utils/treeWalk.js +45 -0
- package/build/cjs/form-builder-v2/utils/treeWalk.js.map +1 -0
- package/build/cjs/hooks/usePCEditorInitializeEvents.js +2 -2
- package/build/cjs/hooks/usePCEditorInitializeEvents.js.map +1 -1
- package/build/esm/containers/PageConstructor/PageConstructor.d.ts +0 -1
- package/build/esm/containers/PageConstructor/PageConstructor.js +0 -1
- package/build/esm/containers/PageConstructor/PageConstructor.js.map +1 -1
- package/build/esm/form-builder-v2/CanvasContentContext.d.ts +18 -0
- package/build/esm/form-builder-v2/CanvasContentContext.js +15 -0
- package/build/esm/form-builder-v2/CanvasContentContext.js.map +1 -0
- package/build/esm/form-builder-v2/FormBuilderV2.css +93 -0
- package/build/esm/form-builder-v2/FormBuilderV2.d.ts +11 -0
- package/build/esm/form-builder-v2/FormBuilderV2.js +119 -0
- package/build/esm/form-builder-v2/FormBuilderV2.js.map +1 -0
- package/build/esm/form-builder-v2/components/Canvas/Canvas.css +15 -0
- package/build/esm/form-builder-v2/components/Canvas/Canvas.d.ts +9 -0
- package/build/esm/form-builder-v2/components/Canvas/Canvas.js +14 -0
- package/build/esm/form-builder-v2/components/Canvas/Canvas.js.map +1 -0
- package/build/esm/form-builder-v2/components/CanvasCard/CanvasCard.css +133 -0
- package/build/esm/form-builder-v2/components/CanvasCard/CanvasCard.d.ts +10 -0
- package/build/esm/form-builder-v2/components/CanvasCard/CanvasCard.js +68 -0
- package/build/esm/form-builder-v2/components/CanvasCard/CanvasCard.js.map +1 -0
- package/build/esm/form-builder-v2/components/CanvasCard/components/FieldPreview.d.ts +6 -0
- package/build/esm/form-builder-v2/components/CanvasCard/components/FieldPreview.js +39 -0
- package/build/esm/form-builder-v2/components/CanvasCard/components/FieldPreview.js.map +1 -0
- package/build/esm/form-builder-v2/components/CanvasCard/components/SectionChildrenDropZone.d.ts +9 -0
- package/build/esm/form-builder-v2/components/CanvasCard/components/SectionChildrenDropZone.js +16 -0
- package/build/esm/form-builder-v2/components/CanvasCard/components/SectionChildrenDropZone.js.map +1 -0
- package/build/esm/form-builder-v2/components/ContentTab/ContentTab.css +22 -0
- package/build/esm/form-builder-v2/components/ContentTab/ContentTab.d.ts +2 -0
- package/build/esm/form-builder-v2/components/ContentTab/ContentTab.js +17 -0
- package/build/esm/form-builder-v2/components/ContentTab/ContentTab.js.map +1 -0
- package/build/esm/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.css +13 -0
- package/build/esm/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.d.ts +8 -0
- package/build/esm/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.js +21 -0
- package/build/esm/form-builder-v2/components/DragOverlayPreview/DragOverlayPreview.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/FieldSettings.css +30 -0
- package/build/esm/form-builder-v2/components/FieldSettings/FieldSettings.d.ts +7 -0
- package/build/esm/form-builder-v2/components/FieldSettings/FieldSettings.js +45 -0
- package/build/esm/form-builder-v2/components/FieldSettings/FieldSettings.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/ColorInputSettings.d.ts +12 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/ColorInputSettings.js +6 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/ColorInputSettings.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/OptionsSettings.d.ts +12 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/OptionsSettings.js +33 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/OptionsSettings.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/Row.d.ts +7 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/Row.js +5 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/Row.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/SectionSettings.d.ts +9 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/SectionSettings.js +35 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/SectionSettings.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/SwitchSettings.d.ts +12 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/SwitchSettings.js +6 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/SwitchSettings.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/TextFieldSettings.d.ts +12 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/TextFieldSettings.js +6 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/TextFieldSettings.js.map +1 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/TextSettings.d.ts +11 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/TextSettings.js +48 -0
- package/build/esm/form-builder-v2/components/FieldSettings/fields/TextSettings.js.map +1 -0
- package/build/esm/form-builder-v2/components/Inspector/Inspector.css +20 -0
- package/build/esm/form-builder-v2/components/Inspector/Inspector.d.ts +2 -0
- package/build/esm/form-builder-v2/components/Inspector/Inspector.js +16 -0
- package/build/esm/form-builder-v2/components/Inspector/Inspector.js.map +1 -0
- package/build/esm/form-builder-v2/components/Palette/Palette.css +56 -0
- package/build/esm/form-builder-v2/components/Palette/Palette.d.ts +5 -0
- package/build/esm/form-builder-v2/components/Palette/Palette.js +58 -0
- package/build/esm/form-builder-v2/components/Palette/Palette.js.map +1 -0
- package/build/esm/form-builder-v2/components/ResizeHandle/ResizeHandle.css +29 -0
- package/build/esm/form-builder-v2/components/ResizeHandle/ResizeHandle.d.ts +10 -0
- package/build/esm/form-builder-v2/components/ResizeHandle/ResizeHandle.js +94 -0
- package/build/esm/form-builder-v2/components/ResizeHandle/ResizeHandle.js.map +1 -0
- package/build/esm/form-builder-v2/components/SchemaPopup/SchemaPopup.css +49 -0
- package/build/esm/form-builder-v2/components/SchemaPopup/SchemaPopup.d.ts +12 -0
- package/build/esm/form-builder-v2/components/SchemaPopup/SchemaPopup.js +78 -0
- package/build/esm/form-builder-v2/components/SchemaPopup/SchemaPopup.js.map +1 -0
- package/build/esm/form-builder-v2/components/WhenEditor/WhenEditor.css +24 -0
- package/build/esm/form-builder-v2/components/WhenEditor/WhenEditor.d.ts +9 -0
- package/build/esm/form-builder-v2/components/WhenEditor/WhenEditor.js +46 -0
- package/build/esm/form-builder-v2/components/WhenEditor/WhenEditor.js.map +1 -0
- package/build/esm/form-builder-v2/components/WhenEditor/utils.d.ts +24 -0
- package/build/esm/form-builder-v2/components/WhenEditor/utils.js +76 -0
- package/build/esm/form-builder-v2/components/WhenEditor/utils.js.map +1 -0
- package/build/esm/form-builder-v2/hooks/FormContext.d.ts +11 -0
- package/build/esm/form-builder-v2/hooks/FormContext.js +16 -0
- package/build/esm/form-builder-v2/hooks/FormContext.js.map +1 -0
- package/build/esm/form-builder-v2/hooks/useFormFields.d.ts +23 -0
- package/build/esm/form-builder-v2/hooks/useFormFields.js +139 -0
- package/build/esm/form-builder-v2/hooks/useFormFields.js.map +1 -0
- package/build/esm/form-builder-v2/index.d.ts +2 -0
- package/build/esm/form-builder-v2/index.js +3 -0
- package/build/esm/form-builder-v2/index.js.map +1 -0
- package/build/esm/form-builder-v2/styles/variables.css +0 -0
- package/build/esm/form-builder-v2/types.d.ts +32 -0
- package/build/esm/form-builder-v2/types.js +2 -0
- package/build/esm/form-builder-v2/types.js.map +1 -0
- package/build/esm/form-builder-v2/utils/cn.d.ts +2 -0
- package/build/esm/form-builder-v2/utils/cn.js +4 -0
- package/build/esm/form-builder-v2/utils/cn.js.map +1 -0
- package/build/esm/form-builder-v2/utils/dragData.d.ts +18 -0
- package/build/esm/form-builder-v2/utils/dragData.js +11 -0
- package/build/esm/form-builder-v2/utils/dragData.js.map +1 -0
- package/build/esm/form-builder-v2/utils/fieldDefaults.d.ts +2 -0
- package/build/esm/form-builder-v2/utils/fieldDefaults.js +47 -0
- package/build/esm/form-builder-v2/utils/fieldDefaults.js.map +1 -0
- package/build/esm/form-builder-v2/utils/fieldGroups.d.ts +3 -0
- package/build/esm/form-builder-v2/utils/fieldGroups.js +21 -0
- package/build/esm/form-builder-v2/utils/fieldGroups.js.map +1 -0
- package/build/esm/form-builder-v2/utils/fieldMeta.d.ts +6 -0
- package/build/esm/form-builder-v2/utils/fieldMeta.js +42 -0
- package/build/esm/form-builder-v2/utils/fieldMeta.js.map +1 -0
- package/build/esm/form-builder-v2/utils/fieldNames.d.ts +8 -0
- package/build/esm/form-builder-v2/utils/fieldNames.js +66 -0
- package/build/esm/form-builder-v2/utils/fieldNames.js.map +1 -0
- package/build/esm/form-builder-v2/utils/fieldTree.d.ts +2 -0
- package/build/esm/form-builder-v2/utils/fieldTree.js +3 -0
- package/build/esm/form-builder-v2/utils/fieldTree.js.map +1 -0
- package/build/esm/form-builder-v2/utils/fieldTreeOps.d.ts +13 -0
- package/build/esm/form-builder-v2/utils/fieldTreeOps.js +69 -0
- package/build/esm/form-builder-v2/utils/fieldTreeOps.js.map +1 -0
- package/build/esm/form-builder-v2/utils/parseSchema.d.ts +9 -0
- package/build/esm/form-builder-v2/utils/parseSchema.js +98 -0
- package/build/esm/form-builder-v2/utils/parseSchema.js.map +1 -0
- package/build/esm/form-builder-v2/utils/stripIds.d.ts +3 -0
- package/build/esm/form-builder-v2/utils/stripIds.js +8 -0
- package/build/esm/form-builder-v2/utils/stripIds.js.map +1 -0
- package/build/esm/form-builder-v2/utils/treeWalk.d.ts +4 -0
- package/build/esm/form-builder-v2/utils/treeWalk.js +39 -0
- package/build/esm/form-builder-v2/utils/treeWalk.js.map +1 -0
- package/build/esm/hooks/usePCEditorInitializeEvents.js +2 -2
- package/build/esm/hooks/usePCEditorInitializeEvents.js.map +1 -1
- package/package.json +13 -1
- package/widget/index.js +1 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ResizeHandle.js","sourceRoot":"../../../../../src","sources":["form-builder-v2/components/ResizeHandle/ResizeHandle.tsx"],"names":[],"mappings":"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","sourcesContent":["import * as React from 'react';\n\nimport {formBuilderV2Cn} from '../../utils/cn';\n\nimport './ResizeHandle.scss';\n\nconst b = formBuilderV2Cn('resize-handle');\n\ninterface ResizeHandleProps {\n value: number;\n min: number;\n max: number;\n direction: 'left' | 'right';\n onChange: (next: number) => void;\n}\n\nexport const ResizeHandle = ({value, min, max, direction, onChange}: ResizeHandleProps) => {\n const [dragging, setDragging] = React.useState(false);\n const onChangeRef = React.useRef(onChange);\n onChangeRef.current = onChange;\n\n const cleanupRef = React.useRef<(() => void) | null>(null);\n const mountedRef = React.useRef(true);\n\n React.useEffect(() => {\n mountedRef.current = true;\n return () => {\n mountedRef.current = false;\n cleanupRef.current?.();\n };\n }, []);\n\n const onMouseDown = React.useCallback(\n (event: React.MouseEvent) => {\n event.preventDefault();\n const startX = event.clientX;\n const startValue = value;\n setDragging(true);\n\n let rafId: number | null = null;\n let pendingValue = startValue;\n const controller = new AbortController();\n const {signal} = controller;\n\n const flush = () => {\n rafId = null;\n onChangeRef.current(pendingValue);\n };\n\n const cleanup = () => {\n if (signal.aborted) return;\n controller.abort();\n if (rafId !== null) {\n cancelAnimationFrame(rafId);\n rafId = null;\n }\n document.body.style.cursor = '';\n document.body.style.userSelect = '';\n cleanupRef.current = null;\n };\n\n const handleMove = (moveEvent: MouseEvent) => {\n const deltaRaw = moveEvent.clientX - startX;\n const delta = direction === 'left' ? deltaRaw : -deltaRaw;\n pendingValue = Math.max(min, Math.min(max, startValue + delta));\n if (rafId === null) {\n rafId = requestAnimationFrame(flush);\n }\n };\n\n const handleUp = () => {\n if (rafId !== null) {\n cancelAnimationFrame(rafId);\n rafId = null;\n onChangeRef.current(pendingValue);\n }\n if (mountedRef.current) {\n setDragging(false);\n }\n cleanup();\n };\n\n document.body.style.cursor = 'col-resize';\n document.body.style.userSelect = 'none';\n\n document.addEventListener('mousemove', handleMove, {signal});\n document.addEventListener('mouseup', handleUp, {signal});\n\n cleanupRef.current = cleanup;\n },\n [direction, max, min, value],\n );\n\n const onKeyDown = React.useCallback(\n (event: React.KeyboardEvent<HTMLDivElement>) => {\n const STEP = event.shiftKey ? 32 : 8;\n const sign = direction === 'left' ? 1 : -1;\n let next: number | null = null;\n switch (event.key) {\n case 'ArrowLeft':\n next = value + sign * -STEP;\n break;\n case 'ArrowRight':\n next = value + sign * STEP;\n break;\n case 'Home':\n next = direction === 'left' ? min : max;\n break;\n case 'End':\n next = direction === 'left' ? max : min;\n break;\n default:\n return;\n }\n event.preventDefault();\n onChangeRef.current(Math.max(min, Math.min(max, next)));\n },\n [direction, max, min, value],\n );\n\n return (\n <div\n className={b({dragging})}\n onMouseDown={onMouseDown}\n onKeyDown={onKeyDown}\n role=\"slider\"\n aria-orientation=\"vertical\"\n aria-valuemin={min}\n aria-valuemax={max}\n aria-valuenow={value}\n aria-label={direction === 'left' ? 'Resize palette' : 'Resize inspector'}\n tabIndex={0}\n />\n );\n};\n"]}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
.pcformbuilderv2-schema-popup {
|
|
2
|
+
width: min(560px, 90vw);
|
|
3
|
+
padding: var(--g-spacing-4);
|
|
4
|
+
display: flex;
|
|
5
|
+
flex-direction: column;
|
|
6
|
+
gap: 10px;
|
|
7
|
+
}
|
|
8
|
+
.pcformbuilderv2-schema-popup__header {
|
|
9
|
+
display: flex;
|
|
10
|
+
flex-direction: column;
|
|
11
|
+
gap: var(--g-spacing-half);
|
|
12
|
+
}
|
|
13
|
+
.pcformbuilderv2-schema-popup__textarea {
|
|
14
|
+
box-sizing: border-box;
|
|
15
|
+
width: 100%;
|
|
16
|
+
max-width: 100%;
|
|
17
|
+
min-height: 400px;
|
|
18
|
+
max-height: min(70vh, 700px);
|
|
19
|
+
padding: var(--g-spacing-3);
|
|
20
|
+
background: var(--g-color-base-generic);
|
|
21
|
+
border: 1px solid var(--g-color-line-generic);
|
|
22
|
+
border-radius: var(--g-border-radius-m);
|
|
23
|
+
font-family: monospace;
|
|
24
|
+
font-size: 12px;
|
|
25
|
+
line-height: 1.4;
|
|
26
|
+
color: var(--g-color-text-primary);
|
|
27
|
+
white-space: pre;
|
|
28
|
+
overflow: auto;
|
|
29
|
+
resize: vertical;
|
|
30
|
+
outline: none;
|
|
31
|
+
transition: border-color 0.15s ease;
|
|
32
|
+
tab-size: 2;
|
|
33
|
+
}
|
|
34
|
+
.pcformbuilderv2-schema-popup__textarea:focus {
|
|
35
|
+
border-color: var(--g-color-line-brand);
|
|
36
|
+
}
|
|
37
|
+
.pcformbuilderv2-schema-popup__textarea_error {
|
|
38
|
+
border-color: var(--g-color-line-danger);
|
|
39
|
+
}
|
|
40
|
+
.pcformbuilderv2-schema-popup__error-message {
|
|
41
|
+
padding: var(--g-spacing-2) var(--g-spacing-3);
|
|
42
|
+
background: var(--g-color-base-danger-light);
|
|
43
|
+
border-radius: var(--g-border-radius-m);
|
|
44
|
+
}
|
|
45
|
+
.pcformbuilderv2-schema-popup__actions {
|
|
46
|
+
display: flex;
|
|
47
|
+
gap: var(--g-spacing-2);
|
|
48
|
+
justify-content: flex-end;
|
|
49
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { Fields } from "../../../form-generator-v2/types.js";
|
|
2
|
+
import type { FormField } from "../../types.js";
|
|
3
|
+
import './SchemaPopup.css';
|
|
4
|
+
interface SchemaPopupProps {
|
|
5
|
+
schema: Fields;
|
|
6
|
+
onApply: (fields: FormField[]) => void;
|
|
7
|
+
open: boolean;
|
|
8
|
+
onOpenChange: (open: boolean) => void;
|
|
9
|
+
anchorElement: HTMLElement | null;
|
|
10
|
+
}
|
|
11
|
+
export declare const SchemaPopup: ({ schema, onApply, open, onOpenChange, anchorElement, }: SchemaPopupProps) => import("react/jsx-runtime").JSX.Element;
|
|
12
|
+
export {};
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import * as React from 'react';
|
|
3
|
+
import { Button, Card, Popup, Text } from '@gravity-ui/uikit';
|
|
4
|
+
import { formBuilderV2Cn } from "../../utils/cn.js";
|
|
5
|
+
import { parseSchema } from "../../utils/parseSchema.js";
|
|
6
|
+
import './SchemaPopup.css';
|
|
7
|
+
const b = formBuilderV2Cn('schema-popup');
|
|
8
|
+
const stringify = (schema) => JSON.stringify(schema, null, 2);
|
|
9
|
+
export const SchemaPopup = ({ schema, onApply, open, onOpenChange, anchorElement, }) => {
|
|
10
|
+
const textareaRef = React.useRef(null);
|
|
11
|
+
const copyTimeoutRef = React.useRef(null);
|
|
12
|
+
const [draft, setDraft] = React.useState(() => stringify(schema));
|
|
13
|
+
const [error, setError] = React.useState(null);
|
|
14
|
+
const [copied, setCopied] = React.useState(false);
|
|
15
|
+
const [isFocused, setIsFocused] = React.useState(false);
|
|
16
|
+
React.useEffect(() => {
|
|
17
|
+
if (!isFocused) {
|
|
18
|
+
setDraft(stringify(schema));
|
|
19
|
+
setError(null);
|
|
20
|
+
}
|
|
21
|
+
}, [schema, isFocused]);
|
|
22
|
+
React.useEffect(() => () => {
|
|
23
|
+
if (copyTimeoutRef.current)
|
|
24
|
+
clearTimeout(copyTimeoutRef.current);
|
|
25
|
+
}, []);
|
|
26
|
+
const applyDraft = (value) => {
|
|
27
|
+
setDraft(value);
|
|
28
|
+
const result = parseSchema(value);
|
|
29
|
+
if (result.ok) {
|
|
30
|
+
setError(null);
|
|
31
|
+
onApply(result.fields);
|
|
32
|
+
}
|
|
33
|
+
else {
|
|
34
|
+
setError(result.error);
|
|
35
|
+
}
|
|
36
|
+
};
|
|
37
|
+
const replaceSelection = (insert, caretOffset) => {
|
|
38
|
+
const t = textareaRef.current;
|
|
39
|
+
if (!t)
|
|
40
|
+
return;
|
|
41
|
+
const { selectionStart, selectionEnd } = t;
|
|
42
|
+
const next = draft.slice(0, selectionStart) + insert + draft.slice(selectionEnd);
|
|
43
|
+
const nextCaret = selectionStart + caretOffset;
|
|
44
|
+
applyDraft(next);
|
|
45
|
+
requestAnimationFrame(() => {
|
|
46
|
+
t.selectionStart = nextCaret;
|
|
47
|
+
t.selectionEnd = nextCaret;
|
|
48
|
+
});
|
|
49
|
+
};
|
|
50
|
+
const handleKeyDown = (event) => {
|
|
51
|
+
if (event.key === 'Tab') {
|
|
52
|
+
event.preventDefault();
|
|
53
|
+
replaceSelection(' ', 2);
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
if (event.key === 'Enter') {
|
|
57
|
+
event.preventDefault();
|
|
58
|
+
const t = event.currentTarget;
|
|
59
|
+
const before = draft.slice(0, t.selectionStart);
|
|
60
|
+
const lineStart = before.lastIndexOf('\n') + 1;
|
|
61
|
+
const indent = before.slice(lineStart).match(/^[ \t]*/)?.[0] ?? '';
|
|
62
|
+
const insert = `\n${indent}`;
|
|
63
|
+
replaceSelection(insert, insert.length);
|
|
64
|
+
}
|
|
65
|
+
};
|
|
66
|
+
const handleCopy = React.useCallback(async () => {
|
|
67
|
+
try {
|
|
68
|
+
await navigator.clipboard.writeText(draft);
|
|
69
|
+
setCopied(true);
|
|
70
|
+
if (copyTimeoutRef.current)
|
|
71
|
+
clearTimeout(copyTimeoutRef.current);
|
|
72
|
+
copyTimeoutRef.current = setTimeout(() => setCopied(false), 1500);
|
|
73
|
+
}
|
|
74
|
+
catch { }
|
|
75
|
+
}, [draft]);
|
|
76
|
+
return (_jsx(Popup, { anchorElement: anchorElement, open: open, onOpenChange: onOpenChange, placement: "bottom-end", children: _jsxs(Card, { className: b(), view: "outlined", children: [_jsxs("div", { className: b('header'), children: [_jsx(Text, { variant: "subheader-2", children: "Form schema" }), _jsx(Text, { variant: "caption-1", color: "hint", children: "Edit JSON directly \u2014 the canvas updates live. Paste your own schema to load it. Selection on canvas resets after each edit." })] }), _jsx("textarea", { ref: textareaRef, className: b('textarea', { error: Boolean(error) }), value: draft, onChange: (event) => applyDraft(event.target.value), onKeyDown: handleKeyDown, onFocus: () => setIsFocused(true), onBlur: () => setIsFocused(false), spellCheck: false, wrap: "off" }), error && (_jsx("div", { className: b('error-message'), children: _jsx(Text, { variant: "caption-1", color: "danger", children: error }) })), _jsxs("div", { className: b('actions'), children: [_jsx(Button, { view: "action", size: "m", onClick: handleCopy, children: copied ? '✓ Copied' : 'Copy' }), _jsx(Button, { view: "flat", size: "m", onClick: () => onOpenChange(false), children: "Close" })] })] }) }));
|
|
77
|
+
};
|
|
78
|
+
//# sourceMappingURL=SchemaPopup.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SchemaPopup.js","sourceRoot":"../../../../../src","sources":["form-builder-v2/components/SchemaPopup/SchemaPopup.tsx"],"names":[],"mappings":"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","sourcesContent":["import * as React from 'react';\n\nimport {Button, Card, Popup, Text} from '@gravity-ui/uikit';\n\nimport type {Fields} from '../../../form-generator-v2/types';\nimport type {FormField} from '../../types';\nimport {formBuilderV2Cn} from '../../utils/cn';\nimport {parseSchema} from '../../utils/parseSchema';\n\nimport './SchemaPopup.scss';\n\nconst b = formBuilderV2Cn('schema-popup');\n\nconst stringify = (schema: Fields): string => JSON.stringify(schema, null, 2);\n\ninterface SchemaPopupProps {\n schema: Fields;\n onApply: (fields: FormField[]) => void;\n open: boolean;\n onOpenChange: (open: boolean) => void;\n anchorElement: HTMLElement | null;\n}\n\nexport const SchemaPopup = ({\n schema,\n onApply,\n open,\n onOpenChange,\n anchorElement,\n}: SchemaPopupProps) => {\n const textareaRef = React.useRef<HTMLTextAreaElement>(null);\n const copyTimeoutRef = React.useRef<ReturnType<typeof setTimeout> | null>(null);\n const [draft, setDraft] = React.useState<string>(() => stringify(schema));\n const [error, setError] = React.useState<string | null>(null);\n const [copied, setCopied] = React.useState(false);\n const [isFocused, setIsFocused] = React.useState(false);\n\n React.useEffect(() => {\n if (!isFocused) {\n setDraft(stringify(schema));\n setError(null);\n }\n }, [schema, isFocused]);\n\n React.useEffect(\n () => () => {\n if (copyTimeoutRef.current) clearTimeout(copyTimeoutRef.current);\n },\n [],\n );\n\n const applyDraft = (value: string) => {\n setDraft(value);\n const result = parseSchema(value);\n if (result.ok) {\n setError(null);\n onApply(result.fields);\n } else {\n setError(result.error);\n }\n };\n\n const replaceSelection = (insert: string, caretOffset: number) => {\n const t = textareaRef.current;\n if (!t) return;\n const {selectionStart, selectionEnd} = t;\n const next = draft.slice(0, selectionStart) + insert + draft.slice(selectionEnd);\n const nextCaret = selectionStart + caretOffset;\n applyDraft(next);\n requestAnimationFrame(() => {\n t.selectionStart = nextCaret;\n t.selectionEnd = nextCaret;\n });\n };\n\n const handleKeyDown = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {\n if (event.key === 'Tab') {\n event.preventDefault();\n replaceSelection(' ', 2);\n return;\n }\n if (event.key === 'Enter') {\n event.preventDefault();\n const t = event.currentTarget;\n const before = draft.slice(0, t.selectionStart);\n const lineStart = before.lastIndexOf('\\n') + 1;\n const indent = before.slice(lineStart).match(/^[ \\t]*/)?.[0] ?? '';\n const insert = `\\n${indent}`;\n replaceSelection(insert, insert.length);\n }\n };\n\n const handleCopy = React.useCallback(async () => {\n try {\n await navigator.clipboard.writeText(draft);\n setCopied(true);\n if (copyTimeoutRef.current) clearTimeout(copyTimeoutRef.current);\n copyTimeoutRef.current = setTimeout(() => setCopied(false), 1500);\n } catch {}\n }, [draft]);\n\n return (\n <Popup\n anchorElement={anchorElement}\n open={open}\n onOpenChange={onOpenChange}\n placement=\"bottom-end\"\n >\n <Card className={b()} view=\"outlined\">\n <div className={b('header')}>\n <Text variant=\"subheader-2\">Form schema</Text>\n <Text variant=\"caption-1\" color=\"hint\">\n Edit JSON directly — the canvas updates live. Paste your own schema to load\n it. Selection on canvas resets after each edit.\n </Text>\n </div>\n <textarea\n ref={textareaRef}\n className={b('textarea', {error: Boolean(error)})}\n value={draft}\n onChange={(event) => applyDraft(event.target.value)}\n onKeyDown={handleKeyDown}\n onFocus={() => setIsFocused(true)}\n onBlur={() => setIsFocused(false)}\n spellCheck={false}\n wrap=\"off\"\n />\n {error && (\n <div className={b('error-message')}>\n <Text variant=\"caption-1\" color=\"danger\">\n {error}\n </Text>\n </div>\n )}\n <div className={b('actions')}>\n <Button view=\"action\" size=\"m\" onClick={handleCopy}>\n {copied ? '✓ Copied' : 'Copy'}\n </Button>\n <Button view=\"flat\" size=\"m\" onClick={() => onOpenChange(false)}>\n Close\n </Button>\n </div>\n </Card>\n </Popup>\n );\n};\n"]}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
.pcformbuilderv2-when-editor {
|
|
2
|
+
display: flex;
|
|
3
|
+
flex-direction: column;
|
|
4
|
+
gap: 6px;
|
|
5
|
+
padding: var(--g-spacing-2);
|
|
6
|
+
border: 1px solid var(--g-color-line-generic);
|
|
7
|
+
border-radius: var(--g-border-radius-m);
|
|
8
|
+
}
|
|
9
|
+
.pcformbuilderv2-when-editor__condition {
|
|
10
|
+
display: grid;
|
|
11
|
+
grid-template-columns: minmax(0, 1fr) 80px minmax(0, 1fr) auto;
|
|
12
|
+
gap: 6px;
|
|
13
|
+
align-items: center;
|
|
14
|
+
}
|
|
15
|
+
.pcformbuilderv2-when-editor__combinator {
|
|
16
|
+
display: flex;
|
|
17
|
+
align-items: center;
|
|
18
|
+
}
|
|
19
|
+
.pcformbuilderv2-when-editor__actions {
|
|
20
|
+
display: flex;
|
|
21
|
+
gap: var(--g-spacing-2);
|
|
22
|
+
align-items: center;
|
|
23
|
+
margin-top: var(--g-spacing-1);
|
|
24
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import type { When } from "../../../form-generator-v2/types.js";
|
|
2
|
+
import './WhenEditor.css';
|
|
3
|
+
interface WhenEditorProps {
|
|
4
|
+
when: When | undefined;
|
|
5
|
+
availableFields: string[];
|
|
6
|
+
onChange: (next: When | undefined) => void;
|
|
7
|
+
}
|
|
8
|
+
export declare const WhenEditor: ({ when, availableFields, onChange }: WhenEditorProps) => import("react/jsx-runtime").JSX.Element;
|
|
9
|
+
export {};
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import * as React from 'react';
|
|
3
|
+
import { TrashBin } from '@gravity-ui/icons';
|
|
4
|
+
import { Button, Icon, Select, TextInput } from '@gravity-ui/uikit';
|
|
5
|
+
import { formBuilderV2Cn } from "../../utils/cn.js";
|
|
6
|
+
import { COMBINATOR_OPTIONS, OPERATOR_OPTIONS, coerceValue, displayValue, parse, serialize, } from "./utils.js";
|
|
7
|
+
import './WhenEditor.css';
|
|
8
|
+
const b = formBuilderV2Cn('when-editor');
|
|
9
|
+
export const WhenEditor = ({ when, availableFields, onChange }) => {
|
|
10
|
+
const { conditions, combinators } = parse(when);
|
|
11
|
+
const fieldOptions = React.useMemo(() => availableFields.map((name) => ({ value: name, content: name })), [availableFields]);
|
|
12
|
+
const commit = (nextConditions, nextCombinators) => {
|
|
13
|
+
onChange(serialize(nextConditions, nextCombinators));
|
|
14
|
+
};
|
|
15
|
+
const addCondition = () => {
|
|
16
|
+
const nextConditions = [
|
|
17
|
+
...conditions,
|
|
18
|
+
{ field: availableFields[0] ?? '', operator: '===', value: '' },
|
|
19
|
+
];
|
|
20
|
+
const nextCombinators = conditions.length === 0 ? combinators : [...combinators, { operator: '&&' }];
|
|
21
|
+
commit(nextConditions, nextCombinators);
|
|
22
|
+
};
|
|
23
|
+
const removeCondition = (index) => {
|
|
24
|
+
const nextConditions = conditions.filter((_, i) => i !== index);
|
|
25
|
+
const nextCombinators = index === 0
|
|
26
|
+
? combinators.slice(1)
|
|
27
|
+
: [...combinators.slice(0, index - 1), ...combinators.slice(index)];
|
|
28
|
+
commit(nextConditions, nextCombinators);
|
|
29
|
+
};
|
|
30
|
+
const updateCondition = (index, patch) => {
|
|
31
|
+
const nextConditions = conditions.map((c, i) => (i === index ? { ...c, ...patch } : c));
|
|
32
|
+
commit(nextConditions, combinators);
|
|
33
|
+
};
|
|
34
|
+
const updateCombinator = (index, op) => {
|
|
35
|
+
const nextCombinators = combinators.map((c, i) => (i === index ? { operator: op } : c));
|
|
36
|
+
commit(conditions, nextCombinators);
|
|
37
|
+
};
|
|
38
|
+
const clearAll = () => {
|
|
39
|
+
onChange(undefined);
|
|
40
|
+
};
|
|
41
|
+
if (conditions.length === 0) {
|
|
42
|
+
return (_jsx("div", { className: b(), children: _jsx(Button, { view: "normal", size: "s", onClick: addCondition, children: "+ Add condition" }) }));
|
|
43
|
+
}
|
|
44
|
+
return (_jsxs("div", { className: b(), children: [conditions.map((cond, i) => (_jsxs(React.Fragment, { children: [i > 0 && (_jsx("div", { className: b('combinator'), children: _jsx(Select, { size: "s", value: [combinators[i - 1]?.operator ?? '&&'], options: COMBINATOR_OPTIONS, onUpdate: (next) => updateCombinator(i - 1, next[0]), width: 80 }) })), _jsxs("div", { className: b('condition'), children: [_jsx(Select, { size: "s", value: [cond.field], options: fieldOptions, onUpdate: (next) => updateCondition(i, { field: next[0] ?? '' }), placeholder: "field", filterable: true, width: "max" }), _jsx(Select, { size: "s", value: [cond.operator], options: OPERATOR_OPTIONS, onUpdate: (next) => updateCondition(i, { operator: next[0] }), width: 80 }), _jsx(TextInput, { size: "s", value: displayValue(cond.value), onUpdate: (value) => updateCondition(i, { value: coerceValue(value) }), placeholder: 'value ("true"/"false" \u2192 bool)' }), _jsx(Button, { view: "flat-danger", size: "s", onClick: () => removeCondition(i), title: "Remove condition", children: _jsx(Icon, { data: TrashBin, size: 12 }) })] })] }, i))), _jsxs("div", { className: b('actions'), children: [_jsx(Button, { view: "normal", size: "s", onClick: addCondition, children: "+ Add condition" }), _jsx(Button, { view: "flat", size: "s", onClick: clearAll, children: "Clear" })] })] }));
|
|
45
|
+
};
|
|
46
|
+
//# sourceMappingURL=WhenEditor.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"WhenEditor.js","sourceRoot":"../../../../../src","sources":["form-builder-v2/components/WhenEditor/WhenEditor.tsx"],"names":[],"mappings":"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","sourcesContent":["import * as React from 'react';\n\nimport {TrashBin} from '@gravity-ui/icons';\nimport {Button, Icon, Select, TextInput} from '@gravity-ui/uikit';\n\nimport type {When} from '../../../form-generator-v2/types';\nimport {formBuilderV2Cn} from '../../utils/cn';\n\nimport {\n COMBINATOR_OPTIONS,\n Combinator,\n Condition,\n OPERATOR_OPTIONS,\n coerceValue,\n displayValue,\n parse,\n serialize,\n} from './utils';\n\nimport './WhenEditor.scss';\n\nconst b = formBuilderV2Cn('when-editor');\n\ninterface WhenEditorProps {\n when: When | undefined;\n availableFields: string[];\n onChange: (next: When | undefined) => void;\n}\n\nexport const WhenEditor = ({when, availableFields, onChange}: WhenEditorProps) => {\n const {conditions, combinators} = parse(when);\n\n const fieldOptions = React.useMemo(\n () => availableFields.map((name) => ({value: name, content: name})),\n [availableFields],\n );\n\n const commit = (nextConditions: Condition[], nextCombinators: Combinator[]) => {\n onChange(serialize(nextConditions, nextCombinators));\n };\n\n const addCondition = () => {\n const nextConditions: Condition[] = [\n ...conditions,\n {field: availableFields[0] ?? '', operator: '===', value: ''},\n ];\n const nextCombinators: Combinator[] =\n conditions.length === 0 ? combinators : [...combinators, {operator: '&&'}];\n commit(nextConditions, nextCombinators);\n };\n\n const removeCondition = (index: number) => {\n const nextConditions = conditions.filter((_, i) => i !== index);\n const nextCombinators =\n index === 0\n ? combinators.slice(1)\n : [...combinators.slice(0, index - 1), ...combinators.slice(index)];\n commit(nextConditions, nextCombinators);\n };\n\n const updateCondition = (index: number, patch: Partial<Condition>) => {\n const nextConditions = conditions.map((c, i) => (i === index ? {...c, ...patch} : c));\n commit(nextConditions, combinators);\n };\n\n const updateCombinator = (index: number, op: '&&' | '||') => {\n const nextCombinators = combinators.map((c, i) => (i === index ? {operator: op} : c));\n commit(conditions, nextCombinators);\n };\n\n const clearAll = () => {\n onChange(undefined);\n };\n\n if (conditions.length === 0) {\n return (\n <div className={b()}>\n <Button view=\"normal\" size=\"s\" onClick={addCondition}>\n + Add condition\n </Button>\n </div>\n );\n }\n\n return (\n <div className={b()}>\n {conditions.map((cond, i) => (\n <React.Fragment key={i}>\n {i > 0 && (\n <div className={b('combinator')}>\n <Select\n size=\"s\"\n value={[combinators[i - 1]?.operator ?? '&&']}\n options={COMBINATOR_OPTIONS}\n onUpdate={(next) => updateCombinator(i - 1, next[0] as '&&' | '||')}\n width={80}\n />\n </div>\n )}\n <div className={b('condition')}>\n <Select\n size=\"s\"\n value={[cond.field]}\n options={fieldOptions}\n onUpdate={(next) => updateCondition(i, {field: next[0] ?? ''})}\n placeholder=\"field\"\n filterable\n width=\"max\"\n />\n <Select\n size=\"s\"\n value={[cond.operator]}\n options={OPERATOR_OPTIONS}\n onUpdate={(next) =>\n updateCondition(i, {operator: next[0] as '===' | '!=='})\n }\n width={80}\n />\n <TextInput\n size=\"s\"\n value={displayValue(cond.value)}\n onUpdate={(value) => updateCondition(i, {value: coerceValue(value)})}\n placeholder='value (\"true\"/\"false\" → bool)'\n />\n <Button\n view=\"flat-danger\"\n size=\"s\"\n onClick={() => removeCondition(i)}\n title=\"Remove condition\"\n >\n <Icon data={TrashBin} size={12} />\n </Button>\n </div>\n </React.Fragment>\n ))}\n <div className={b('actions')}>\n <Button view=\"normal\" size=\"s\" onClick={addCondition}>\n + Add condition\n </Button>\n <Button view=\"flat\" size=\"s\" onClick={clearAll}>\n Clear\n </Button>\n </div>\n </div>\n );\n};\n"]}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import type { When } from "../../../form-generator-v2/types.js";
|
|
2
|
+
export type Condition = {
|
|
3
|
+
field: string;
|
|
4
|
+
operator: '===' | '!==';
|
|
5
|
+
value: string | boolean;
|
|
6
|
+
};
|
|
7
|
+
export type Combinator = {
|
|
8
|
+
operator: '&&' | '||';
|
|
9
|
+
};
|
|
10
|
+
export declare const OPERATOR_OPTIONS: {
|
|
11
|
+
value: string;
|
|
12
|
+
content: string;
|
|
13
|
+
}[];
|
|
14
|
+
export declare const COMBINATOR_OPTIONS: {
|
|
15
|
+
value: string;
|
|
16
|
+
content: string;
|
|
17
|
+
}[];
|
|
18
|
+
export declare const parse: (when: When | undefined) => {
|
|
19
|
+
conditions: Condition[];
|
|
20
|
+
combinators: Combinator[];
|
|
21
|
+
};
|
|
22
|
+
export declare const serialize: (conditions: Condition[], combinators: Combinator[]) => When | undefined;
|
|
23
|
+
export declare const coerceValue: (raw: string) => string | boolean;
|
|
24
|
+
export declare const displayValue: (value: string | boolean | undefined) => string;
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
export const OPERATOR_OPTIONS = [
|
|
2
|
+
{ value: '===', content: 'is' },
|
|
3
|
+
{ value: '!==', content: 'is not' },
|
|
4
|
+
];
|
|
5
|
+
export const COMBINATOR_OPTIONS = [
|
|
6
|
+
{ value: '&&', content: 'AND' },
|
|
7
|
+
{ value: '||', content: 'OR' },
|
|
8
|
+
];
|
|
9
|
+
export const parse = (when) => {
|
|
10
|
+
const conditions = [];
|
|
11
|
+
const combinators = [];
|
|
12
|
+
if (!when)
|
|
13
|
+
return { conditions, combinators };
|
|
14
|
+
let lastWasCombinator = false;
|
|
15
|
+
for (const entry of when) {
|
|
16
|
+
const isCondition = Boolean(entry.field) && (entry.operator === '===' || entry.operator === '!==');
|
|
17
|
+
const isCombinator = !entry.field && (entry.operator === '&&' || entry.operator === '||');
|
|
18
|
+
if (isCondition) {
|
|
19
|
+
conditions.push({
|
|
20
|
+
field: entry.field,
|
|
21
|
+
operator: entry.operator,
|
|
22
|
+
value: entry.value ?? '',
|
|
23
|
+
});
|
|
24
|
+
lastWasCombinator = false;
|
|
25
|
+
continue;
|
|
26
|
+
}
|
|
27
|
+
if (isCombinator) {
|
|
28
|
+
if (conditions.length === 0 || lastWasCombinator)
|
|
29
|
+
continue;
|
|
30
|
+
combinators.push({ operator: entry.operator });
|
|
31
|
+
lastWasCombinator = true;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
if (lastWasCombinator) {
|
|
35
|
+
combinators.pop();
|
|
36
|
+
}
|
|
37
|
+
while (combinators.length < conditions.length - 1) {
|
|
38
|
+
combinators.push({ operator: '&&' });
|
|
39
|
+
}
|
|
40
|
+
if (combinators.length > Math.max(0, conditions.length - 1)) {
|
|
41
|
+
combinators.length = Math.max(0, conditions.length - 1);
|
|
42
|
+
}
|
|
43
|
+
return { conditions, combinators };
|
|
44
|
+
};
|
|
45
|
+
export const serialize = (conditions, combinators) => {
|
|
46
|
+
if (conditions.length === 0)
|
|
47
|
+
return undefined;
|
|
48
|
+
const out = [];
|
|
49
|
+
conditions.forEach((cond, i) => {
|
|
50
|
+
if (i > 0) {
|
|
51
|
+
const combinator = combinators[i - 1] ?? { operator: '&&' };
|
|
52
|
+
out.push({ operator: combinator.operator });
|
|
53
|
+
}
|
|
54
|
+
out.push({
|
|
55
|
+
field: cond.field,
|
|
56
|
+
operator: cond.operator,
|
|
57
|
+
value: cond.value,
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
return out;
|
|
61
|
+
};
|
|
62
|
+
export const coerceValue = (raw) => {
|
|
63
|
+
if (raw === 'true')
|
|
64
|
+
return true;
|
|
65
|
+
if (raw === 'false')
|
|
66
|
+
return false;
|
|
67
|
+
return raw;
|
|
68
|
+
};
|
|
69
|
+
export const displayValue = (value) => {
|
|
70
|
+
if (value === true)
|
|
71
|
+
return 'true';
|
|
72
|
+
if (value === false)
|
|
73
|
+
return 'false';
|
|
74
|
+
return value ?? '';
|
|
75
|
+
};
|
|
76
|
+
//# sourceMappingURL=utils.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.js","sourceRoot":"../../../../../src","sources":["form-builder-v2/components/WhenEditor/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import type {When} from '../../../form-generator-v2/types';\n\nexport type Condition = {field: string; operator: '===' | '!=='; value: string | boolean};\nexport type Combinator = {operator: '&&' | '||'};\n\nexport const OPERATOR_OPTIONS = [\n {value: '===', content: 'is'},\n {value: '!==', content: 'is not'},\n];\n\nexport const COMBINATOR_OPTIONS = [\n {value: '&&', content: 'AND'},\n {value: '||', content: 'OR'},\n];\n\nexport const parse = (\n when: When | undefined,\n): {conditions: Condition[]; combinators: Combinator[]} => {\n const conditions: Condition[] = [];\n const combinators: Combinator[] = [];\n if (!when) return {conditions, combinators};\n\n let lastWasCombinator = false;\n\n for (const entry of when) {\n const isCondition =\n Boolean(entry.field) && (entry.operator === '===' || entry.operator === '!==');\n const isCombinator = !entry.field && (entry.operator === '&&' || entry.operator === '||');\n\n if (isCondition) {\n conditions.push({\n field: entry.field as string,\n operator: entry.operator as '===' | '!==',\n value: entry.value ?? '',\n });\n lastWasCombinator = false;\n continue;\n }\n\n if (isCombinator) {\n if (conditions.length === 0 || lastWasCombinator) continue;\n combinators.push({operator: entry.operator as '&&' | '||'});\n lastWasCombinator = true;\n }\n }\n\n if (lastWasCombinator) {\n combinators.pop();\n }\n\n while (combinators.length < conditions.length - 1) {\n combinators.push({operator: '&&'});\n }\n\n if (combinators.length > Math.max(0, conditions.length - 1)) {\n combinators.length = Math.max(0, conditions.length - 1);\n }\n\n return {conditions, combinators};\n};\n\nexport const serialize = (conditions: Condition[], combinators: Combinator[]): When | undefined => {\n if (conditions.length === 0) return undefined;\n const out: When = [];\n conditions.forEach((cond, i) => {\n if (i > 0) {\n const combinator = combinators[i - 1] ?? {operator: '&&'};\n out.push({operator: combinator.operator});\n }\n out.push({\n field: cond.field,\n operator: cond.operator,\n value: cond.value,\n });\n });\n return out;\n};\n\nexport const coerceValue = (raw: string): string | boolean => {\n if (raw === 'true') return true;\n if (raw === 'false') return false;\n return raw;\n};\n\nexport const displayValue = (value: string | boolean | undefined): string => {\n if (value === true) return 'true';\n if (value === false) return 'false';\n return value ?? '';\n};\n"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { FormContextType, FormField } from "../types.js";
|
|
3
|
+
export declare const FormContext: React.Context<FormContextType | null>;
|
|
4
|
+
interface FormProviderProps {
|
|
5
|
+
children: React.ReactNode;
|
|
6
|
+
formFields: FormField[];
|
|
7
|
+
onChange?: (fields: FormField[]) => void;
|
|
8
|
+
}
|
|
9
|
+
export declare const FormProvider: ({ children, formFields, onChange }: FormProviderProps) => import("react/jsx-runtime").JSX.Element;
|
|
10
|
+
export declare const useFormContext: () => FormContextType;
|
|
11
|
+
export {};
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import * as React from 'react';
|
|
3
|
+
import { useFormFields } from "./useFormFields.js";
|
|
4
|
+
export const FormContext = React.createContext(null);
|
|
5
|
+
export const FormProvider = ({ children, formFields, onChange }) => {
|
|
6
|
+
const value = useFormFields({ initialFields: formFields, onChange });
|
|
7
|
+
return _jsx(FormContext.Provider, { value: value, children: children });
|
|
8
|
+
};
|
|
9
|
+
export const useFormContext = () => {
|
|
10
|
+
const context = React.useContext(FormContext);
|
|
11
|
+
if (!context) {
|
|
12
|
+
throw new Error('useFormContext must be used within a FormProvider');
|
|
13
|
+
}
|
|
14
|
+
return context;
|
|
15
|
+
};
|
|
16
|
+
//# sourceMappingURL=FormContext.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"FormContext.js","sourceRoot":"../../../../src","sources":["form-builder-v2/hooks/FormContext.tsx"],"names":[],"mappings":";AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAI/B,OAAO,EAAC,aAAa,EAAC,2BAAwB;AAE9C,MAAM,CAAC,MAAM,WAAW,GAAG,KAAK,CAAC,aAAa,CAAyB,IAAI,CAAC,CAAC;AAQ7E,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,EAAC,QAAQ,EAAE,UAAU,EAAE,QAAQ,EAAoB,EAAE,EAAE;IAChF,MAAM,KAAK,GAAG,aAAa,CAAC,EAAC,aAAa,EAAE,UAAU,EAAE,QAAQ,EAAC,CAAC,CAAC;IACnE,OAAO,KAAC,WAAW,CAAC,QAAQ,IAAC,KAAK,EAAE,KAAK,YAAG,QAAQ,GAAwB,CAAC;AACjF,CAAC,CAAC;AAEF,MAAM,CAAC,MAAM,cAAc,GAAG,GAAoB,EAAE;IAChD,MAAM,OAAO,GAAG,KAAK,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;IAC9C,IAAI,CAAC,OAAO,EAAE,CAAC;QACX,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;IACzE,CAAC;IACD,OAAO,OAAO,CAAC;AACnB,CAAC,CAAC","sourcesContent":["import * as React from 'react';\n\nimport {FormContextType, FormField} from '../types';\n\nimport {useFormFields} from './useFormFields';\n\nexport const FormContext = React.createContext<FormContextType | null>(null);\n\ninterface FormProviderProps {\n children: React.ReactNode;\n formFields: FormField[];\n onChange?: (fields: FormField[]) => void;\n}\n\nexport const FormProvider = ({children, formFields, onChange}: FormProviderProps) => {\n const value = useFormFields({initialFields: formFields, onChange});\n return <FormContext.Provider value={value}>{children}</FormContext.Provider>;\n};\n\nexport const useFormContext = (): FormContextType => {\n const context = React.useContext(FormContext);\n if (!context) {\n throw new Error('useFormContext must be used within a FormProvider');\n }\n return context;\n};\n"]}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { BuilderFieldType, FieldUpdate, FormField } from "../types.js";
|
|
2
|
+
interface UseFormFieldsProps {
|
|
3
|
+
initialFields: FormField[];
|
|
4
|
+
onChange?: (fields: FormField[]) => void;
|
|
5
|
+
}
|
|
6
|
+
export declare const useFormFields: ({ initialFields, onChange }: UseFormFieldsProps) => {
|
|
7
|
+
formFields: FormField[];
|
|
8
|
+
selectedFieldId: string | null;
|
|
9
|
+
addField: (type: BuilderFieldType) => void;
|
|
10
|
+
addFieldToSection: (sectionId: string, type: BuilderFieldType) => void;
|
|
11
|
+
insertFieldBefore: (targetId: string, type: BuilderFieldType) => void;
|
|
12
|
+
insertFieldAfter: (targetId: string, type: BuilderFieldType) => void;
|
|
13
|
+
moveFieldToSection: (fieldId: string, targetSectionId: string) => void;
|
|
14
|
+
removeField: (fieldId: string) => void;
|
|
15
|
+
duplicateField: (fieldId: string) => void;
|
|
16
|
+
updateField: (fieldId: string, updates: FieldUpdate) => void;
|
|
17
|
+
moveFieldUp: (fieldId: string) => void;
|
|
18
|
+
moveFieldDown: (fieldId: string) => void;
|
|
19
|
+
setAllFields: (next: FormField[]) => void;
|
|
20
|
+
resetForm: () => void;
|
|
21
|
+
selectField: (fieldId: string | null) => void;
|
|
22
|
+
};
|
|
23
|
+
export {};
|
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { createDefaultField } from "../utils/fieldDefaults.js";
|
|
3
|
+
import { collectNames, maxIdNumericSuffix, prefixNameForArrayMode, stripArrayModePrefix, } from "../utils/fieldNames.js";
|
|
4
|
+
import { findFieldById } from "../utils/fieldTree.js";
|
|
5
|
+
import { addChildToSectionDeep, containsId, duplicateFieldDeep, insertAtTargetDeep, isArrayModeSection, removeFieldDeep, swapInListDeep, updateFieldDeep, } from "../utils/fieldTreeOps.js";
|
|
6
|
+
export const useFormFields = ({ initialFields, onChange }) => {
|
|
7
|
+
const [formFields, setFormFields] = React.useState(initialFields);
|
|
8
|
+
const [selectedFieldId, setSelectedFieldId] = React.useState(null);
|
|
9
|
+
const idCounter = React.useRef(maxIdNumericSuffix(initialFields) + 1);
|
|
10
|
+
const generateId = React.useCallback(() => {
|
|
11
|
+
const id = `fb2_${idCounter.current}`;
|
|
12
|
+
idCounter.current += 1;
|
|
13
|
+
return id;
|
|
14
|
+
}, []);
|
|
15
|
+
const generateName = React.useCallback(() => `field_${idCounter.current}`, []);
|
|
16
|
+
const commit = React.useCallback((next) => {
|
|
17
|
+
setFormFields(next);
|
|
18
|
+
onChange?.(next);
|
|
19
|
+
}, [onChange]);
|
|
20
|
+
const addField = React.useCallback((type) => {
|
|
21
|
+
const id = generateId();
|
|
22
|
+
const name = generateName();
|
|
23
|
+
const newField = createDefaultField(type, name, id);
|
|
24
|
+
commit([...formFields, newField]);
|
|
25
|
+
}, [commit, formFields, generateId, generateName]);
|
|
26
|
+
const addFieldToSection = React.useCallback((sectionId, type) => {
|
|
27
|
+
const id = generateId();
|
|
28
|
+
const baseName = generateName();
|
|
29
|
+
const arrayMode = isArrayModeSection(formFields, sectionId);
|
|
30
|
+
const name = arrayMode ? prefixNameForArrayMode(baseName) : baseName;
|
|
31
|
+
const newChild = createDefaultField(type, name, id);
|
|
32
|
+
commit(addChildToSectionDeep(formFields, sectionId, newChild));
|
|
33
|
+
}, [commit, formFields, generateId, generateName]);
|
|
34
|
+
const insertFieldRelative = React.useCallback((targetId, type, offset) => {
|
|
35
|
+
const makeField = (arrayMode) => {
|
|
36
|
+
const id = generateId();
|
|
37
|
+
const baseName = generateName();
|
|
38
|
+
const name = arrayMode ? prefixNameForArrayMode(baseName) : baseName;
|
|
39
|
+
return createDefaultField(type, name, id);
|
|
40
|
+
};
|
|
41
|
+
const result = insertAtTargetDeep(formFields, targetId, offset, makeField);
|
|
42
|
+
if (result === null) {
|
|
43
|
+
const fallback = makeField(false);
|
|
44
|
+
commit([...formFields, fallback]);
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
commit(result);
|
|
48
|
+
}, [commit, formFields, generateId, generateName]);
|
|
49
|
+
const insertFieldBefore = React.useCallback((targetId, type) => insertFieldRelative(targetId, type, 0), [insertFieldRelative]);
|
|
50
|
+
const insertFieldAfter = React.useCallback((targetId, type) => insertFieldRelative(targetId, type, 1), [insertFieldRelative]);
|
|
51
|
+
const removeField = React.useCallback((fieldId) => {
|
|
52
|
+
commit(removeFieldDeep(formFields, fieldId));
|
|
53
|
+
}, [commit, formFields]);
|
|
54
|
+
const moveFieldToSection = React.useCallback((fieldId, targetSectionId) => {
|
|
55
|
+
if (fieldId === targetSectionId)
|
|
56
|
+
return;
|
|
57
|
+
const sourceField = findFieldById(formFields, fieldId);
|
|
58
|
+
if (!sourceField)
|
|
59
|
+
return;
|
|
60
|
+
const targetField = findFieldById(formFields, targetSectionId);
|
|
61
|
+
if (!targetField || targetField.type !== 'section') {
|
|
62
|
+
return;
|
|
63
|
+
}
|
|
64
|
+
if (sourceField.type === 'section' && containsId(sourceField, targetSectionId)) {
|
|
65
|
+
return;
|
|
66
|
+
}
|
|
67
|
+
const withoutField = removeFieldDeep(formFields, fieldId);
|
|
68
|
+
const targetIsArray = isArrayModeSection(withoutField, targetSectionId);
|
|
69
|
+
let movedField = sourceField;
|
|
70
|
+
if (sourceField.type !== 'section' && sourceField.type !== 'text') {
|
|
71
|
+
const currentHasIndex = sourceField.name.includes('{{index}}');
|
|
72
|
+
if (targetIsArray && !currentHasIndex) {
|
|
73
|
+
movedField = {
|
|
74
|
+
...sourceField,
|
|
75
|
+
name: prefixNameForArrayMode(sourceField.name),
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
else if (!targetIsArray && currentHasIndex) {
|
|
79
|
+
movedField = {
|
|
80
|
+
...sourceField,
|
|
81
|
+
name: stripArrayModePrefix(sourceField.name),
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
const result = addChildToSectionDeep(withoutField, targetSectionId, movedField);
|
|
86
|
+
commit(result);
|
|
87
|
+
}, [commit, formFields]);
|
|
88
|
+
const duplicateField = React.useCallback((fieldId) => {
|
|
89
|
+
const existingNames = collectNames(formFields);
|
|
90
|
+
const result = duplicateFieldDeep(formFields, fieldId, generateId, existingNames);
|
|
91
|
+
if (result !== null) {
|
|
92
|
+
commit(result);
|
|
93
|
+
}
|
|
94
|
+
}, [commit, formFields, generateId]);
|
|
95
|
+
const updateField = React.useCallback((fieldId, updates) => {
|
|
96
|
+
commit(updateFieldDeep(formFields, fieldId, updates));
|
|
97
|
+
}, [commit, formFields]);
|
|
98
|
+
const moveFieldUp = React.useCallback((fieldId) => {
|
|
99
|
+
const result = swapInListDeep(formFields, fieldId, -1);
|
|
100
|
+
if (result.handled) {
|
|
101
|
+
commit(result.fields);
|
|
102
|
+
}
|
|
103
|
+
}, [commit, formFields]);
|
|
104
|
+
const moveFieldDown = React.useCallback((fieldId) => {
|
|
105
|
+
const result = swapInListDeep(formFields, fieldId, 1);
|
|
106
|
+
if (result.handled) {
|
|
107
|
+
commit(result.fields);
|
|
108
|
+
}
|
|
109
|
+
}, [commit, formFields]);
|
|
110
|
+
const setAllFields = React.useCallback((next) => {
|
|
111
|
+
idCounter.current = Math.max(idCounter.current, maxIdNumericSuffix(next) + 1);
|
|
112
|
+
commit(next);
|
|
113
|
+
}, [commit]);
|
|
114
|
+
const resetForm = React.useCallback(() => {
|
|
115
|
+
commit([]);
|
|
116
|
+
setSelectedFieldId(null);
|
|
117
|
+
}, [commit]);
|
|
118
|
+
const selectField = React.useCallback((fieldId) => {
|
|
119
|
+
setSelectedFieldId(fieldId);
|
|
120
|
+
}, []);
|
|
121
|
+
return {
|
|
122
|
+
formFields,
|
|
123
|
+
selectedFieldId,
|
|
124
|
+
addField,
|
|
125
|
+
addFieldToSection,
|
|
126
|
+
insertFieldBefore,
|
|
127
|
+
insertFieldAfter,
|
|
128
|
+
moveFieldToSection,
|
|
129
|
+
removeField,
|
|
130
|
+
duplicateField,
|
|
131
|
+
updateField,
|
|
132
|
+
moveFieldUp,
|
|
133
|
+
moveFieldDown,
|
|
134
|
+
setAllFields,
|
|
135
|
+
resetForm,
|
|
136
|
+
selectField,
|
|
137
|
+
};
|
|
138
|
+
};
|
|
139
|
+
//# sourceMappingURL=useFormFields.js.map
|