From e95b3b03f342702a6843951ad3c61d3268fbd6cb Mon Sep 17 00:00:00 2001 From: Dan Funk Date: Tue, 12 Sep 2023 16:21:18 -0400 Subject: [PATCH] Feature/editor cursor jumping (#485) * To avoid the cursor jumps in the Monoco editor, never use "value=xxx" in conjuntion with "onChange". Use defaultValue instead, and make sure that the editor is rerendered when that value is changed. * linting --- .../ReactFormBuilder/ReactFormBuilder.tsx | 110 ++++++++++++++---- .../src/routes/ProcessModelEditDiagram.tsx | 14 ++- 2 files changed, 95 insertions(+), 29 deletions(-) diff --git a/spiffworkflow-frontend/src/components/ReactFormBuilder/ReactFormBuilder.tsx b/spiffworkflow-frontend/src/components/ReactFormBuilder/ReactFormBuilder.tsx index 1d99f9e0..436f7e52 100644 --- a/spiffworkflow-frontend/src/components/ReactFormBuilder/ReactFormBuilder.tsx +++ b/spiffworkflow-frontend/src/components/ReactFormBuilder/ReactFormBuilder.tsx @@ -1,4 +1,4 @@ -import React, { useCallback, useEffect, useState } from 'react'; +import React, { useCallback, useEffect, useState, useRef } from 'react'; import Editor from '@monaco-editor/react'; // eslint-disable-next-line import/no-extraneous-dependencies import merge from 'lodash/merge'; @@ -37,6 +37,7 @@ export default function ReactFormBuilder({ const DATA_EXTENSION = '-exampledata.json'; const [fetchFailed, setFetchFailed] = useState(false); + const [ready, setReady] = useState(false); const [strSchema, setStrSchema] = useState(''); const [debouncedStrSchema] = useDebounce(strSchema, 500); @@ -55,6 +56,24 @@ export default function ReactFormBuilder({ const [baseFileName, setBaseFileName] = useState(''); const [newFileName, setNewFileName] = useState(''); + /** + * This section gives us direct pointers to the monoco editors so that + * we can update their values. Using state variables directly on the monoco editor + * causes the cursor to jump to the bottom if two letters are pressed simultaneously. + */ + const schemaEditorRef = useRef(null); + const uiEditorRef = useRef(null); + const dataEditorRef = useRef(null); + function handleSchemaEditorDidMount(editor: any) { + schemaEditorRef.current = editor; + } + function handleUiEditorDidMount(editor: any) { + uiEditorRef.current = editor; + } + function handleDataEditorDidMount(editor: any) { + dataEditorRef.current = editor; + } + const saveFile = useCallback( (file: File, create: boolean = false) => { let httpMethod = 'PUT'; @@ -88,7 +107,16 @@ export default function ReactFormBuilder({ }; const isReady = () => { - return strSchema !== '' && strUI !== '' && strFormData !== ''; + // Use a ready flag so that we still allow people to completely delete + // the schema, ui or data if they want to clear it out. + if (ready) { + return true; + } + if (strSchema !== '' && strUI !== '' && strFormData !== '') { + setReady(true); + return true; + } + return false; }; // Auto save schema changes @@ -114,7 +142,7 @@ export default function ReactFormBuilder({ useEffect(() => { /** - * we need to run the schema and ui through a backend call before rendering the form + * we need to run the schema and ui through a backend call before rendering the form, * so it can handle certain server side changes, such as jinja rendering and populating dropdowns, etc. */ const url: string = '/tasks/prepare-form'; @@ -173,6 +201,51 @@ export default function ReactFormBuilder({ setSelectedIndex(evt.selectedIndex); }; + const updateStrSchema = (value: string) => { + if (schemaEditorRef && schemaEditorRef.current) { + // @ts-ignore + schemaEditorRef.current.setValue(value); + } + }; + + const updateStrUi = (value: string) => { + if (uiEditorRef && uiEditorRef.current) { + // @ts-ignore + uiEditorRef.current.setValue(value); + } + }; + + const updateStrData = (value: string) => { + // Only update the data if it is different from what is already there, this prevents + // cursor from jumping to the top each time you type a letter. + if ( + dataEditorRef && + dataEditorRef.current && + // @ts-ignore + value !== dataEditorRef.current.getValue() + ) { + // @ts-ignore + dataEditorRef.current.setValue(value); + } + }; + + function updateData(newData: object) { + setFormData(newData); + const newDataStr = JSON.stringify(newData, null, 2); + if (newDataStr !== strFormData) { + updateStrData(newDataStr); + } + } + + function updateDataFromStr(newDataStr: string) { + try { + const newData = JSON.parse(newDataStr); + setFormData(newData); + } catch (e) { + /* empty */ + } + } + function setJsonSchemaFromResponseJson(result: any) { setStrSchema(result.file_contents); } @@ -232,29 +305,13 @@ export default function ReactFormBuilder({ function insertFields(schema: any, ui: any, data: any) { setFormData(merge(formData, data)); - setStrFormData(JSON.stringify(formData, null, 2)); + updateStrData(JSON.stringify(formData, null, 2)); const tempSchema = merge(JSON.parse(strSchema), schema); - setStrSchema(JSON.stringify(tempSchema, null, 2)); + updateStrSchema(JSON.stringify(tempSchema, null, 2)); const tempUI = merge(JSON.parse(strUI), ui); - setStrUI(JSON.stringify(tempUI, null, 2)); - } - - function updateData(newData: object) { - setFormData(newData); - const newDataStr = JSON.stringify(newData, null, 2); - if (newDataStr !== strFormData) { - setStrFormData(newDataStr); - } - } - function updateDataFromStr(newDataStr: string) { - try { - const newData = JSON.parse(newDataStr); - setFormData(newData); - } catch (e) { - /* empty */ - } + updateStrUi(JSON.stringify(tempUI, null, 2)); } if (!isReady()) { @@ -340,8 +397,9 @@ export default function ReactFormBuilder({ height={600} width="auto" defaultLanguage="json" - value={strSchema} + defaultValue={strSchema} onChange={(value) => setStrSchema(value || '')} + onMount={handleSchemaEditorDidMount} /> @@ -359,8 +417,9 @@ export default function ReactFormBuilder({ height={600} width="auto" defaultLanguage="json" - value={strUI} + defaultValue={strUI} onChange={(value) => setStrUI(value || '')} + onMount={handleUiEditorDidMount} /> @@ -372,8 +431,9 @@ export default function ReactFormBuilder({ height={600} width="auto" defaultLanguage="json" - value={strFormData} + defaultValue={strFormData} onChange={(value: any) => updateDataFromStr(value || '')} + onMount={handleDataEditorDidMount} /> diff --git a/spiffworkflow-frontend/src/routes/ProcessModelEditDiagram.tsx b/spiffworkflow-frontend/src/routes/ProcessModelEditDiagram.tsx index 4211223c..0c9567e9 100644 --- a/spiffworkflow-frontend/src/routes/ProcessModelEditDiagram.tsx +++ b/spiffworkflow-frontend/src/routes/ProcessModelEditDiagram.tsx @@ -432,7 +432,7 @@ export default function ProcessModelEditDiagram() { const unitTestModdleElements = extensionElements .get('values') .filter(function getInstanceOfType(e: any) { - return e.$instanceOf('spiffworkflow:unitTests'); + return e.$instanceOf('spiffworkflow:UnitTests'); })[0]; if (unitTestModdleElements) { return unitTestModdleElements.unitTests; @@ -782,7 +782,7 @@ export default function ProcessModelEditDiagram() { width="auto" defaultLanguage="json" options={Object.assign(jsonEditorOptions(), {})} - value={inputJson} + defaultValue={inputJson} onChange={handleEditorScriptTestUnitInputChange} /> @@ -795,7 +795,7 @@ export default function ProcessModelEditDiagram() { width="auto" defaultLanguage="json" options={Object.assign(jsonEditorOptions(), {})} - value={outputJson} + defaultValue={outputJson} onChange={handleEditorScriptTestUnitOutputChange} /> @@ -807,19 +807,25 @@ export default function ProcessModelEditDiagram() { return null; }; const scriptEditor = () => { + if (!showScriptEditor) { + return null; + } return ( ); }; const scriptEditorAndTests = () => { + if (!showScriptEditor) { + return null; + } let scriptName = ''; if (scriptElement) { scriptName = (scriptElement as any).di.bpmnElement.name;