@dxos/react-ui-editor 0.8.4-main.b97322e → 0.8.4-main.dedc0f3
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/index.mjs +2074 -888
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/testing/index.mjs.map +2 -2
- package/dist/lib/node-esm/index.mjs +2074 -888
- package/dist/lib/node-esm/index.mjs.map +4 -4
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/lib/node-esm/testing/index.mjs.map +2 -2
- package/dist/types/src/components/Editor/Editor.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/blocks.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/formatting.d.ts.map +1 -1
- 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/lists.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/search.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/util.d.ts +2 -2
- package/dist/types/src/components/EditorToolbar/util.d.ts.map +1 -1
- package/dist/types/src/components/EditorToolbar/view-mode.d.ts +1 -1
- package/dist/types/src/components/EditorToolbar/view-mode.d.ts.map +1 -1
- package/dist/types/src/components/Popover/CommandMenu.d.ts.map +1 -1
- package/dist/types/src/components/Popover/RefDropdownMenu.d.ts +2 -9
- package/dist/types/src/components/Popover/RefDropdownMenu.d.ts.map +1 -1
- package/dist/types/src/components/Popover/RefPopover.d.ts +20 -17
- package/dist/types/src/components/Popover/RefPopover.d.ts.map +1 -1
- package/dist/types/src/defaults.d.ts.map +1 -1
- package/dist/types/src/extensions/autocomplete.d.ts +20 -7
- package/dist/types/src/extensions/autocomplete.d.ts.map +1 -1
- package/dist/types/src/extensions/automerge/automerge.d.ts.map +1 -1
- package/dist/types/src/extensions/automerge/automerge.stories.d.ts +9 -18
- package/dist/types/src/extensions/automerge/automerge.stories.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.map +1 -1
- package/dist/types/src/extensions/automerge/update-automerge.d.ts.map +1 -1
- package/dist/types/src/extensions/autoscroll.d.ts +10 -0
- package/dist/types/src/extensions/autoscroll.d.ts.map +1 -0
- package/dist/types/src/extensions/command/action.d.ts +1 -1
- package/dist/types/src/extensions/command/action.d.ts.map +1 -1
- package/dist/types/src/extensions/command/command-menu.d.ts +1 -1
- package/dist/types/src/extensions/command/command-menu.d.ts.map +1 -1
- package/dist/types/src/extensions/command/command.d.ts.map +1 -1
- package/dist/types/src/extensions/command/hint.d.ts +2 -7
- package/dist/types/src/extensions/command/hint.d.ts.map +1 -1
- package/dist/types/src/extensions/command/state.d.ts +1 -1
- package/dist/types/src/extensions/command/state.d.ts.map +1 -1
- package/dist/types/src/extensions/command/typeahead.d.ts.map +1 -1
- package/dist/types/src/extensions/command/useCommandMenu.d.ts +4 -4
- package/dist/types/src/extensions/command/useCommandMenu.d.ts.map +1 -1
- 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 +2 -7
- package/dist/types/src/extensions/factories.d.ts.map +1 -1
- package/dist/types/src/extensions/index.d.ts +2 -0
- package/dist/types/src/extensions/index.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 +1 -1
- 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/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 +3 -5
- package/dist/types/src/extensions/preview/preview.d.ts.map +1 -1
- 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 +71 -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.map +1 -1
- package/dist/types/src/stories/Command.stories.d.ts +12 -4
- package/dist/types/src/stories/Command.stories.d.ts.map +1 -1
- package/dist/types/src/stories/CommandMenu.stories.d.ts +10 -3
- package/dist/types/src/stories/CommandMenu.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Comments.stories.d.ts +21 -9
- package/dist/types/src/stories/Comments.stories.d.ts.map +1 -1
- package/dist/types/src/stories/EditorToolbar.stories.d.ts +39 -2
- package/dist/types/src/stories/EditorToolbar.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Experimental.stories.d.ts +22 -12
- package/dist/types/src/stories/Experimental.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Markdown.stories.d.ts +32 -42
- package/dist/types/src/stories/Markdown.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Outliner.stories.d.ts +15 -20
- package/dist/types/src/stories/Outliner.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Preview.stories.d.ts +21 -6
- package/dist/types/src/stories/Preview.stories.d.ts.map +1 -1
- package/dist/types/src/stories/Tags.stories.d.ts +17 -0
- package/dist/types/src/stories/Tags.stories.d.ts.map +1 -0
- package/dist/types/src/stories/TextEditor.stories.d.ts +38 -51
- package/dist/types/src/stories/TextEditor.stories.d.ts.map +1 -1
- package/dist/types/src/stories/components/EditorStory.d.ts +3 -6
- package/dist/types/src/stories/components/EditorStory.d.ts.map +1 -1
- package/dist/types/src/styles/theme.d.ts.map +1 -1
- package/dist/types/src/testing/util.d.ts +1 -0
- package/dist/types/src/testing/util.d.ts.map +1 -1
- package/dist/types/src/translations.d.ts +1 -1
- package/dist/types/src/types/types.d.ts +1 -1
- package/dist/types/src/util/cursor.d.ts.map +1 -1
- package/dist/types/src/util/debug.d.ts +1 -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/domino.d.ts +18 -0
- package/dist/types/src/util/domino.d.ts.map +1 -0
- package/dist/types/src/util/index.d.ts +2 -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 +57 -51
- package/src/components/Editor/Editor.tsx +1 -1
- package/src/components/EditorToolbar/EditorToolbar.tsx +40 -30
- package/src/components/EditorToolbar/blocks.ts +21 -24
- package/src/components/EditorToolbar/formatting.ts +22 -25
- package/src/components/EditorToolbar/headings.ts +10 -5
- package/src/components/EditorToolbar/image.ts +8 -4
- package/src/components/EditorToolbar/lists.ts +16 -19
- package/src/components/EditorToolbar/search.ts +8 -4
- package/src/components/EditorToolbar/util.ts +16 -5
- package/src/components/EditorToolbar/view-mode.ts +11 -6
- package/src/components/Popover/CommandMenu.tsx +3 -3
- package/src/components/Popover/RefDropdownMenu.tsx +20 -16
- package/src/components/Popover/RefPopover.tsx +63 -45
- package/src/defaults.ts +5 -2
- package/src/extensions/autocomplete.ts +204 -54
- package/src/extensions/automerge/automerge.stories.tsx +25 -18
- package/src/extensions/automerge/automerge.ts +4 -3
- package/src/extensions/automerge/defs.ts +1 -1
- package/src/extensions/automerge/sync.ts +1 -1
- package/src/extensions/automerge/update-automerge.ts +1 -1
- package/src/extensions/autoscroll.ts +151 -0
- package/src/extensions/awareness/awareness.ts +2 -2
- package/src/extensions/command/action.ts +1 -1
- package/src/extensions/command/command-menu.ts +6 -5
- package/src/extensions/command/command.ts +3 -3
- package/src/extensions/command/floating-menu.ts +1 -1
- package/src/extensions/command/hint.ts +2 -1
- package/src/extensions/command/placeholder.ts +1 -1
- package/src/extensions/command/state.ts +4 -3
- package/src/extensions/command/typeahead.ts +2 -2
- package/src/extensions/command/useCommandMenu.ts +5 -4
- package/src/extensions/comments.ts +18 -13
- package/src/extensions/dnd.ts +1 -1
- package/src/extensions/factories.ts +9 -21
- package/src/extensions/folding.tsx +2 -2
- package/src/extensions/index.ts +2 -0
- package/src/extensions/markdown/action.ts +2 -1
- package/src/extensions/markdown/bundle.ts +25 -3
- package/src/extensions/markdown/changes.ts +1 -1
- package/src/extensions/markdown/decorate.ts +23 -14
- package/src/extensions/markdown/formatting.test.ts +6 -6
- package/src/extensions/markdown/formatting.ts +3 -3
- package/src/extensions/markdown/highlight.ts +1 -1
- package/src/extensions/markdown/image.ts +3 -4
- package/src/extensions/markdown/table.ts +7 -1
- package/src/extensions/mention.ts +1 -1
- package/src/extensions/outliner/outliner.test.ts +3 -2
- package/src/extensions/outliner/outliner.ts +5 -4
- package/src/extensions/outliner/selection.ts +1 -1
- package/src/extensions/outliner/tree.test.ts +2 -1
- package/src/extensions/outliner/tree.ts +2 -2
- package/src/extensions/preview/preview.ts +59 -59
- package/src/extensions/tags/extended-markdown.test.ts +261 -0
- package/src/extensions/tags/extended-markdown.ts +78 -0
- package/src/extensions/tags/index.ts +7 -0
- package/src/extensions/tags/streamer.ts +244 -0
- package/src/extensions/tags/xml-tags.ts +335 -0
- package/src/extensions/tags/xml-util.ts +94 -0
- package/src/hooks/useTextEditor.ts +3 -3
- package/src/stories/Command.stories.tsx +24 -31
- package/src/stories/CommandMenu.stories.tsx +23 -22
- package/src/stories/Comments.stories.tsx +10 -6
- package/src/stories/EditorToolbar.stories.tsx +8 -8
- package/src/stories/Experimental.stories.tsx +12 -8
- package/src/stories/Markdown.stories.tsx +21 -17
- package/src/stories/Outliner.stories.tsx +17 -14
- package/src/stories/Preview.stories.tsx +27 -26
- package/src/stories/Tags.stories.tsx +81 -0
- package/src/stories/TextEditor.stories.tsx +40 -34
- package/src/stories/components/EditorStory.tsx +9 -10
- package/src/styles/theme.ts +8 -6
- package/src/testing/util.ts +2 -0
- package/src/translations.ts +1 -1
- package/src/util/cursor.ts +2 -1
- package/src/util/debug.ts +2 -2
- package/src/util/decorations.ts +21 -0
- package/src/util/dom.ts +5 -27
- package/src/util/domino.ts +51 -0
- package/src/util/index.ts +2 -0
- package/src/util/react.tsx +1 -1
@@ -3,40 +3,45 @@
|
|
3
3
|
//
|
4
4
|
|
5
5
|
import { createContext } from '@radix-ui/react-context';
|
6
|
-
import React, { type PropsWithChildren,
|
6
|
+
import React, { type PropsWithChildren, type RefObject, useCallback, useEffect, useRef, useState } from 'react';
|
7
7
|
|
8
8
|
import { addEventListener } from '@dxos/async';
|
9
|
-
import { type
|
9
|
+
import { type DxAnchor, type DxAnchorActivate } from '@dxos/lit-ui';
|
10
10
|
import { DropdownMenu } from '@dxos/react-ui';
|
11
11
|
|
12
12
|
import { type PreviewLinkRef, type PreviewLinkTarget, type PreviewLookup } from '../../extensions';
|
13
13
|
|
14
|
-
// TODO(burdon):
|
14
|
+
// TODO(burdon): Move to @dxos/lit-ui
|
15
15
|
|
16
|
-
|
16
|
+
//
|
17
|
+
// Context
|
18
|
+
//
|
17
19
|
|
18
|
-
// Create a context for the dxn value.
|
19
20
|
type RefDropdownMenuValue = Partial<{
|
20
21
|
link: PreviewLinkRef;
|
21
22
|
target: PreviewLinkTarget;
|
22
23
|
pending: boolean;
|
23
24
|
}>;
|
24
25
|
|
25
|
-
const
|
26
|
-
|
26
|
+
const [RefDropdownMenuContextProvider, useRefDropdownMenu] = createContext<RefDropdownMenuValue>('RefDropdownMenu', {});
|
27
|
+
|
28
|
+
//
|
29
|
+
// Context Provider
|
30
|
+
// NOTE: This is handled by the preview-plugin.
|
31
|
+
//
|
27
32
|
|
28
33
|
type RefDropdownMenuProviderProps = PropsWithChildren<{
|
29
34
|
onLookup?: PreviewLookup;
|
30
35
|
}>;
|
31
36
|
|
32
37
|
const RefDropdownMenuProvider = ({ children, onLookup }: RefDropdownMenuProviderProps) => {
|
33
|
-
const trigger = useRef<
|
38
|
+
const trigger = useRef<DxAnchor | null>(null);
|
34
39
|
const [value, setValue] = useState<RefDropdownMenuValue>({});
|
35
40
|
const [rootRef, setRootRef] = useState<HTMLDivElement | null>(null);
|
36
41
|
const [open, setOpen] = useState(false);
|
37
42
|
|
38
|
-
const
|
39
|
-
(event:
|
43
|
+
const handleDxAnchorActivate = useCallback(
|
44
|
+
(event: DxAnchorActivate) => {
|
40
45
|
const { refId, label, trigger: dxTrigger } = event;
|
41
46
|
setValue((value) => ({
|
42
47
|
...value,
|
@@ -61,7 +66,10 @@ const RefDropdownMenuProvider = ({ children, onLookup }: RefDropdownMenuProvider
|
|
61
66
|
return;
|
62
67
|
}
|
63
68
|
|
64
|
-
return addEventListener(rootRef, 'dx-
|
69
|
+
return addEventListener(rootRef, 'dx-anchor-activate' as any, handleDxAnchorActivate, {
|
70
|
+
capture: true,
|
71
|
+
passive: false,
|
72
|
+
});
|
65
73
|
}, [rootRef]);
|
66
74
|
|
67
75
|
return (
|
@@ -76,10 +84,6 @@ const RefDropdownMenuProvider = ({ children, onLookup }: RefDropdownMenuProvider
|
|
76
84
|
);
|
77
85
|
};
|
78
86
|
|
79
|
-
export
|
80
|
-
Provider: RefDropdownMenuProvider,
|
81
|
-
};
|
82
|
-
|
83
|
-
export { useRefDropdownMenu };
|
87
|
+
export { RefDropdownMenuProvider };
|
84
88
|
|
85
89
|
export type { RefDropdownMenuProviderProps, RefDropdownMenuValue };
|
@@ -5,67 +5,49 @@
|
|
5
5
|
import { createContext } from '@radix-ui/react-context';
|
6
6
|
import React, {
|
7
7
|
type PropsWithChildren,
|
8
|
-
useRef,
|
9
|
-
useState,
|
10
|
-
useEffect,
|
11
|
-
useCallback,
|
12
8
|
type RefObject,
|
13
9
|
forwardRef,
|
10
|
+
useCallback,
|
11
|
+
useEffect,
|
12
|
+
useRef,
|
13
|
+
useState,
|
14
14
|
} from 'react';
|
15
15
|
|
16
16
|
import { addEventListener } from '@dxos/async';
|
17
|
-
import { type
|
17
|
+
import { type DxAnchor, type DxAnchorActivate } from '@dxos/lit-ui';
|
18
18
|
import { Popover } from '@dxos/react-ui';
|
19
19
|
|
20
20
|
import { type PreviewLinkRef, type PreviewLinkTarget, type PreviewLookup } from '../../extensions';
|
21
21
|
|
22
|
-
|
22
|
+
// TODO(burdon): Move to @dxos/lit-ui
|
23
23
|
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
onOpenChange?: (open: boolean) => void;
|
28
|
-
onActivate?: (event: DxRefTagActivate) => void;
|
29
|
-
}>;
|
30
|
-
|
31
|
-
export const RefPopover = forwardRef<DxRefTag | null, RefPopoverProps>(
|
32
|
-
({ children, open, onOpenChange, modal, onActivate }, ref) => {
|
33
|
-
const [rootRef, setRootRef] = useState<HTMLDivElement | null>(null);
|
34
|
-
|
35
|
-
useEffect(() => {
|
36
|
-
if (!rootRef || !onActivate) {
|
37
|
-
return;
|
38
|
-
}
|
39
|
-
|
40
|
-
return addEventListener(rootRef, 'dx-ref-tag-activate' as any, onActivate, customEventOptions);
|
41
|
-
}, [rootRef, onActivate]);
|
24
|
+
//
|
25
|
+
// Context
|
26
|
+
//
|
42
27
|
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
</div>
|
49
|
-
</Popover.Root>
|
50
|
-
);
|
51
|
-
},
|
52
|
-
);
|
28
|
+
type RefPopoverValue = Partial<{
|
29
|
+
link: PreviewLinkRef;
|
30
|
+
target: PreviewLinkTarget;
|
31
|
+
pending: boolean;
|
32
|
+
}>;
|
53
33
|
|
54
|
-
|
55
|
-
type RefPopoverValue = Partial<{ link: PreviewLinkRef; target: PreviewLinkTarget; pending: boolean }>;
|
34
|
+
const [RefPopoverContextProvider, useRefPopover] = createContext<RefPopoverValue>('RefPopover', {});
|
56
35
|
|
57
|
-
|
58
|
-
|
36
|
+
//
|
37
|
+
// ContextProvider
|
38
|
+
//
|
59
39
|
|
60
|
-
type PreviewProviderProps = PropsWithChildren<{
|
40
|
+
type PreviewProviderProps = PropsWithChildren<{
|
41
|
+
onLookup?: PreviewLookup;
|
42
|
+
}>;
|
61
43
|
|
62
44
|
const PreviewProvider = ({ children, onLookup }: PreviewProviderProps) => {
|
63
|
-
const trigger = useRef<
|
45
|
+
const trigger = useRef<DxAnchor | null>(null);
|
64
46
|
const [value, setValue] = useState<RefPopoverValue>({});
|
65
47
|
const [open, setOpen] = useState(false);
|
66
48
|
|
67
|
-
const
|
68
|
-
(event:
|
49
|
+
const handleDxAnchorActivate = useCallback(
|
50
|
+
(event: DxAnchorActivate) => {
|
69
51
|
const { refId, label, trigger: dxTrigger } = event;
|
70
52
|
setValue((value) => ({
|
71
53
|
...value,
|
@@ -87,13 +69,49 @@ const PreviewProvider = ({ children, onLookup }: PreviewProviderProps) => {
|
|
87
69
|
|
88
70
|
return (
|
89
71
|
<RefPopoverContextProvider pending={value.pending} link={value.link} target={value.target}>
|
90
|
-
<RefPopover ref={trigger} open={open} onOpenChange={setOpen} onActivate={
|
72
|
+
<RefPopover ref={trigger} open={open} onOpenChange={setOpen} onActivate={handleDxAnchorActivate}>
|
91
73
|
{children}
|
92
74
|
</RefPopover>
|
93
75
|
</RefPopoverContextProvider>
|
94
76
|
);
|
95
77
|
};
|
96
78
|
|
97
|
-
|
79
|
+
//
|
80
|
+
// Popover
|
81
|
+
//
|
82
|
+
|
83
|
+
type RefPopoverProps = PropsWithChildren<{
|
84
|
+
modal?: boolean;
|
85
|
+
open?: boolean;
|
86
|
+
onOpenChange?: (open: boolean) => void;
|
87
|
+
onActivate?: (event: DxAnchorActivate) => void;
|
88
|
+
}>;
|
89
|
+
|
90
|
+
/**
|
91
|
+
* Wraps components that contain <dx-anchor> elements?
|
92
|
+
*/
|
93
|
+
const RefPopover = forwardRef<DxAnchor | null, RefPopoverProps>(
|
94
|
+
({ children, open, onOpenChange, modal, onActivate }, ref) => {
|
95
|
+
const [rootRef, setRootRef] = useState<HTMLDivElement | null>(null);
|
96
|
+
useEffect(() => {
|
97
|
+
if (!rootRef || !onActivate) {
|
98
|
+
return;
|
99
|
+
}
|
100
|
+
|
101
|
+
return addEventListener(rootRef, 'dx-anchor-activate' as any, onActivate, { capture: true, passive: false });
|
102
|
+
}, [rootRef, onActivate]);
|
103
|
+
|
104
|
+
return (
|
105
|
+
<Popover.Root open={open} onOpenChange={onOpenChange} modal={modal}>
|
106
|
+
<Popover.VirtualTrigger virtualRef={ref as unknown as RefObject<HTMLButtonElement>} />
|
107
|
+
<div role='none' className='contents' ref={setRootRef}>
|
108
|
+
{children}
|
109
|
+
</div>
|
110
|
+
</Popover.Root>
|
111
|
+
);
|
112
|
+
},
|
113
|
+
);
|
114
|
+
|
115
|
+
export { RefPopover, PreviewProvider, useRefPopover };
|
98
116
|
|
99
|
-
export type { PreviewProviderProps, RefPopoverValue };
|
117
|
+
export type { RefPopoverProps, PreviewProviderProps, RefPopoverValue };
|
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
|
);
|
@@ -1,69 +1,219 @@
|
|
1
1
|
//
|
2
|
-
// Copyright
|
2
|
+
// Copyright 2025 DXOS.org
|
3
3
|
//
|
4
4
|
|
5
|
+
import { type Extension, Prec } from '@codemirror/state';
|
5
6
|
import {
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
type
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
import { type Extension } from '@codemirror/state';
|
15
|
-
import { keymap } from '@codemirror/view';
|
16
|
-
|
17
|
-
export type AutocompleteResult = Completion;
|
7
|
+
Decoration,
|
8
|
+
type DecorationSet,
|
9
|
+
EditorView,
|
10
|
+
ViewPlugin,
|
11
|
+
type ViewUpdate,
|
12
|
+
WidgetType,
|
13
|
+
keymap,
|
14
|
+
} from '@codemirror/view';
|
18
15
|
|
19
16
|
export type AutocompleteOptions = {
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
17
|
+
fireIfEmpty?: boolean;
|
18
|
+
|
19
|
+
/**
|
20
|
+
* Callback triggered when Enter is pressed.
|
21
|
+
* @param text The current text in the editor
|
22
|
+
* @returns true if the editor should reset the document.
|
23
|
+
*/
|
24
|
+
onSubmit?: (text: string) => boolean | void;
|
24
25
|
|
25
|
-
|
26
|
-
|
27
|
-
|
26
|
+
/**
|
27
|
+
* Function that returns a list of suggestions based on the current text.
|
28
|
+
* @param text The current text before the cursor
|
29
|
+
* @returns Array of suggestion strings
|
30
|
+
*/
|
31
|
+
onSuggest?: (text: string) => string[];
|
32
|
+
|
33
|
+
/**
|
34
|
+
* ESC pressed.
|
35
|
+
*/
|
36
|
+
onCancel?: () => void;
|
37
|
+
};
|
28
38
|
|
29
39
|
/**
|
30
|
-
*
|
40
|
+
* Creates an autocomplete extension that shows inline suggestions.
|
41
|
+
* Pressing Tab will complete the suggestion.
|
31
42
|
*/
|
32
|
-
export const autocomplete = ({
|
33
|
-
const
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
43
|
+
export const autocomplete = ({ fireIfEmpty, onSubmit, onSuggest, onCancel }: AutocompleteOptions): Extension => {
|
44
|
+
const suggest = ViewPlugin.fromClass(
|
45
|
+
class {
|
46
|
+
_decorations: DecorationSet;
|
47
|
+
_currentSuggestion: string | null = null;
|
48
|
+
|
49
|
+
constructor(view: EditorView) {
|
50
|
+
this._decorations = this.computeDecorations(view);
|
51
|
+
}
|
52
|
+
|
53
|
+
update(update: ViewUpdate) {
|
54
|
+
if (update.docChanged || update.selectionSet) {
|
55
|
+
this._decorations = this.computeDecorations(update.view);
|
56
|
+
}
|
57
|
+
}
|
58
|
+
|
59
|
+
private computeDecorations(view: EditorView): DecorationSet {
|
60
|
+
const text = view.state.doc.toString();
|
61
|
+
const suggestions = onSuggest?.(text) ?? [];
|
62
|
+
if (!suggestions.length) {
|
63
|
+
this._currentSuggestion = null;
|
64
|
+
return Decoration.none;
|
65
|
+
}
|
66
|
+
|
67
|
+
// Get the first suggestion.
|
68
|
+
this._currentSuggestion = suggestions[0];
|
69
|
+
const suffix = this._currentSuggestion.slice(text.length);
|
70
|
+
if (!suffix) {
|
71
|
+
return Decoration.none;
|
72
|
+
}
|
73
|
+
|
74
|
+
// Always show ghost text at the end of the document.
|
75
|
+
return Decoration.set([
|
76
|
+
Decoration.widget({
|
77
|
+
widget: new InlineSuggestionWidget(suffix),
|
78
|
+
side: 1,
|
79
|
+
}).range(view.state.doc.length),
|
80
|
+
]);
|
81
|
+
}
|
82
|
+
|
83
|
+
completeSuggestion(view: EditorView): boolean {
|
84
|
+
if (!this._currentSuggestion) {
|
85
|
+
return false;
|
86
|
+
}
|
87
|
+
|
88
|
+
const text = view.state.doc.toString();
|
89
|
+
const suffix = this._currentSuggestion.slice(text.length);
|
90
|
+
if (!suffix) {
|
91
|
+
return false;
|
92
|
+
}
|
93
|
+
|
94
|
+
view.dispatch({
|
95
|
+
changes: {
|
96
|
+
from: view.state.doc.length,
|
97
|
+
insert: suffix,
|
98
|
+
},
|
99
|
+
selection: {
|
100
|
+
anchor: view.state.doc.length + suffix.length,
|
101
|
+
},
|
102
|
+
});
|
103
|
+
|
104
|
+
return true;
|
105
|
+
}
|
106
|
+
},
|
107
|
+
{
|
108
|
+
decorations: (v) => v._decorations,
|
109
|
+
},
|
110
|
+
);
|
111
|
+
|
112
|
+
return [
|
113
|
+
suggest,
|
114
|
+
EditorView.theme({
|
115
|
+
'.cm-inline-suggestion': {
|
116
|
+
opacity: 0.4,
|
117
|
+
},
|
46
118
|
}),
|
47
|
-
];
|
48
119
|
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
return
|
57
|
-
}
|
58
|
-
|
59
|
-
return {
|
60
|
-
from: match.from,
|
61
|
-
options: onSearch(match.text.toLowerCase()),
|
62
|
-
};
|
120
|
+
Prec.highest(
|
121
|
+
keymap.of([
|
122
|
+
{
|
123
|
+
key: 'Tab',
|
124
|
+
preventDefault: true,
|
125
|
+
run: (view) => {
|
126
|
+
const plugin = view.plugin(suggest);
|
127
|
+
return plugin?.completeSuggestion(view) ?? false;
|
128
|
+
},
|
63
129
|
},
|
64
|
-
|
65
|
-
|
66
|
-
|
130
|
+
{
|
131
|
+
key: 'ArrowRight',
|
132
|
+
preventDefault: true,
|
133
|
+
run: (view) => {
|
134
|
+
// Only complete if cursor is at the end
|
135
|
+
if (view.state.selection.main.head !== view.state.doc.length) {
|
136
|
+
return false;
|
137
|
+
}
|
138
|
+
|
139
|
+
const plugin = view.plugin(suggest);
|
140
|
+
return plugin?.completeSuggestion(view) ?? false;
|
141
|
+
},
|
142
|
+
},
|
143
|
+
{
|
144
|
+
key: 'Enter',
|
145
|
+
preventDefault: true,
|
146
|
+
run: (view) => {
|
147
|
+
const text = view.state.doc.toString().trim();
|
148
|
+
if (onSubmit && (fireIfEmpty || text.length > 0)) {
|
149
|
+
const reset = onSubmit(text);
|
150
|
+
|
151
|
+
// Clear the document after calling onEnter.
|
152
|
+
if (reset) {
|
153
|
+
view.dispatch({
|
154
|
+
changes: {
|
155
|
+
from: 0,
|
156
|
+
to: view.state.doc.length,
|
157
|
+
insert: '',
|
158
|
+
},
|
159
|
+
});
|
160
|
+
}
|
161
|
+
}
|
67
162
|
|
68
|
-
|
163
|
+
return true;
|
164
|
+
},
|
165
|
+
},
|
166
|
+
{
|
167
|
+
key: 'Shift-Enter',
|
168
|
+
preventDefault: true,
|
169
|
+
run: (view) => {
|
170
|
+
view.dispatch({
|
171
|
+
changes: {
|
172
|
+
from: view.state.selection.main.head,
|
173
|
+
insert: '\n',
|
174
|
+
},
|
175
|
+
selection: {
|
176
|
+
anchor: view.state.selection.main.head + 1,
|
177
|
+
head: view.state.selection.main.head + 1,
|
178
|
+
},
|
179
|
+
});
|
180
|
+
return true;
|
181
|
+
},
|
182
|
+
},
|
183
|
+
{
|
184
|
+
key: 'Escape',
|
185
|
+
preventDefault: true,
|
186
|
+
run: (view) => {
|
187
|
+
// Clear the entire document.
|
188
|
+
view.dispatch({
|
189
|
+
changes: {
|
190
|
+
from: 0,
|
191
|
+
to: view.state.doc.length,
|
192
|
+
insert: '',
|
193
|
+
},
|
194
|
+
});
|
195
|
+
onCancel?.();
|
196
|
+
return true;
|
197
|
+
},
|
198
|
+
},
|
199
|
+
]),
|
200
|
+
),
|
201
|
+
];
|
69
202
|
};
|
203
|
+
|
204
|
+
class InlineSuggestionWidget extends WidgetType {
|
205
|
+
constructor(private suffix: string) {
|
206
|
+
super();
|
207
|
+
}
|
208
|
+
|
209
|
+
override toDOM(): HTMLSpanElement {
|
210
|
+
const span = document.createElement('span');
|
211
|
+
span.textContent = this.suffix;
|
212
|
+
span.className = 'cm-inline-suggestion';
|
213
|
+
return span;
|
214
|
+
}
|
215
|
+
|
216
|
+
override eq(other: InlineSuggestionWidget): boolean {
|
217
|
+
return other.suffix === this.suffix;
|
218
|
+
}
|
219
|
+
}
|
@@ -8,14 +8,15 @@ import '@preact/signals-react';
|
|
8
8
|
|
9
9
|
import { Repo } from '@automerge/automerge-repo';
|
10
10
|
import { BroadcastChannelNetworkAdapter } from '@automerge/automerge-repo-network-broadcastchannel';
|
11
|
+
import { type Meta, type StoryObj } from '@storybook/react-vite';
|
11
12
|
import React, { useEffect, useState } from 'react';
|
12
13
|
|
13
14
|
import { Obj, Ref, Type } from '@dxos/echo';
|
14
|
-
import { DocAccessor,
|
15
|
-
import {
|
16
|
-
import {
|
15
|
+
import { DocAccessor, Query, type Space, createDocAccessor, useQuery, useSpace } from '@dxos/react-client/echo';
|
16
|
+
import { type Identity, useIdentity } from '@dxos/react-client/halo';
|
17
|
+
import { type ClientRepeatedComponentProps, ClientRepeater } from '@dxos/react-client/testing';
|
17
18
|
import { useThemeContext } from '@dxos/react-ui';
|
18
|
-
import { withLayout, withTheme } from '@dxos/storybook-utils';
|
19
|
+
import { render, withLayout, withTheme } from '@dxos/storybook-utils';
|
19
20
|
|
20
21
|
import { editorSlots } from '../../defaults';
|
21
22
|
import { useTextEditor } from '../../hooks';
|
@@ -41,7 +42,7 @@ const Editor = ({ source, autoFocus, space, identity }: EditorProps) => {
|
|
41
42
|
() => ({
|
42
43
|
initialValue: DocAccessor.getValue(source),
|
43
44
|
extensions: [
|
44
|
-
createBasicExtensions({ placeholder: 'Type here...' }),
|
45
|
+
createBasicExtensions({ placeholder: 'Type here...', search: true }),
|
45
46
|
createThemeExtensions({ themeMode, slots: editorSlots }),
|
46
47
|
createDataExtensions({ id: 'test', text: source, space, identity }),
|
47
48
|
],
|
@@ -53,7 +54,7 @@ const Editor = ({ source, autoFocus, space, identity }: EditorProps) => {
|
|
53
54
|
return <div ref={parentRef} className='flex w-full' />;
|
54
55
|
};
|
55
56
|
|
56
|
-
const
|
57
|
+
const DefaultStory = () => {
|
57
58
|
const [object1, setObject1] = useState<DocAccessor<TestObject>>();
|
58
59
|
const [object2, setObject2] = useState<DocAccessor<TestObject>>();
|
59
60
|
|
@@ -88,16 +89,6 @@ const Story = () => {
|
|
88
89
|
);
|
89
90
|
};
|
90
91
|
|
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
92
|
const EchoStory = ({ spaceKey }: ClientRepeatedComponentProps) => {
|
102
93
|
const identity = useIdentity();
|
103
94
|
const space = useSpace(spaceKey);
|
@@ -118,9 +109,25 @@ const EchoStory = ({ spaceKey }: ClientRepeatedComponentProps) => {
|
|
118
109
|
return <Editor source={source} space={space} identity={identity ?? undefined} />;
|
119
110
|
};
|
120
111
|
|
121
|
-
|
112
|
+
const meta = {
|
113
|
+
title: 'ui/react-ui-editor/Automerge',
|
114
|
+
component: Editor as any,
|
115
|
+
render: render(DefaultStory),
|
116
|
+
decorators: [withTheme, withLayout({ fullscreen: true })],
|
117
|
+
parameters: {
|
118
|
+
translations,
|
119
|
+
},
|
120
|
+
} satisfies Meta<typeof DefaultStory>;
|
121
|
+
|
122
|
+
export default meta;
|
123
|
+
|
124
|
+
type Story = StoryObj<typeof meta>;
|
125
|
+
|
126
|
+
export const Default: Story = {
|
127
|
+
args: {},
|
128
|
+
};
|
122
129
|
|
123
|
-
export const WithEcho = {
|
130
|
+
export const WithEcho: Story = {
|
124
131
|
decorators: [withTheme],
|
125
132
|
render: () => {
|
126
133
|
return (
|
@@ -5,15 +5,16 @@
|
|
5
5
|
//
|
6
6
|
|
7
7
|
import { next as A } from '@automerge/automerge';
|
8
|
-
import {
|
8
|
+
import { type Extension, StateField } from '@codemirror/state';
|
9
9
|
import { EditorView, ViewPlugin } from '@codemirror/view';
|
10
10
|
|
11
11
|
import { type DocAccessor } from '@dxos/react-client/echo';
|
12
12
|
|
13
|
+
import { Cursor } from '../../util';
|
14
|
+
|
13
15
|
import { cursorConverter } from './cursor';
|
14
|
-
import {
|
16
|
+
import { type State, isReconcile, updateHeadsEffect } from './defs';
|
15
17
|
import { Syncer } from './sync';
|
16
|
-
import { Cursor } from '../../util';
|
17
18
|
|
18
19
|
export const automerge = (accessor: DocAccessor): Extension => {
|
19
20
|
const syncState = StateField.define<State>({
|
@@ -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[];
|
@@ -10,7 +10,7 @@ import { type EditorView } from '@codemirror/view';
|
|
10
10
|
|
11
11
|
import { type IDocHandle } from '@dxos/react-client/echo';
|
12
12
|
|
13
|
-
import { getLastHeads, getPath, isReconcile, reconcileAnnotation,
|
13
|
+
import { type State, getLastHeads, getPath, isReconcile, reconcileAnnotation, updateHeads } from './defs';
|
14
14
|
import { updateAutomerge } from './update-automerge';
|
15
15
|
import { updateCodeMirror } from './update-codemirror';
|
16
16
|
|
@@ -5,7 +5,7 @@
|
|
5
5
|
//
|
6
6
|
|
7
7
|
import { next as A, type Heads } from '@automerge/automerge';
|
8
|
-
import { type EditorState, type StateField, type
|
8
|
+
import { type EditorState, type StateField, type Text, type Transaction } from '@codemirror/state';
|
9
9
|
|
10
10
|
import { type IDocHandle } from '@dxos/react-client/echo';
|
11
11
|
|