From bdd0361452c0888c13e7e48d36631faed1db98a9 Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Wed, 13 Sep 2023 15:13:54 +0200 Subject: [PATCH 01/12] remove unnecessary empty string to fix error --- src/zui/ZUITextEditor/TextElement.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/zui/ZUITextEditor/TextElement.tsx b/src/zui/ZUITextEditor/TextElement.tsx index 7cc7952074..013c08cf93 100644 --- a/src/zui/ZUITextEditor/TextElement.tsx +++ b/src/zui/ZUITextEditor/TextElement.tsx @@ -22,7 +22,7 @@ const TextElement: React.FunctionComponent = ({ if (element.type === 'bulleted-list') { return ( ); } From 79fc184d5414dfaaa11aa5c2b9d1198f8c8178e6 Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Wed, 13 Sep 2023 16:01:03 +0200 Subject: [PATCH 02/12] update slate and slate-react dependencies --- package.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index 658ec9f527..50e948f3b7 100644 --- a/package.json +++ b/package.json @@ -76,9 +76,9 @@ "remark-gfm": "^3.0.1", "remark-parse": "^10.0.1", "remark-slate": "^1.8.6", - "slate": "^0.78.0", + "slate": "^0.94.1", "slate-history": "^0.66.0", - "slate-react": "^0.77.4", + "slate-react": "^0.98.3", "slugify": "^1.6.5", "tsconfig-paths": "^4.1.2", "unified": "^10.1.2", From 2963d7580c08831a8c944ab248fade9b875cab62 Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Wed, 13 Sep 2023 16:01:36 +0200 Subject: [PATCH 03/12] update value property in Slate component due to upgrade of dependency --- src/zui/ZUITextEditor/index.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/zui/ZUITextEditor/index.tsx b/src/zui/ZUITextEditor/index.tsx index a5a5f7e4d8..83e5d24e78 100644 --- a/src/zui/ZUITextEditor/index.tsx +++ b/src/zui/ZUITextEditor/index.tsx @@ -124,7 +124,7 @@ const ZUITextEditor: React.FunctionComponent = ({ onChange(slateToMarkdown(slateArray))} - value={initialValueSlate} + initialValue={initialValueSlate} > Date: Wed, 13 Sep 2023 16:29:12 +0200 Subject: [PATCH 04/12] remove focus border --- src/zui/ZUITextEditor/index.tsx | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/zui/ZUITextEditor/index.tsx b/src/zui/ZUITextEditor/index.tsx index 83e5d24e78..f8aef6e90e 100644 --- a/src/zui/ZUITextEditor/index.tsx +++ b/src/zui/ZUITextEditor/index.tsx @@ -134,7 +134,10 @@ const ZUITextEditor: React.FunctionComponent = ({ renderElement={renderElement} renderLeaf={renderLeaf} spellCheck - style={{ overflowY: 'scroll' }} + style={{ + overflowY: 'scroll', + outline: 'none', + }} /> From ef611bbb8199301a3ee63e234e6371401579d574 Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Wed, 13 Sep 2023 16:33:18 +0200 Subject: [PATCH 05/12] sort alphabetically the properties --- src/zui/ZUITextEditor/index.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/zui/ZUITextEditor/index.tsx b/src/zui/ZUITextEditor/index.tsx index f8aef6e90e..473f23441a 100644 --- a/src/zui/ZUITextEditor/index.tsx +++ b/src/zui/ZUITextEditor/index.tsx @@ -123,8 +123,8 @@ const ZUITextEditor: React.FunctionComponent = ({ {initialValueSlate && ( onChange(slateToMarkdown(slateArray))} initialValue={initialValueSlate} + onChange={(slateArray) => onChange(slateToMarkdown(slateArray))} > = ({ renderLeaf={renderLeaf} spellCheck style={{ - overflowY: 'scroll', outline: 'none', + overflowY: 'scroll', }} /> From 7085145112a4604780b35fd24ddb132e1b331ea6 Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Wed, 13 Sep 2023 16:56:27 +0200 Subject: [PATCH 06/12] add yarn.lock file for upgrade dependencies --- yarn.lock | 45 ++++++++++++++++++++++++++++----------------- 1 file changed, 28 insertions(+), 17 deletions(-) diff --git a/yarn.lock b/yarn.lock index b26b8a303a..d242edf3f0 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2221,6 +2221,11 @@ "@jridgewell/resolve-uri" "^3.0.3" "@jridgewell/sourcemap-codec" "^1.4.10" +"@juggle/resize-observer@^3.4.0": + version "3.4.0" + resolved "https://registry.yarnpkg.com/@juggle/resize-observer/-/resize-observer-3.4.0.tgz#08d6c5e20cf7e4cc02fd181c4b0c225cd31dbb60" + integrity sha512-dfLbk+PwWvFzSxwk3n5ySL0hfBog779o8h68wK/7/APo/7cgyWp5jcXockbxdk5kFRkbeXWm4Fbi9FrdN381sA== + "@mdx-js/mdx@^1.6.22": version "1.6.22" resolved "https://registry.yarnpkg.com/@mdx-js/mdx/-/mdx-1.6.22.tgz#8a723157bf90e78f17dc0f27995398e6c731f1ba" @@ -4362,7 +4367,12 @@ dependencies: "@types/geojson" "*" -"@types/lodash@^4.14.149", "@types/lodash@^4.14.167": +"@types/lodash@^4.14.149": + version "4.14.198" + resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.198.tgz#4d27465257011aedc741a809f1269941fa2c5d4c" + integrity sha512-trNJ/vtMZYMLhfN45uLq4ShQSw0/S7xCTLLVM+WM1rmFpba/VS42jVUgaO3w/NOLiWR/09lnYk0yMaA/atdIsg== + +"@types/lodash@^4.14.167": version "4.14.182" resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.182.tgz#05301a4d5e62963227eaafe0ce04dd77c54ea5c2" integrity sha512-/THyiqyQAP9AfARo4pF+aCGcyiQ94tX/Is2I7HofNRqoYLgN1PBoOWu2/zTA5zMxzP5EFutMtWtGAFRKUe961Q== @@ -6582,10 +6592,10 @@ compression@^1.7.4: safe-buffer "5.1.2" vary "~1.1.2" -compute-scroll-into-view@^1.0.17: - version "1.0.17" - resolved "https://registry.yarnpkg.com/compute-scroll-into-view/-/compute-scroll-into-view-1.0.17.tgz#6a88f18acd9d42e9cf4baa6bec7e0522607ab7ab" - integrity sha512-j4dx+Fb0URmzbwwMUrhqWM2BEWHdFGx+qZ9qqASHRPqvTYdqvWnHg0H1hIbcyLnvgnoNAVMlwkepyqM3DaIFUg== +compute-scroll-into-view@^1.0.20: + version "1.0.20" + resolved "https://registry.yarnpkg.com/compute-scroll-into-view/-/compute-scroll-into-view-1.0.20.tgz#1768b5522d1172754f5d0c9b02de3af6be506a43" + integrity sha512-UCB0ioiyj8CRjtrvaceBLqqhZCVP+1B8+NWQhmdsm0VXOJtobBCf1dBQmebCCo34qZmUwZfIH2MZLqNHazrfjg== concat-map@0.0.1: version "0.0.1" @@ -14275,11 +14285,11 @@ schema-utils@^3.0.0, schema-utils@^3.1.0, schema-utils@^3.1.1: ajv-keywords "^3.5.2" scroll-into-view-if-needed@^2.2.20: - version "2.2.29" - resolved "https://registry.yarnpkg.com/scroll-into-view-if-needed/-/scroll-into-view-if-needed-2.2.29.tgz#551791a84b7e2287706511f8c68161e4990ab885" - integrity sha512-hxpAR6AN+Gh53AdAimHM6C8oTN1ppwVZITihix+WqalywBeFcQ6LdQP5ABNl26nX8GTEL7VT+b8lKpdqq65wXg== + version "2.2.31" + resolved "https://registry.yarnpkg.com/scroll-into-view-if-needed/-/scroll-into-view-if-needed-2.2.31.tgz#d3c482959dc483e37962d1521254e3295d0d1587" + integrity sha512-dGCXy99wZQivjmjIqihaBQNjryrz5rueJY7eHfTdyWEiR4ttYpsajb14rn9s5d4DY4EcY6+4+U/maARBXJedkA== dependencies: - compute-scroll-into-view "^1.0.17" + compute-scroll-into-view "^1.0.20" "semver@2 || 3 || 4 || 5", semver@^5.4.1, semver@^5.5.0, semver@^5.6.0: version "5.7.1" @@ -14488,11 +14498,12 @@ slate-history@^0.66.0: dependencies: is-plain-object "^5.0.0" -slate-react@^0.77.4: - version "0.77.4" - resolved "https://registry.yarnpkg.com/slate-react/-/slate-react-0.77.4.tgz#b2532b57e47c4e088b1583ed8befc6e238a15ad5" - integrity sha512-e3gYuEhjbEX4IhydC4kWZgcvH5fgJkJMuu9paarO6gdW5vRRSWnys/EbOs5ALp4e9NWr5u1XzQkJnbpyW5AqBA== +slate-react@^0.98.3: + version "0.98.3" + resolved "https://registry.yarnpkg.com/slate-react/-/slate-react-0.98.3.tgz#5090d269d69186f3ec2a6b5862d2645f01772eda" + integrity sha512-p1BnF9eRyRM0i5hkgOb11KgmpWLQm9Zyp6jVkOAj5fPdIGheKhg48Z7aWKrayeJ4nmRyi/NjRZz/io5hQcphmw== dependencies: + "@juggle/resize-observer" "^3.4.0" "@types/is-hotkey" "^0.1.1" "@types/lodash" "^4.14.149" direction "^1.0.3" @@ -14502,10 +14513,10 @@ slate-react@^0.77.4: scroll-into-view-if-needed "^2.2.20" tiny-invariant "1.0.6" -slate@^0.78.0: - version "0.78.0" - resolved "https://registry.yarnpkg.com/slate/-/slate-0.78.0.tgz#cd0328d22b0a99c543987d2a2dd30903bb950ee9" - integrity sha512-VwQ0RafT3JPf9SFrXI02Dh3S4Iz9en7d1nn50C/LJjjqjfgv+a2ORbgWMdYjhycPYldaxJwcI3OpP9D1g4SXEg== +slate@^0.94.1: + version "0.94.1" + resolved "https://registry.yarnpkg.com/slate/-/slate-0.94.1.tgz#13b0ba7d0a7eeb0ec89a87598e9111cbbd685696" + integrity sha512-GH/yizXr1ceBoZ9P9uebIaHe3dC/g6Plpf9nlUwnvoyf6V1UOYrRwkabtOCd3ZfIGxomY4P7lfgLr7FPH8/BKA== dependencies: immer "^9.0.6" is-plain-object "^5.0.0" From 49c64fd3e983ffb067f85792480f081ccc5a6372 Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Fri, 15 Sep 2023 15:13:36 +0200 Subject: [PATCH 07/12] add helper function to check if node should be removed --- src/zui/ZUITextEditor/helpers.ts | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/src/zui/ZUITextEditor/helpers.ts b/src/zui/ZUITextEditor/helpers.ts index ff6cf11897..0a49bc7543 100644 --- a/src/zui/ZUITextEditor/helpers.ts +++ b/src/zui/ZUITextEditor/helpers.ts @@ -8,6 +8,8 @@ import { Element as SlateElement, Text, Transforms, + NodeEntry, + Ancestor, } from 'slate'; import isHotkey, { isKeyHotkey } from 'is-hotkey'; @@ -225,6 +227,23 @@ const convertSlateToRemarked = ( return convertedChildren; }; +const shouldBeRemoved = (node: NodeEntry): boolean => { + if (node && node[0].hasOwnProperty('children')) { + if ( + 'children' in node[0].children[0] && + node[0].children[0].hasOwnProperty('children') + ) { + if ( + node[0].children[0].children[0] && + node[0].children[0].children[0].text === '' + ) { + return true; + } + } + } + return false; +}; + const slateToMarkdown = (slateArray: Descendant[]): string => { const nodeTypes = { block_quote: 'block-quote', @@ -269,6 +288,7 @@ export { isMarkActive, keyDownHandler, LIST_TYPES, + shouldBeRemoved, slateToMarkdown, withInlines, unwrapLink, From f29816907ea4b747b31332f0ca6334885411ccca Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Fri, 15 Sep 2023 15:27:58 +0200 Subject: [PATCH 08/12] sort imports alphabetically --- src/zui/ZUITextEditor/helpers.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/zui/ZUITextEditor/helpers.ts b/src/zui/ZUITextEditor/helpers.ts index 0a49bc7543..f38bcb12af 100644 --- a/src/zui/ZUITextEditor/helpers.ts +++ b/src/zui/ZUITextEditor/helpers.ts @@ -1,16 +1,16 @@ /* eslint-disable @typescript-eslint/ban-ts-comment */ import isUrl from 'is-url'; -import { BlockType, LeafType, NodeTypes, serialize } from 'remark-slate'; import { + Ancestor, Descendant, Editor, + NodeEntry, Range, Element as SlateElement, Text, Transforms, - NodeEntry, - Ancestor, } from 'slate'; +import { BlockType, LeafType, NodeTypes, serialize } from 'remark-slate'; import isHotkey, { isKeyHotkey } from 'is-hotkey'; const LIST_TYPES = ['numbered-list', 'bulleted-list']; From 758ccffa70cb59aedf78a93a96a99fd253a16b13 Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Fri, 15 Sep 2023 15:28:32 +0200 Subject: [PATCH 09/12] use object.prototype to access property --- src/zui/ZUITextEditor/helpers.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/zui/ZUITextEditor/helpers.ts b/src/zui/ZUITextEditor/helpers.ts index f38bcb12af..9a44d6e401 100644 --- a/src/zui/ZUITextEditor/helpers.ts +++ b/src/zui/ZUITextEditor/helpers.ts @@ -228,10 +228,10 @@ const convertSlateToRemarked = ( }; const shouldBeRemoved = (node: NodeEntry): boolean => { - if (node && node[0].hasOwnProperty('children')) { + if (node && Object.prototype.hasOwnProperty.call(node[0], 'children')) { if ( 'children' in node[0].children[0] && - node[0].children[0].hasOwnProperty('children') + Object.prototype.hasOwnProperty.call(node[0].children[0], 'children') ) { if ( node[0].children[0].children[0] && From 06f49d8af1e2b584c62cc622871c6be02c1f454a Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Fri, 15 Sep 2023 15:29:38 +0200 Subject: [PATCH 10/12] update deleteBackwards function in editor to fix missing bullet point at the root of editor --- src/zui/ZUITextEditor/index.tsx | 43 +++++++++++++++++++++++++++++++-- 1 file changed, 41 insertions(+), 2 deletions(-) diff --git a/src/zui/ZUITextEditor/index.tsx b/src/zui/ZUITextEditor/index.tsx index 473f23441a..28fca5b230 100644 --- a/src/zui/ZUITextEditor/index.tsx +++ b/src/zui/ZUITextEditor/index.tsx @@ -4,7 +4,14 @@ import { makeStyles } from '@mui/styles'; import { markdownToSlate } from './utils/markdownToSlate'; import { withHistory } from 'slate-history'; import { Box, ClickAwayListener, Collapse } from '@mui/material'; -import { createEditor, Descendant, Editor, Transforms } from 'slate'; +import { + createEditor, + deleteBackward, + Descendant, + Editor, + Node, + Transforms, +} from 'slate'; import { Editable, ReactEditor, @@ -21,7 +28,12 @@ import TextElement from './TextElement'; import theme from 'theme'; import Toolbar from './Toolbar'; import { ZetkinFileUploadChip } from 'zui/ZUIFileChip'; -import { keyDownHandler, slateToMarkdown, withInlines } from './helpers'; +import { + keyDownHandler, + shouldBeRemoved, + slateToMarkdown, + withInlines, +} from './helpers'; const emptySlate = [ { @@ -86,6 +98,33 @@ const ZUITextEditor: React.FunctionComponent = ({ () => withInlines(withHistory(withReact(createEditor()))), [] ); + + //fixes deleting the missing bullet point in empty list in root + editor.deleteBackward = (...args) => { + deleteBackward(editor, ...args); + + const bulletListNode = Editor.above(editor, { + match: (n: Node) => + 'type' in n && + n.type === 'list-item' && + Object.prototype.hasOwnProperty.call(n, 'children'), + }); + + if (bulletListNode) { + if (shouldBeRemoved(bulletListNode)) { + Transforms.setNodes( + editor, + { type: 'paragraph' }, + { + at: bulletListNode[1], + match: (n) => 'type' in n && n.type === 'list-item', + } + ); + } + } + return editor; + }; + const [initialValueSlate, setInitialValueSlate] = useState< Descendant[] | null >(null); From d06fcd31c9a9f534f69acbe0674f57dd09be08dc Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Mon, 18 Sep 2023 16:50:18 +0200 Subject: [PATCH 11/12] trim instructions to avoid white spaces --- .../callAssignments/models/CallerInstructionsModel.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/features/callAssignments/models/CallerInstructionsModel.ts b/src/features/callAssignments/models/CallerInstructionsModel.ts index d3c33c7471..26bbf238d9 100644 --- a/src/features/callAssignments/models/CallerInstructionsModel.ts +++ b/src/features/callAssignments/models/CallerInstructionsModel.ts @@ -50,9 +50,10 @@ export default class CallerInstructionsModel extends ModelBase { return false; } - const lsInstructions = localStorage.getItem(this._key) || ''; + const lsInstructions = localStorage.getItem(this._key)?.trim() || ''; + const dataInstructions = data.instructions.trim(); - return data.instructions != lsInstructions; + return dataInstructions != lsInstructions; } get isSaving(): boolean { From 3f84b0d4ce54ae1fa927274bca45f64b8e9a2f6f Mon Sep 17 00:00:00 2001 From: Rebe R <36491300+rebecarubio@users.noreply.github.com> Date: Mon, 18 Sep 2023 16:58:44 +0200 Subject: [PATCH 12/12] avoid unnecessary render of Slate --- src/zui/ZUITextEditor/index.tsx | 20 ++++++++++++++++---- 1 file changed, 16 insertions(+), 4 deletions(-) diff --git a/src/zui/ZUITextEditor/index.tsx b/src/zui/ZUITextEditor/index.tsx index 28fca5b230..e5093a1f17 100644 --- a/src/zui/ZUITextEditor/index.tsx +++ b/src/zui/ZUITextEditor/index.tsx @@ -20,7 +20,13 @@ import { Slate, withReact, } from 'slate-react'; -import React, { useCallback, useEffect, useMemo, useState } from 'react'; +import React, { + useCallback, + useEffect, + useMemo, + useRef, + useState, +} from 'react'; import './types'; import { FileUpload } from 'features/files/hooks/useFileUploads'; @@ -125,6 +131,7 @@ const ZUITextEditor: React.FunctionComponent = ({ return editor; }; + const markdownValue = useRef(''); const [initialValueSlate, setInitialValueSlate] = useState< Descendant[] | null >(null); @@ -132,8 +139,10 @@ const ZUITextEditor: React.FunctionComponent = ({ useEffect(() => { (async () => { if (initialValue) { - const slate = await markdownToSlate(initialValue as string); - setInitialValueSlate(slate as Descendant[]); + if (initialValue !== markdownValue.current) { + const slate = await markdownToSlate(initialValue); + setInitialValueSlate(slate as Descendant[]); + } } else { setInitialValueSlate(emptySlate); } @@ -163,7 +172,10 @@ const ZUITextEditor: React.FunctionComponent = ({ onChange(slateToMarkdown(slateArray))} + onChange={(slateArray) => { + markdownValue.current = slateToMarkdown(slateArray); + onChange(markdownValue.current); + }} >