@dxos/react-ui-editor 0.8.4-main.67995b8 → 0.8.4-main.72ec0f3
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/dist/lib/browser/{chunk-22UMM3QJ.mjs → chunk-HL3YF6WC.mjs} +2 -2
- package/dist/lib/browser/chunk-HL3YF6WC.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +8336 -6712
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/types/index.mjs +1 -1
- package/dist/lib/node-esm/{chunk-YXYQPV6R.mjs → chunk-YJZGD3LY.mjs} +2 -2
- package/dist/lib/node-esm/chunk-YJZGD3LY.mjs.map +7 -0
- package/dist/lib/node-esm/index.mjs +8336 -6712
- package/dist/lib/node-esm/index.mjs.map +4 -4
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/lib/node-esm/types/index.mjs +1 -1
- package/dist/types/src/components/Editor/Editor.d.ts +37 -15
- package/dist/types/src/components/Editor/Editor.d.ts.map +1 -1
- package/dist/types/src/components/Editor/Editor.stories.d.ts +20 -0
- package/dist/types/src/components/Editor/Editor.stories.d.ts.map +1 -0
- package/dist/types/src/components/EditorContent/EditorContent.d.ts +29 -0
- package/dist/types/src/components/EditorContent/EditorContent.d.ts.map +1 -0
- package/dist/types/src/components/EditorContent/EditorContent.stories.d.ts +26 -0
- package/dist/types/src/components/EditorContent/EditorContent.stories.d.ts.map +1 -0
- package/dist/types/src/components/EditorContent/controller.d.ts +10 -0
- package/dist/types/src/components/EditorContent/controller.d.ts.map +1 -0
- package/dist/types/src/components/EditorContent/index.d.ts +3 -0
- package/dist/types/src/components/EditorContent/index.d.ts.map +1 -0
- package/dist/types/src/components/EditorMenuProvider/EditorMenuProvider.d.ts +36 -0
- package/dist/types/src/components/EditorMenuProvider/EditorMenuProvider.d.ts.map +1 -0
- package/dist/types/src/components/EditorMenuProvider/index.d.ts +7 -0
- package/dist/types/src/components/EditorMenuProvider/index.d.ts.map +1 -0
- package/dist/types/src/components/EditorMenuProvider/menu-presets.d.ts +4 -0
- package/dist/types/src/components/EditorMenuProvider/menu-presets.d.ts.map +1 -0
- package/dist/types/src/components/EditorMenuProvider/menu.d.ts +28 -0
- package/dist/types/src/components/EditorMenuProvider/menu.d.ts.map +1 -0
- package/dist/types/src/components/EditorMenuProvider/popover.d.ts +47 -0
- package/dist/types/src/components/EditorMenuProvider/popover.d.ts.map +1 -0
- package/dist/types/src/components/EditorMenuProvider/useEditorMenu.d.ts +34 -0
- package/dist/types/src/components/EditorMenuProvider/useEditorMenu.d.ts.map +1 -0
- package/dist/types/src/components/EditorMenuProvider/util.d.ts +8 -0
- package/dist/types/src/components/EditorMenuProvider/util.d.ts.map +1 -0
- package/dist/types/src/components/EditorPreviewProvider/EditorPreviewProvider.d.ts +16 -0
- package/dist/types/src/components/EditorPreviewProvider/EditorPreviewProvider.d.ts.map +1 -0
- package/dist/types/src/components/EditorPreviewProvider/index.d.ts +2 -0
- package/dist/types/src/components/EditorPreviewProvider/index.d.ts.map +1 -0
- package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts +26 -2
- package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/actions.d.ts +39 -0
- package/dist/types/src/components/EditorToolbar/actions.d.ts.map +1 -0
- package/dist/types/src/components/EditorToolbar/blocks.d.ts +3 -3
- package/dist/types/src/components/EditorToolbar/blocks.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/formatting.d.ts +3 -3
- package/dist/types/src/components/EditorToolbar/formatting.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/headings.d.ts +3 -3
- package/dist/types/src/components/EditorToolbar/headings.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/image.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/index.d.ts +2 -1
- package/dist/types/src/components/EditorToolbar/index.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/search.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/useEditorToolbar.d.ts +11 -0
- package/dist/types/src/components/EditorToolbar/useEditorToolbar.d.ts.map +1 -0
- package/dist/types/src/components/EditorToolbar/view-mode.d.ts +3 -3
- package/dist/types/src/components/EditorToolbar/view-mode.d.ts.map +1 -1
- package/dist/types/src/components/index.d.ts +4 -2
- package/dist/types/src/components/index.d.ts.map +1 -1
- package/dist/types/src/defaults.d.ts.map +1 -1
- package/dist/types/src/extensions/autocomplete/autocomplete.d.ts +17 -0
- package/dist/types/src/extensions/autocomplete/autocomplete.d.ts.map +1 -0
- package/dist/types/src/extensions/autocomplete/index.d.ts +5 -0
- package/dist/types/src/extensions/autocomplete/index.d.ts.map +1 -0
- package/dist/types/src/extensions/autocomplete/match.d.ts +13 -0
- package/dist/types/src/extensions/autocomplete/match.d.ts.map +1 -0
- package/dist/types/src/extensions/autocomplete/placeholder.d.ts +20 -0
- package/dist/types/src/extensions/autocomplete/placeholder.d.ts.map +1 -0
- package/dist/types/src/extensions/autocomplete/typeahead.d.ts +10 -0
- package/dist/types/src/extensions/autocomplete/typeahead.d.ts.map +1 -0
- package/dist/types/src/extensions/automerge/automerge.d.ts +1 -1
- package/dist/types/src/extensions/automerge/automerge.d.ts.map +1 -1
- package/dist/types/src/extensions/automerge/automerge.stories.d.ts +10 -19
- package/dist/types/src/extensions/automerge/automerge.stories.d.ts.map +1 -1
- package/dist/types/src/extensions/automerge/cursor.d.ts +1 -1
- package/dist/types/src/extensions/automerge/cursor.d.ts.map +1 -1
- package/dist/types/src/extensions/automerge/defs.d.ts +1 -1
- package/dist/types/src/extensions/automerge/defs.d.ts.map +1 -1
- package/dist/types/src/extensions/automerge/sync.d.ts +3 -3
- package/dist/types/src/extensions/automerge/sync.d.ts.map +1 -1
- package/dist/types/src/extensions/automerge/update-automerge.d.ts +1 -1
- package/dist/types/src/extensions/automerge/update-automerge.d.ts.map +1 -1
- package/dist/types/src/extensions/autoscroll.d.ts +20 -0
- package/dist/types/src/extensions/autoscroll.d.ts.map +1 -0
- package/dist/types/src/extensions/awareness/awareness-provider.d.ts +1 -1
- package/dist/types/src/extensions/awareness/awareness-provider.d.ts.map +1 -1
- package/dist/types/src/extensions/blocks.d.ts +2 -0
- package/dist/types/src/extensions/blocks.d.ts.map +1 -0
- package/dist/types/src/extensions/bookmarks.d.ts +12 -0
- package/dist/types/src/extensions/bookmarks.d.ts.map +1 -0
- package/dist/types/src/extensions/comments.d.ts +1 -1
- package/dist/types/src/extensions/comments.d.ts.map +1 -1
- package/dist/types/src/extensions/dnd.d.ts.map +1 -1
- package/dist/types/src/extensions/factories.d.ts +11 -11
- package/dist/types/src/extensions/factories.d.ts.map +1 -1
- package/dist/types/src/extensions/focus.d.ts.map +1 -1
- package/dist/types/src/extensions/folding.d.ts.map +1 -1
- package/dist/types/src/extensions/index.d.ts +10 -1
- package/dist/types/src/extensions/index.d.ts.map +1 -1
- package/dist/types/src/extensions/json.d.ts +1 -1
- package/dist/types/src/extensions/json.d.ts.map +1 -1
- package/dist/types/src/extensions/listener.d.ts +8 -6
- package/dist/types/src/extensions/listener.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/action.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/bundle.d.ts +8 -2
- package/dist/types/src/extensions/markdown/bundle.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/changes.d.ts +1 -1
- package/dist/types/src/extensions/markdown/changes.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/decorate.d.ts +9 -1
- package/dist/types/src/extensions/markdown/decorate.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/formatting.d.ts +2 -4
- package/dist/types/src/extensions/markdown/formatting.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/formatting.test.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/highlight.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/image.d.ts.map +1 -1
- package/dist/types/src/extensions/markdown/link.d.ts.map +1 -1
- package/dist/types/src/extensions/modal.d.ts +7 -0
- package/dist/types/src/extensions/modal.d.ts.map +1 -0
- package/dist/types/src/extensions/modes.d.ts +1 -1
- package/dist/types/src/extensions/modes.d.ts.map +1 -1
- package/dist/types/src/extensions/outliner/menu.d.ts +8 -0
- package/dist/types/src/extensions/outliner/menu.d.ts.map +1 -0
- package/dist/types/src/extensions/outliner/outliner.d.ts +1 -1
- package/dist/types/src/extensions/outliner/outliner.d.ts.map +1 -1
- package/dist/types/src/extensions/outliner/selection.d.ts.map +1 -1
- package/dist/types/src/extensions/outliner/tree.d.ts +2 -2
- package/dist/types/src/extensions/outliner/tree.d.ts.map +1 -1
- package/dist/types/src/extensions/preview/preview.d.ts +8 -8
- package/dist/types/src/extensions/preview/preview.d.ts.map +1 -1
- package/dist/types/src/extensions/replacer.d.ts +21 -0
- package/dist/types/src/extensions/replacer.d.ts.map +1 -0
- package/dist/types/src/extensions/replacer.test.d.ts +2 -0
- package/dist/types/src/extensions/replacer.test.d.ts.map +1 -0
- package/dist/types/src/extensions/scrolling.d.ts +78 -0
- package/dist/types/src/extensions/scrolling.d.ts.map +1 -0
- package/dist/types/src/extensions/state.d.ts +2 -0
- package/dist/types/src/extensions/state.d.ts.map +1 -0
- package/dist/types/src/extensions/submit.d.ts +10 -0
- package/dist/types/src/extensions/submit.d.ts.map +1 -0
- package/dist/types/src/extensions/tab.d.ts +4 -0
- package/dist/types/src/extensions/tab.d.ts.map +1 -0
- package/dist/types/src/extensions/tags/extended-markdown.d.ts +10 -0
- package/dist/types/src/extensions/tags/extended-markdown.d.ts.map +1 -0
- package/dist/types/src/extensions/tags/extended-markdown.test.d.ts +2 -0
- package/dist/types/src/extensions/tags/extended-markdown.test.d.ts.map +1 -0
- package/dist/types/src/extensions/tags/index.d.ts +4 -0
- package/dist/types/src/extensions/tags/index.d.ts.map +1 -0
- package/dist/types/src/extensions/tags/streamer.d.ts +12 -0
- package/dist/types/src/extensions/tags/streamer.d.ts.map +1 -0
- package/dist/types/src/extensions/tags/xml-tags.d.ts +97 -0
- package/dist/types/src/extensions/tags/xml-tags.d.ts.map +1 -0
- package/dist/types/src/extensions/tags/xml-util.d.ts +10 -0
- package/dist/types/src/extensions/tags/xml-util.d.ts.map +1 -0
- package/dist/types/src/hooks/useTextEditor.d.ts +5 -9
- package/dist/types/src/hooks/useTextEditor.d.ts.map +1 -1
- package/dist/types/src/stories/CommandDialog.stories.d.ts +14 -0
- package/dist/types/src/stories/CommandDialog.stories.d.ts.map +1 -0
- package/dist/types/src/stories/Comments.stories.d.ts +21 -10
- package/dist/types/src/stories/Comments.stories.d.ts.map +1 -1
- package/dist/types/src/stories/EditorToolbar.stories.d.ts +39 -3
- package/dist/types/src/stories/EditorToolbar.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Experimental.stories.d.ts +22 -13
- package/dist/types/src/stories/Experimental.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Markdown.stories.d.ts +32 -43
- package/dist/types/src/stories/Markdown.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Outliner.stories.d.ts +15 -21
- package/dist/types/src/stories/Outliner.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Popover.stories.d.ts +20 -0
- package/dist/types/src/stories/Popover.stories.d.ts.map +1 -0
- package/dist/types/src/stories/Preview.stories.d.ts +22 -7
- package/dist/types/src/stories/Preview.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Tags.stories.d.ts +16 -0
- package/dist/types/src/stories/Tags.stories.d.ts.map +1 -0
- package/dist/types/src/stories/TextEditor.stories.d.ts +37 -52
- package/dist/types/src/stories/TextEditor.stories.d.ts.map +1 -1
- package/dist/types/src/stories/components/EditorStory.d.ts +8 -9
- package/dist/types/src/stories/components/EditorStory.d.ts.map +1 -1
- package/dist/types/src/stories/components/util.d.ts.map +1 -1
- package/dist/types/src/styles/theme.d.ts.map +1 -1
- package/dist/types/src/translations.d.ts +1 -1
- package/dist/types/src/types/types.d.ts +2 -2
- package/dist/types/src/types/types.d.ts.map +1 -1
- package/dist/types/src/util/cursor.d.ts.map +1 -1
- package/dist/types/src/util/debug.d.ts +5 -1
- package/dist/types/src/util/debug.d.ts.map +1 -1
- package/dist/types/src/util/decorations.d.ts +4 -0
- package/dist/types/src/util/decorations.d.ts.map +1 -0
- package/dist/types/src/util/dom.d.ts +2 -12
- package/dist/types/src/util/dom.d.ts.map +1 -1
- package/dist/types/src/util/index.d.ts +1 -0
- package/dist/types/src/util/index.d.ts.map +1 -1
- package/dist/types/src/util/react.d.ts +1 -1
- package/dist/types/src/util/react.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +74 -72
- package/src/components/Editor/Editor.stories.tsx +89 -0
- package/src/components/Editor/Editor.tsx +160 -25
- package/src/components/EditorContent/EditorContent.stories.tsx +70 -0
- package/src/components/EditorContent/EditorContent.tsx +70 -0
- package/src/components/EditorContent/controller.ts +50 -0
- package/src/components/EditorContent/index.ts +6 -0
- package/src/components/EditorMenuProvider/EditorMenuProvider.tsx +233 -0
- package/src/components/EditorMenuProvider/index.ts +11 -0
- package/src/components/EditorMenuProvider/menu-presets.ts +123 -0
- package/src/components/EditorMenuProvider/menu.ts +71 -0
- package/src/components/EditorMenuProvider/popover.ts +287 -0
- package/src/components/EditorMenuProvider/useEditorMenu.ts +175 -0
- package/src/components/EditorMenuProvider/util.ts +31 -0
- package/src/components/EditorPreviewProvider/EditorPreviewProvider.tsx +82 -0
- package/src/components/EditorPreviewProvider/index.ts +5 -0
- package/src/components/EditorToolbar/EditorToolbar.tsx +116 -96
- package/src/components/EditorToolbar/actions.ts +86 -0
- package/src/components/EditorToolbar/blocks.ts +22 -24
- package/src/components/EditorToolbar/formatting.ts +23 -25
- package/src/components/EditorToolbar/headings.ts +17 -9
- package/src/components/EditorToolbar/image.ts +9 -5
- package/src/components/EditorToolbar/index.ts +3 -7
- package/src/components/EditorToolbar/search.ts +9 -5
- package/src/components/EditorToolbar/useEditorToolbar.ts +20 -0
- package/src/components/EditorToolbar/view-mode.ts +12 -6
- package/src/components/index.ts +8 -2
- package/src/defaults.ts +5 -2
- package/src/extensions/autocomplete/autocomplete.ts +151 -0
- package/src/extensions/autocomplete/index.ts +8 -0
- package/src/extensions/autocomplete/match.ts +46 -0
- package/src/extensions/{command → autocomplete}/placeholder.ts +22 -18
- package/src/extensions/{command → autocomplete}/typeahead.ts +8 -50
- package/src/extensions/automerge/automerge.stories.tsx +32 -25
- package/src/extensions/automerge/automerge.ts +31 -11
- package/src/extensions/automerge/cursor.ts +1 -1
- package/src/extensions/automerge/defs.ts +1 -1
- package/src/extensions/automerge/sync.ts +9 -5
- package/src/extensions/automerge/update-automerge.ts +2 -2
- package/src/extensions/autoscroll.ts +163 -0
- package/src/extensions/awareness/awareness-provider.ts +2 -2
- package/src/extensions/awareness/awareness.ts +2 -2
- package/src/extensions/blocks.ts +131 -0
- package/src/extensions/bookmarks.ts +75 -0
- package/src/extensions/comments.ts +20 -14
- package/src/extensions/dnd.ts +1 -1
- package/src/extensions/factories.ts +54 -35
- package/src/extensions/focus.ts +5 -4
- package/src/extensions/folding.tsx +3 -6
- package/src/extensions/hashtag.tsx +2 -2
- package/src/extensions/index.ts +10 -1
- package/src/extensions/json.ts +1 -1
- package/src/extensions/listener.ts +14 -20
- package/src/extensions/markdown/action.ts +2 -1
- package/src/extensions/markdown/bundle.ts +40 -6
- package/src/extensions/markdown/changes.ts +1 -1
- package/src/extensions/markdown/decorate.ts +32 -22
- package/src/extensions/markdown/formatting.test.ts +6 -6
- package/src/extensions/markdown/formatting.ts +11 -13
- package/src/extensions/markdown/highlight.ts +2 -2
- package/src/extensions/markdown/image.ts +5 -6
- package/src/extensions/markdown/link.ts +3 -0
- package/src/extensions/markdown/table.ts +13 -7
- package/src/extensions/mention.ts +1 -1
- package/src/extensions/modal.ts +24 -0
- package/src/extensions/modes.ts +2 -2
- package/src/extensions/{command/floating-menu.ts → outliner/menu.ts} +16 -21
- package/src/extensions/outliner/outliner.test.ts +3 -2
- package/src/extensions/outliner/outliner.ts +8 -7
- package/src/extensions/outliner/selection.ts +1 -1
- package/src/extensions/outliner/tree.test.ts +3 -2
- package/src/extensions/outliner/tree.ts +2 -2
- package/src/extensions/preview/index.ts +1 -1
- package/src/extensions/preview/preview.ts +69 -69
- package/src/extensions/replacer.test.ts +75 -0
- package/src/extensions/replacer.ts +93 -0
- package/src/extensions/scrolling.ts +189 -0
- package/src/extensions/selection.ts +3 -3
- package/src/extensions/state.ts +7 -0
- package/src/extensions/submit.ts +62 -0
- package/src/extensions/tab.ts +29 -0
- package/src/extensions/tags/extended-markdown.test.ts +262 -0
- package/src/extensions/tags/extended-markdown.ts +78 -0
- package/src/extensions/tags/index.ts +7 -0
- package/src/extensions/tags/streamer.ts +243 -0
- package/src/extensions/tags/xml-tags.ts +500 -0
- package/src/extensions/tags/xml-util.ts +94 -0
- package/src/extensions/typewriter.ts +1 -1
- package/src/hooks/useTextEditor.ts +31 -43
- package/src/stories/CommandDialog.stories.tsx +83 -0
- package/src/stories/Comments.stories.tsx +17 -14
- package/src/stories/EditorToolbar.stories.tsx +21 -20
- package/src/stories/Experimental.stories.tsx +17 -13
- package/src/stories/Markdown.stories.tsx +26 -22
- package/src/stories/Outliner.stories.tsx +51 -39
- package/src/stories/Popover.stories.tsx +162 -0
- package/src/stories/Preview.stories.tsx +56 -51
- package/src/stories/Tags.stories.tsx +95 -0
- package/src/stories/TextEditor.stories.tsx +42 -61
- package/src/stories/components/EditorStory.tsx +36 -24
- package/src/stories/components/util.tsx +40 -8
- package/src/styles/markdown.ts +1 -1
- package/src/styles/theme.ts +16 -12
- package/src/translations.ts +1 -1
- package/src/types/types.ts +1 -1
- package/src/util/cursor.ts +2 -1
- package/src/util/debug.ts +7 -2
- package/src/util/decorations.ts +21 -0
- package/src/util/dom.ts +5 -27
- package/src/util/index.ts +1 -0
- package/src/util/react.tsx +1 -1
- package/dist/lib/browser/chunk-22UMM3QJ.mjs.map +0 -7
- package/dist/lib/browser/testing/index.mjs +0 -6
- package/dist/lib/browser/testing/index.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-YXYQPV6R.mjs.map +0 -7
- package/dist/lib/node-esm/testing/index.mjs +0 -8
- package/dist/lib/node-esm/testing/index.mjs.map +0 -7
- package/dist/types/src/components/EditorToolbar/lists.d.ts +0 -19
- package/dist/types/src/components/EditorToolbar/lists.d.ts.map +0 -1
- package/dist/types/src/components/EditorToolbar/util.d.ts +0 -51
- package/dist/types/src/components/EditorToolbar/util.d.ts.map +0 -1
- package/dist/types/src/components/Popover/CommandMenu.d.ts +0 -34
- package/dist/types/src/components/Popover/CommandMenu.d.ts.map +0 -1
- package/dist/types/src/components/Popover/RefDropdownMenu.d.ts +0 -21
- package/dist/types/src/components/Popover/RefDropdownMenu.d.ts.map +0 -1
- package/dist/types/src/components/Popover/RefPopover.d.ts +0 -34
- package/dist/types/src/components/Popover/RefPopover.d.ts.map +0 -1
- package/dist/types/src/components/Popover/index.d.ts +0 -4
- package/dist/types/src/components/Popover/index.d.ts.map +0 -1
- package/dist/types/src/extensions/autocomplete.d.ts +0 -13
- package/dist/types/src/extensions/autocomplete.d.ts.map +0 -1
- package/dist/types/src/extensions/command/action.d.ts +0 -17
- package/dist/types/src/extensions/command/action.d.ts.map +0 -1
- package/dist/types/src/extensions/command/command-menu.d.ts +0 -20
- package/dist/types/src/extensions/command/command-menu.d.ts.map +0 -1
- package/dist/types/src/extensions/command/command.d.ts +0 -6
- package/dist/types/src/extensions/command/command.d.ts.map +0 -1
- package/dist/types/src/extensions/command/floating-menu.d.ts +0 -7
- package/dist/types/src/extensions/command/floating-menu.d.ts.map +0 -1
- package/dist/types/src/extensions/command/hint.d.ts +0 -24
- package/dist/types/src/extensions/command/hint.d.ts.map +0 -1
- package/dist/types/src/extensions/command/index.d.ts +0 -7
- package/dist/types/src/extensions/command/index.d.ts.map +0 -1
- package/dist/types/src/extensions/command/placeholder.d.ts +0 -10
- package/dist/types/src/extensions/command/placeholder.d.ts.map +0 -1
- package/dist/types/src/extensions/command/state.d.ts +0 -16
- package/dist/types/src/extensions/command/state.d.ts.map +0 -1
- package/dist/types/src/extensions/command/typeahead.d.ts +0 -22
- package/dist/types/src/extensions/command/typeahead.d.ts.map +0 -1
- package/dist/types/src/extensions/command/useCommandMenu.d.ts +0 -26
- package/dist/types/src/extensions/command/useCommandMenu.d.ts.map +0 -1
- package/dist/types/src/stories/Command.stories.d.ts +0 -7
- package/dist/types/src/stories/Command.stories.d.ts.map +0 -1
- package/dist/types/src/stories/CommandMenu.stories.d.ts +0 -13
- package/dist/types/src/stories/CommandMenu.stories.d.ts.map +0 -1
- package/dist/types/src/testing/index.d.ts +0 -2
- package/dist/types/src/testing/index.d.ts.map +0 -1
- package/dist/types/src/testing/util.d.ts +0 -2
- package/dist/types/src/testing/util.d.ts.map +0 -1
- package/src/components/EditorToolbar/lists.ts +0 -59
- package/src/components/EditorToolbar/util.ts +0 -65
- package/src/components/Popover/CommandMenu.tsx +0 -279
- package/src/components/Popover/RefDropdownMenu.tsx +0 -85
- package/src/components/Popover/RefPopover.tsx +0 -99
- package/src/components/Popover/index.ts +0 -7
- package/src/extensions/autocomplete.ts +0 -69
- package/src/extensions/command/action.ts +0 -56
- package/src/extensions/command/command-menu.ts +0 -210
- package/src/extensions/command/command.ts +0 -34
- package/src/extensions/command/hint.ts +0 -102
- package/src/extensions/command/index.ts +0 -10
- package/src/extensions/command/state.ts +0 -89
- package/src/extensions/command/useCommandMenu.ts +0 -118
- package/src/stories/Command.stories.tsx +0 -97
- package/src/stories/CommandMenu.stories.tsx +0 -159
- package/src/testing/index.ts +0 -5
- package/src/testing/util.ts +0 -5
package/src/defaults.ts
CHANGED
|
@@ -28,7 +28,9 @@ export const editorSlots: ThemeExtensionsOptions['slots'] = {
|
|
|
28
28
|
|
|
29
29
|
export const editorGutter = EditorView.theme({
|
|
30
30
|
'.cm-gutters': {
|
|
31
|
-
|
|
31
|
+
// NOTE: Color required to cover content if scrolling horizontally.
|
|
32
|
+
// TODO(burdon): Non-transparent background clips the focus ring.
|
|
33
|
+
background: 'var(--dx-baseSurface) !important',
|
|
32
34
|
paddingRight: '1rem',
|
|
33
35
|
},
|
|
34
36
|
});
|
|
@@ -42,8 +44,9 @@ export const editorMonospace = EditorView.theme({
|
|
|
42
44
|
export const editorWithToolbarLayout =
|
|
43
45
|
'grid grid-cols-1 grid-rows-[min-content_1fr] data-[toolbar=disabled]:grid-rows-[1fr] justify-center content-start overflow-hidden';
|
|
44
46
|
|
|
47
|
+
// NOTE: Padding is added to the editor to account for the focus ring (since otherwise the CM gutter will clip it)
|
|
45
48
|
export const stackItemContentEditorClassNames = (role?: string) =>
|
|
46
49
|
mx(
|
|
47
|
-
'
|
|
50
|
+
'p-0.5 dx-focus-ring-inset attention-surface data-[toolbar=disabled]:pbs-2',
|
|
48
51
|
role === 'section' ? '[&_.cm-scroller]:overflow-hidden [&_.cm-scroller]:min-bs-24' : 'min-bs-0',
|
|
49
52
|
);
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { type Extension, Prec } from '@codemirror/state';
|
|
6
|
+
import {
|
|
7
|
+
Decoration,
|
|
8
|
+
type DecorationSet,
|
|
9
|
+
EditorView,
|
|
10
|
+
ViewPlugin,
|
|
11
|
+
type ViewUpdate,
|
|
12
|
+
WidgetType,
|
|
13
|
+
keymap,
|
|
14
|
+
} from '@codemirror/view';
|
|
15
|
+
|
|
16
|
+
export type AutocompleteOptions = {
|
|
17
|
+
/**
|
|
18
|
+
* Function that returns a list of suggestions based on the current text.
|
|
19
|
+
* @param text The current text before the cursor
|
|
20
|
+
* @returns Array of suggestion strings
|
|
21
|
+
*/
|
|
22
|
+
onSuggest?: (text: string) => string[];
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Creates an autocomplete extension that shows inline suggestions.
|
|
27
|
+
* Pressing Tab will complete the suggestion.
|
|
28
|
+
*
|
|
29
|
+
* @deprecated Use typeahead.
|
|
30
|
+
*/
|
|
31
|
+
export const autocomplete = ({ onSuggest }: AutocompleteOptions = {}): Extension => {
|
|
32
|
+
const suggest = ViewPlugin.fromClass(
|
|
33
|
+
class {
|
|
34
|
+
_decorations: DecorationSet;
|
|
35
|
+
_currentSuggestion: string | null = null;
|
|
36
|
+
|
|
37
|
+
constructor(view: EditorView) {
|
|
38
|
+
this._decorations = this.computeDecorations(view);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
update(update: ViewUpdate) {
|
|
42
|
+
if (update.docChanged || update.selectionSet) {
|
|
43
|
+
this._decorations = this.computeDecorations(update.view);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
private computeDecorations(view: EditorView): DecorationSet {
|
|
48
|
+
const text = view.state.doc.toString();
|
|
49
|
+
const suggestions = onSuggest?.(text) ?? [];
|
|
50
|
+
if (!suggestions.length) {
|
|
51
|
+
this._currentSuggestion = null;
|
|
52
|
+
return Decoration.none;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// Get the first suggestion.
|
|
56
|
+
this._currentSuggestion = suggestions[0];
|
|
57
|
+
const suffix = this._currentSuggestion.slice(text.length);
|
|
58
|
+
if (!suffix) {
|
|
59
|
+
return Decoration.none;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// Always show ghost text at the end of the document.
|
|
63
|
+
return Decoration.set([
|
|
64
|
+
Decoration.widget({
|
|
65
|
+
widget: new InlineSuggestionWidget(suffix),
|
|
66
|
+
side: 1,
|
|
67
|
+
}).range(view.state.doc.length),
|
|
68
|
+
]);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
completeSuggestion(view: EditorView): boolean {
|
|
72
|
+
if (!this._currentSuggestion) {
|
|
73
|
+
return false;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
const text = view.state.doc.toString();
|
|
77
|
+
const suffix = this._currentSuggestion.slice(text.length);
|
|
78
|
+
if (!suffix) {
|
|
79
|
+
return false;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
view.dispatch({
|
|
83
|
+
changes: {
|
|
84
|
+
from: view.state.doc.length,
|
|
85
|
+
insert: suffix,
|
|
86
|
+
},
|
|
87
|
+
selection: {
|
|
88
|
+
anchor: view.state.doc.length + suffix.length,
|
|
89
|
+
},
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
return true;
|
|
93
|
+
}
|
|
94
|
+
},
|
|
95
|
+
{
|
|
96
|
+
decorations: (v) => v._decorations,
|
|
97
|
+
},
|
|
98
|
+
);
|
|
99
|
+
|
|
100
|
+
return [
|
|
101
|
+
suggest,
|
|
102
|
+
EditorView.theme({
|
|
103
|
+
'.cm-inline-suggestion': {
|
|
104
|
+
opacity: 0.4,
|
|
105
|
+
},
|
|
106
|
+
}),
|
|
107
|
+
|
|
108
|
+
Prec.highest(
|
|
109
|
+
keymap.of([
|
|
110
|
+
{
|
|
111
|
+
key: 'Tab',
|
|
112
|
+
preventDefault: true,
|
|
113
|
+
run: (view) => {
|
|
114
|
+
const plugin = view.plugin(suggest);
|
|
115
|
+
return plugin?.completeSuggestion(view) ?? false;
|
|
116
|
+
},
|
|
117
|
+
},
|
|
118
|
+
{
|
|
119
|
+
key: 'ArrowRight',
|
|
120
|
+
preventDefault: true,
|
|
121
|
+
run: (view) => {
|
|
122
|
+
// Only complete if cursor is at the end
|
|
123
|
+
if (view.state.selection.main.head !== view.state.doc.length) {
|
|
124
|
+
return false;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
const plugin = view.plugin(suggest);
|
|
128
|
+
return plugin?.completeSuggestion(view) ?? false;
|
|
129
|
+
},
|
|
130
|
+
},
|
|
131
|
+
]),
|
|
132
|
+
),
|
|
133
|
+
];
|
|
134
|
+
};
|
|
135
|
+
|
|
136
|
+
class InlineSuggestionWidget extends WidgetType {
|
|
137
|
+
constructor(private suffix: string) {
|
|
138
|
+
super();
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
override eq(other: this) {
|
|
142
|
+
return this.suffix === other.suffix;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
override toDOM() {
|
|
146
|
+
const span = document.createElement('span');
|
|
147
|
+
span.textContent = this.suffix;
|
|
148
|
+
span.className = 'cm-inline-suggestion';
|
|
149
|
+
return span;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
export type CompoetionContext = { line: string };
|
|
6
|
+
|
|
7
|
+
export type CompletionOptions = {
|
|
8
|
+
default?: string;
|
|
9
|
+
minLength?: number;
|
|
10
|
+
};
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Util to match current line to a static list of completions.
|
|
14
|
+
*/
|
|
15
|
+
export const staticCompletion =
|
|
16
|
+
(completions: string[], options: CompletionOptions = {}) =>
|
|
17
|
+
({ line }: CompoetionContext) => {
|
|
18
|
+
if (line.length === 0 && options.default) {
|
|
19
|
+
return options.default;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const parts = line.split(/\s+/).filter(Boolean);
|
|
23
|
+
if (parts.length) {
|
|
24
|
+
const str = parts.at(-1)!;
|
|
25
|
+
if (str.length >= (options.minLength ?? 0)) {
|
|
26
|
+
for (const completion of completions) {
|
|
27
|
+
const match = matchCompletion(completion, str);
|
|
28
|
+
if (match) {
|
|
29
|
+
return match;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
|
|
36
|
+
export const matchCompletion = (completion: string, str: string, minLength = 0): string | undefined => {
|
|
37
|
+
if (
|
|
38
|
+
str.length >= minLength &&
|
|
39
|
+
completion.length > str.length &&
|
|
40
|
+
completion.startsWith(str)
|
|
41
|
+
// TODO(burdon): If case insensitive, need to replace existing chars.
|
|
42
|
+
// completion.toLowerCase().startsWith(str.toLowerCase())
|
|
43
|
+
) {
|
|
44
|
+
return completion.slice(str.length);
|
|
45
|
+
}
|
|
46
|
+
};
|
|
@@ -4,27 +4,30 @@
|
|
|
4
4
|
//
|
|
5
5
|
|
|
6
6
|
import { type Extension } from '@codemirror/state';
|
|
7
|
-
import { Decoration, EditorView,
|
|
7
|
+
import { Decoration, EditorView, ViewPlugin, type ViewUpdate, WidgetType } from '@codemirror/view';
|
|
8
8
|
|
|
9
9
|
import { clientRectsFor, flattenRect } from '../../util';
|
|
10
10
|
|
|
11
11
|
type Content = string | HTMLElement | ((view: EditorView) => HTMLElement);
|
|
12
12
|
|
|
13
13
|
export type PlaceholderOptions = {
|
|
14
|
-
delay?: number;
|
|
15
14
|
content: Content;
|
|
15
|
+
delay?: number;
|
|
16
16
|
};
|
|
17
17
|
|
|
18
|
-
|
|
18
|
+
/**
|
|
19
|
+
* Shows a transient placeholder at the current cursor position.
|
|
20
|
+
*/
|
|
21
|
+
export const placeholder = ({ content, delay = 3_000 }: PlaceholderOptions): Extension => {
|
|
19
22
|
const plugin = ViewPlugin.fromClass(
|
|
20
23
|
class {
|
|
21
|
-
|
|
22
|
-
|
|
24
|
+
_timeout: ReturnType<typeof setTimeout> | undefined;
|
|
25
|
+
_decorations = Decoration.none;
|
|
23
26
|
|
|
24
27
|
update(update: ViewUpdate) {
|
|
25
|
-
if (this.
|
|
26
|
-
window.clearTimeout(this.
|
|
27
|
-
this.
|
|
28
|
+
if (this._timeout) {
|
|
29
|
+
window.clearTimeout(this._timeout);
|
|
30
|
+
this._timeout = undefined;
|
|
28
31
|
}
|
|
29
32
|
|
|
30
33
|
// Check if the active line (where cursor is) is empty.
|
|
@@ -33,10 +36,10 @@ export const placeholder = ({ delay = 3_000, content }: PlaceholderOptions): Ext
|
|
|
33
36
|
if (isEmpty) {
|
|
34
37
|
// Create widget decoration at the start of the current line.
|
|
35
38
|
const lineStart = activeLine.from;
|
|
36
|
-
this.
|
|
37
|
-
this.
|
|
39
|
+
this._timeout = setTimeout(() => {
|
|
40
|
+
this._decorations = Decoration.set([
|
|
38
41
|
Decoration.widget({
|
|
39
|
-
widget: new
|
|
42
|
+
widget: new PlaceholderWidget(content),
|
|
40
43
|
side: 1,
|
|
41
44
|
}).range(lineStart),
|
|
42
45
|
]);
|
|
@@ -45,18 +48,18 @@ export const placeholder = ({ delay = 3_000, content }: PlaceholderOptions): Ext
|
|
|
45
48
|
}, delay);
|
|
46
49
|
}
|
|
47
50
|
|
|
48
|
-
this.
|
|
51
|
+
this._decorations = Decoration.none;
|
|
49
52
|
}
|
|
50
53
|
|
|
51
54
|
destroy() {
|
|
52
|
-
if (this.
|
|
53
|
-
clearTimeout(this.
|
|
55
|
+
if (this._timeout) {
|
|
56
|
+
clearTimeout(this._timeout);
|
|
54
57
|
}
|
|
55
58
|
}
|
|
56
59
|
},
|
|
57
60
|
{
|
|
58
61
|
provide: (plugin) => {
|
|
59
|
-
return [EditorView.decorations.of((view) => view.plugin(plugin)?.
|
|
62
|
+
return [EditorView.decorations.of((view) => view.plugin(plugin)?._decorations ?? Decoration.none)];
|
|
60
63
|
},
|
|
61
64
|
},
|
|
62
65
|
);
|
|
@@ -66,7 +69,7 @@ export const placeholder = ({ delay = 3_000, content }: PlaceholderOptions): Ext
|
|
|
66
69
|
: plugin;
|
|
67
70
|
};
|
|
68
71
|
|
|
69
|
-
class
|
|
72
|
+
export class PlaceholderWidget extends WidgetType {
|
|
70
73
|
constructor(readonly content: Content) {
|
|
71
74
|
super();
|
|
72
75
|
}
|
|
@@ -75,6 +78,7 @@ class Placeholder extends WidgetType {
|
|
|
75
78
|
const wrap = document.createElement('span');
|
|
76
79
|
wrap.className = 'cm-placeholder';
|
|
77
80
|
wrap.style.pointerEvents = 'none';
|
|
81
|
+
wrap.setAttribute('aria-hidden', 'true');
|
|
78
82
|
wrap.appendChild(
|
|
79
83
|
typeof this.content === 'string'
|
|
80
84
|
? document.createTextNode(this.content)
|
|
@@ -82,7 +86,7 @@ class Placeholder extends WidgetType {
|
|
|
82
86
|
? this.content(view)
|
|
83
87
|
: this.content.cloneNode(true),
|
|
84
88
|
);
|
|
85
|
-
|
|
89
|
+
|
|
86
90
|
return wrap;
|
|
87
91
|
}
|
|
88
92
|
|
|
@@ -92,7 +96,7 @@ class Placeholder extends WidgetType {
|
|
|
92
96
|
return null;
|
|
93
97
|
}
|
|
94
98
|
|
|
95
|
-
const style =
|
|
99
|
+
const style = getComputedStyle(dom.parentNode as HTMLElement);
|
|
96
100
|
const rect = flattenRect(rects[0], style.direction !== 'rtl');
|
|
97
101
|
const lineHeight = parseInt(style.lineHeight);
|
|
98
102
|
if (rect.bottom - rect.top > lineHeight * 1.5) {
|
|
@@ -2,28 +2,27 @@
|
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import { EditorSelection, Prec, RangeSetBuilder
|
|
5
|
+
import { EditorSelection, type Extension, Prec, RangeSetBuilder } from '@codemirror/state';
|
|
6
6
|
import {
|
|
7
7
|
type Command,
|
|
8
8
|
Decoration,
|
|
9
9
|
type DecorationSet,
|
|
10
10
|
type EditorView,
|
|
11
|
-
keymap,
|
|
12
11
|
ViewPlugin,
|
|
13
12
|
type ViewUpdate,
|
|
13
|
+
keymap,
|
|
14
14
|
} from '@codemirror/view';
|
|
15
15
|
|
|
16
|
-
import {
|
|
16
|
+
import { type CompoetionContext } from './match';
|
|
17
|
+
import { PlaceholderWidget } from './placeholder';
|
|
17
18
|
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
// TODO(burdon): Option to complete only at end of line?
|
|
19
|
+
// TODO(burdon): Option to complete only at end of line.
|
|
21
20
|
export type TypeaheadOptions = {
|
|
22
|
-
onComplete?: (context:
|
|
21
|
+
onComplete?: (context: CompoetionContext) => string | undefined;
|
|
23
22
|
};
|
|
24
23
|
|
|
25
24
|
/**
|
|
26
|
-
*
|
|
25
|
+
* Shows a completion placeholder.
|
|
27
26
|
*/
|
|
28
27
|
export const typeahead = ({ onComplete }: TypeaheadOptions = {}): Extension => {
|
|
29
28
|
let hint: string | undefined;
|
|
@@ -57,7 +56,7 @@ export const typeahead = ({ onComplete }: TypeaheadOptions = {}): Extension => {
|
|
|
57
56
|
const str = update.state.sliceDoc(line.from, selection.from);
|
|
58
57
|
hint = onComplete?.({ line: str });
|
|
59
58
|
if (hint) {
|
|
60
|
-
builder.add(selection.from, selection.to, Decoration.widget({ widget: new
|
|
59
|
+
builder.add(selection.from, selection.to, Decoration.widget({ widget: new PlaceholderWidget(hint) }));
|
|
61
60
|
}
|
|
62
61
|
}
|
|
63
62
|
|
|
@@ -86,44 +85,3 @@ export const typeahead = ({ onComplete }: TypeaheadOptions = {}): Extension => {
|
|
|
86
85
|
),
|
|
87
86
|
];
|
|
88
87
|
};
|
|
89
|
-
|
|
90
|
-
type CompletionOptions = {
|
|
91
|
-
default?: string;
|
|
92
|
-
minLength?: number;
|
|
93
|
-
};
|
|
94
|
-
|
|
95
|
-
/**
|
|
96
|
-
* Util to match current line to a static list of completions.
|
|
97
|
-
*/
|
|
98
|
-
export const staticCompletion =
|
|
99
|
-
(completions: string[], options: CompletionOptions = {}) =>
|
|
100
|
-
({ line }: TypeaheadContext) => {
|
|
101
|
-
if (line.length === 0 && options.default) {
|
|
102
|
-
return options.default;
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
const parts = line.split(/\s+/).filter(Boolean);
|
|
106
|
-
if (parts.length) {
|
|
107
|
-
const str = parts.at(-1)!;
|
|
108
|
-
if (str.length >= (options.minLength ?? 0)) {
|
|
109
|
-
for (const completion of completions) {
|
|
110
|
-
const match = matchCompletion(completion, str);
|
|
111
|
-
if (match) {
|
|
112
|
-
return match;
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
}
|
|
117
|
-
};
|
|
118
|
-
|
|
119
|
-
export const matchCompletion = (completion: string, str: string, minLength = 0): string | undefined => {
|
|
120
|
-
if (
|
|
121
|
-
str.length >= minLength &&
|
|
122
|
-
completion.length > str.length &&
|
|
123
|
-
completion.startsWith(str)
|
|
124
|
-
// TODO(burdon): If case insensitive, need to replace existing chars.
|
|
125
|
-
// completion.toLowerCase().startsWith(str.toLowerCase())
|
|
126
|
-
) {
|
|
127
|
-
return completion.slice(str.length);
|
|
128
|
-
}
|
|
129
|
-
};
|
|
@@ -2,20 +2,20 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import '@dxos-theme';
|
|
6
|
-
|
|
7
5
|
import '@preact/signals-react';
|
|
8
6
|
|
|
9
7
|
import { Repo } from '@automerge/automerge-repo';
|
|
10
8
|
import { BroadcastChannelNetworkAdapter } from '@automerge/automerge-repo-network-broadcastchannel';
|
|
9
|
+
import { type Meta, type StoryObj } from '@storybook/react-vite';
|
|
11
10
|
import React, { useEffect, useState } from 'react';
|
|
12
11
|
|
|
13
12
|
import { Obj, Ref, Type } from '@dxos/echo';
|
|
14
|
-
import { DocAccessor,
|
|
15
|
-
import {
|
|
16
|
-
import {
|
|
13
|
+
import { DocAccessor, Query, type Space, createDocAccessor, useQuery, useSpace } from '@dxos/react-client/echo';
|
|
14
|
+
import { type Identity, useIdentity } from '@dxos/react-client/halo';
|
|
15
|
+
import { type ClientRepeatedComponentProps, ClientRepeater } from '@dxos/react-client/testing';
|
|
17
16
|
import { useThemeContext } from '@dxos/react-ui';
|
|
18
|
-
import {
|
|
17
|
+
import { withTheme } from '@dxos/react-ui/testing';
|
|
18
|
+
import { render } from '@dxos/storybook-utils';
|
|
19
19
|
|
|
20
20
|
import { editorSlots } from '../../defaults';
|
|
21
21
|
import { useTextEditor } from '../../hooks';
|
|
@@ -41,7 +41,7 @@ const Editor = ({ source, autoFocus, space, identity }: EditorProps) => {
|
|
|
41
41
|
() => ({
|
|
42
42
|
initialValue: DocAccessor.getValue(source),
|
|
43
43
|
extensions: [
|
|
44
|
-
createBasicExtensions({ placeholder: 'Type here...' }),
|
|
44
|
+
createBasicExtensions({ placeholder: 'Type here...', search: true }),
|
|
45
45
|
createThemeExtensions({ themeMode, slots: editorSlots }),
|
|
46
46
|
createDataExtensions({ id: 'test', text: source, space, identity }),
|
|
47
47
|
],
|
|
@@ -50,10 +50,10 @@ const Editor = ({ source, autoFocus, space, identity }: EditorProps) => {
|
|
|
50
50
|
[source, themeMode],
|
|
51
51
|
);
|
|
52
52
|
|
|
53
|
-
return <div ref={parentRef} className='flex
|
|
53
|
+
return <div ref={parentRef} className='flex is-full' />;
|
|
54
54
|
};
|
|
55
55
|
|
|
56
|
-
const
|
|
56
|
+
const DefaultStory = () => {
|
|
57
57
|
const [object1, setObject1] = useState<DocAccessor<TestObject>>();
|
|
58
58
|
const [object2, setObject2] = useState<DocAccessor<TestObject>>();
|
|
59
59
|
|
|
@@ -88,16 +88,6 @@ const Story = () => {
|
|
|
88
88
|
);
|
|
89
89
|
};
|
|
90
90
|
|
|
91
|
-
export default {
|
|
92
|
-
title: 'ui/react-ui-editor/Automerge',
|
|
93
|
-
component: Editor,
|
|
94
|
-
decorators: [withTheme, withLayout({ fullscreen: true })],
|
|
95
|
-
render: () => <Story />,
|
|
96
|
-
parameters: {
|
|
97
|
-
translations,
|
|
98
|
-
},
|
|
99
|
-
};
|
|
100
|
-
|
|
101
91
|
const EchoStory = ({ spaceKey }: ClientRepeatedComponentProps) => {
|
|
102
92
|
const identity = useIdentity();
|
|
103
93
|
const space = useSpace(spaceKey);
|
|
@@ -105,8 +95,9 @@ const EchoStory = ({ spaceKey }: ClientRepeatedComponentProps) => {
|
|
|
105
95
|
const objects = useQuery(space, Query.type(Type.Expando, { type: 'test' }));
|
|
106
96
|
|
|
107
97
|
useEffect(() => {
|
|
108
|
-
|
|
109
|
-
|
|
98
|
+
const content = objects[0]?.content.target;
|
|
99
|
+
if (!source && content) {
|
|
100
|
+
const source = createDocAccessor(content, ['content']);
|
|
110
101
|
setSource(source);
|
|
111
102
|
}
|
|
112
103
|
}, [objects, source]);
|
|
@@ -118,17 +109,33 @@ const EchoStory = ({ spaceKey }: ClientRepeatedComponentProps) => {
|
|
|
118
109
|
return <Editor source={source} space={space} identity={identity ?? undefined} />;
|
|
119
110
|
};
|
|
120
111
|
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
112
|
+
const meta = {
|
|
113
|
+
title: 'ui/react-ui-editor/Automerge',
|
|
114
|
+
component: Editor as any,
|
|
115
|
+
render: render(DefaultStory),
|
|
124
116
|
decorators: [withTheme],
|
|
117
|
+
parameters: {
|
|
118
|
+
layout: 'fullscreen',
|
|
119
|
+
translations,
|
|
120
|
+
},
|
|
121
|
+
} satisfies Meta<typeof DefaultStory>;
|
|
122
|
+
|
|
123
|
+
export default meta;
|
|
124
|
+
|
|
125
|
+
type Story = StoryObj<typeof meta>;
|
|
126
|
+
|
|
127
|
+
export const Default: Story = {
|
|
128
|
+
args: {},
|
|
129
|
+
};
|
|
130
|
+
|
|
131
|
+
export const WithEcho: Story = {
|
|
125
132
|
render: () => {
|
|
126
133
|
return (
|
|
127
134
|
<ClientRepeater
|
|
128
135
|
count={2}
|
|
129
136
|
component={EchoStory}
|
|
130
137
|
createSpace
|
|
131
|
-
|
|
138
|
+
onCreateSpace={async ({ space }) => {
|
|
132
139
|
space.db.add(
|
|
133
140
|
Obj.make(Type.Expando, {
|
|
134
141
|
type: 'test',
|
|
@@ -5,23 +5,27 @@
|
|
|
5
5
|
//
|
|
6
6
|
|
|
7
7
|
import { next as A } from '@automerge/automerge';
|
|
8
|
-
import { StateField,
|
|
8
|
+
import { type Extension, StateField, Transaction } from '@codemirror/state';
|
|
9
9
|
import { EditorView, ViewPlugin } from '@codemirror/view';
|
|
10
10
|
|
|
11
|
-
import {
|
|
11
|
+
import { DocAccessor } from '@dxos/client/echo';
|
|
12
|
+
|
|
13
|
+
import { Cursor } from '../../util';
|
|
14
|
+
import { initialSync } from '../state';
|
|
12
15
|
|
|
13
16
|
import { cursorConverter } from './cursor';
|
|
14
|
-
import {
|
|
17
|
+
import { type State, isReconcile, updateHeadsEffect } from './defs';
|
|
15
18
|
import { Syncer } from './sync';
|
|
16
|
-
import { Cursor } from '../../util';
|
|
17
19
|
|
|
18
20
|
export const automerge = (accessor: DocAccessor): Extension => {
|
|
19
21
|
const syncState = StateField.define<State>({
|
|
20
|
-
create: () =>
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
22
|
+
create: () => {
|
|
23
|
+
return {
|
|
24
|
+
path: accessor.path.slice(),
|
|
25
|
+
lastHeads: A.getHeads(accessor.handle.doc()!),
|
|
26
|
+
unreconciledTransactions: [],
|
|
27
|
+
};
|
|
28
|
+
},
|
|
25
29
|
|
|
26
30
|
update: (value, tr) => {
|
|
27
31
|
const result: State = {
|
|
@@ -63,6 +67,18 @@ export const automerge = (accessor: DocAccessor): Extension => {
|
|
|
63
67
|
class {
|
|
64
68
|
constructor(private readonly _view: EditorView) {
|
|
65
69
|
accessor.handle.addListener('change', this._handleChange);
|
|
70
|
+
|
|
71
|
+
requestAnimationFrame(() => {
|
|
72
|
+
const value = DocAccessor.getValue<string>(accessor);
|
|
73
|
+
const current = this._view.state.doc.toString();
|
|
74
|
+
if (value !== current) {
|
|
75
|
+
// TODO(burdon): This attempts to set the initial state, but creates problems.
|
|
76
|
+
// this._view.dispatch({
|
|
77
|
+
// changes: { from: 0, to: this._view.state.doc.length, insert: value },
|
|
78
|
+
// annotations: initialSync,
|
|
79
|
+
// });
|
|
80
|
+
}
|
|
81
|
+
});
|
|
66
82
|
}
|
|
67
83
|
|
|
68
84
|
destroy() {
|
|
@@ -76,9 +92,13 @@ export const automerge = (accessor: DocAccessor): Extension => {
|
|
|
76
92
|
),
|
|
77
93
|
|
|
78
94
|
// Reconcile local updates.
|
|
79
|
-
EditorView.updateListener.of(({ view, changes }) => {
|
|
95
|
+
EditorView.updateListener.of(({ view, changes, transactions }) => {
|
|
80
96
|
if (!changes.empty) {
|
|
81
|
-
|
|
97
|
+
// Only reconcile if it's not an initial sync (to avoid loops)
|
|
98
|
+
const isInitialSync = transactions.some((tr) => tr.annotation(Transaction.userEvent) === initialSync.value);
|
|
99
|
+
if (!isInitialSync) {
|
|
100
|
+
syncer.reconcile(view, true);
|
|
101
|
+
}
|
|
82
102
|
}
|
|
83
103
|
}),
|
|
84
104
|
];
|
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
// Copyright 2024 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
+
import { type DocAccessor, fromCursor, toCursor } from '@dxos/client/echo';
|
|
5
6
|
import { log } from '@dxos/log';
|
|
6
|
-
import { type DocAccessor, fromCursor, toCursor } from '@dxos/react-client/echo';
|
|
7
7
|
|
|
8
8
|
import { type CursorConverter } from '../../util';
|
|
9
9
|
|
|
@@ -5,7 +5,7 @@
|
|
|
5
5
|
//
|
|
6
6
|
|
|
7
7
|
import { type Heads, type Prop } from '@automerge/automerge';
|
|
8
|
-
import { Annotation, StateEffect, type StateField, type
|
|
8
|
+
import { Annotation, type EditorState, StateEffect, type StateField, type Transaction } from '@codemirror/state';
|
|
9
9
|
|
|
10
10
|
export type State = {
|
|
11
11
|
path: Prop[];
|
|
@@ -8,9 +8,10 @@ import { next as A } from '@automerge/automerge';
|
|
|
8
8
|
import { type StateField } from '@codemirror/state';
|
|
9
9
|
import { type EditorView } from '@codemirror/view';
|
|
10
10
|
|
|
11
|
-
import { type IDocHandle } from '@dxos/
|
|
11
|
+
import { type IDocHandle } from '@dxos/client/echo';
|
|
12
|
+
import { log } from '@dxos/log';
|
|
12
13
|
|
|
13
|
-
import { getLastHeads, getPath, isReconcile, reconcileAnnotation,
|
|
14
|
+
import { type State, getLastHeads, getPath, isReconcile, reconcileAnnotation, updateHeads } from './defs';
|
|
14
15
|
import { updateAutomerge } from './update-automerge';
|
|
15
16
|
import { updateCodeMirror } from './update-codemirror';
|
|
16
17
|
|
|
@@ -27,7 +28,6 @@ export class Syncer {
|
|
|
27
28
|
) {}
|
|
28
29
|
|
|
29
30
|
reconcile(view: EditorView, editor: boolean): void {
|
|
30
|
-
// TODO(burdon): Better way to do mutex?
|
|
31
31
|
if (this._pending) {
|
|
32
32
|
return;
|
|
33
33
|
}
|
|
@@ -41,7 +41,9 @@ export class Syncer {
|
|
|
41
41
|
this._pending = false;
|
|
42
42
|
}
|
|
43
43
|
|
|
44
|
-
onEditorChange(view: EditorView): void {
|
|
44
|
+
private onEditorChange(view: EditorView): void {
|
|
45
|
+
log('onEditorChange');
|
|
46
|
+
|
|
45
47
|
// Apply the unreconciled transactions to the document.
|
|
46
48
|
const transactions = view.state.field(this._state).unreconciledTransactions.filter((tx) => !isReconcile(tx));
|
|
47
49
|
const newHeads = updateAutomerge(this._state, this._handle, transactions, view.state);
|
|
@@ -54,7 +56,9 @@ export class Syncer {
|
|
|
54
56
|
}
|
|
55
57
|
}
|
|
56
58
|
|
|
57
|
-
onAutomergeChange(view: EditorView): void {
|
|
59
|
+
private onAutomergeChange(view: EditorView): void {
|
|
60
|
+
log('onAutomergeChange');
|
|
61
|
+
|
|
58
62
|
// Get the diff between the updated state of the document and the heads and apply that to the codemirror doc.
|
|
59
63
|
const oldHeads = getLastHeads(view.state, this._state);
|
|
60
64
|
const newHeads = A.getHeads(this._handle.doc()!);
|