@dxos/react-ui-editor 0.8.1-staging.9eaf14f → 0.8.2-main.10c050d

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.
Files changed (268) hide show
  1. package/dist/lib/browser/index.mjs +4152 -2852
  2. package/dist/lib/browser/index.mjs.map +4 -4
  3. package/dist/lib/browser/meta.json +1 -1
  4. package/dist/lib/browser/testing/index.mjs +6 -0
  5. package/dist/lib/browser/testing/index.mjs.map +7 -0
  6. package/dist/lib/node/index.cjs +3318 -2009
  7. package/dist/lib/node/index.cjs.map +4 -4
  8. package/dist/lib/node/meta.json +1 -1
  9. package/dist/lib/node/testing/index.cjs +29 -0
  10. package/dist/lib/node/testing/index.cjs.map +7 -0
  11. package/dist/lib/node-esm/index.mjs +4152 -2852
  12. package/dist/lib/node-esm/index.mjs.map +4 -4
  13. package/dist/lib/node-esm/meta.json +1 -1
  14. package/dist/lib/node-esm/testing/index.mjs +8 -0
  15. package/dist/lib/node-esm/testing/index.mjs.map +7 -0
  16. package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts +1 -1
  17. package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts.map +1 -1
  18. package/dist/types/src/components/EditorToolbar/blocks.d.ts +4 -3
  19. package/dist/types/src/components/EditorToolbar/blocks.d.ts.map +1 -1
  20. package/dist/types/src/components/EditorToolbar/formatting.d.ts +4 -3
  21. package/dist/types/src/components/EditorToolbar/formatting.d.ts.map +1 -1
  22. package/dist/types/src/components/EditorToolbar/headings.d.ts +4 -3
  23. package/dist/types/src/components/EditorToolbar/headings.d.ts.map +1 -1
  24. package/dist/types/src/components/EditorToolbar/{comment.d.ts → image.d.ts} +4 -5
  25. package/dist/types/src/components/EditorToolbar/image.d.ts.map +1 -0
  26. package/dist/types/src/components/EditorToolbar/index.d.ts +1 -1
  27. package/dist/types/src/components/EditorToolbar/index.d.ts.map +1 -1
  28. package/dist/types/src/components/EditorToolbar/lists.d.ts +4 -3
  29. package/dist/types/src/components/EditorToolbar/lists.d.ts.map +1 -1
  30. package/dist/types/src/components/EditorToolbar/search.d.ts +17 -0
  31. package/dist/types/src/components/EditorToolbar/search.d.ts.map +1 -0
  32. package/dist/types/src/components/EditorToolbar/util.d.ts +17 -25
  33. package/dist/types/src/components/EditorToolbar/util.d.ts.map +1 -1
  34. package/dist/types/src/components/EditorToolbar/{viewMode.d.ts → view-mode.d.ts} +5 -4
  35. package/dist/types/src/components/EditorToolbar/view-mode.d.ts.map +1 -0
  36. package/dist/types/src/components/Popover/RefDropdownMenu.d.ts +21 -0
  37. package/dist/types/src/components/Popover/RefDropdownMenu.d.ts.map +1 -0
  38. package/dist/types/src/components/Popover/RefPopover.d.ts +21 -0
  39. package/dist/types/src/components/Popover/RefPopover.d.ts.map +1 -0
  40. package/dist/types/src/components/Popover/index.d.ts +3 -0
  41. package/dist/types/src/components/Popover/index.d.ts.map +1 -0
  42. package/dist/types/src/components/index.d.ts +1 -0
  43. package/dist/types/src/components/index.d.ts.map +1 -1
  44. package/dist/types/src/defaults.d.ts +3 -5
  45. package/dist/types/src/defaults.d.ts.map +1 -1
  46. package/dist/types/src/extensions/annotations.d.ts +4 -1
  47. package/dist/types/src/extensions/annotations.d.ts.map +1 -1
  48. package/dist/types/src/extensions/autocomplete.d.ts +1 -2
  49. package/dist/types/src/extensions/autocomplete.d.ts.map +1 -1
  50. package/dist/types/src/extensions/automerge/automerge.d.ts.map +1 -1
  51. package/dist/types/src/extensions/automerge/automerge.stories.d.ts.map +1 -1
  52. package/dist/types/src/extensions/automerge/cursor.d.ts.map +1 -1
  53. package/dist/types/src/extensions/automerge/defs.d.ts +1 -1
  54. package/dist/types/src/extensions/automerge/defs.d.ts.map +1 -1
  55. package/dist/types/src/extensions/automerge/sync.d.ts.map +1 -1
  56. package/dist/types/src/extensions/automerge/update-automerge.d.ts +1 -1
  57. package/dist/types/src/extensions/automerge/update-automerge.d.ts.map +1 -1
  58. package/dist/types/src/extensions/automerge/update-codemirror.d.ts +1 -1
  59. package/dist/types/src/extensions/automerge/update-codemirror.d.ts.map +1 -1
  60. package/dist/types/src/extensions/awareness/awareness-provider.d.ts.map +1 -1
  61. package/dist/types/src/extensions/awareness/awareness.d.ts.map +1 -1
  62. package/dist/types/src/extensions/blast.d.ts.map +1 -1
  63. package/dist/types/src/extensions/command/action.d.ts +17 -0
  64. package/dist/types/src/extensions/command/action.d.ts.map +1 -0
  65. package/dist/types/src/extensions/command/command.d.ts +4 -10
  66. package/dist/types/src/extensions/command/command.d.ts.map +1 -1
  67. package/dist/types/src/extensions/command/hint.d.ts +18 -4
  68. package/dist/types/src/extensions/command/hint.d.ts.map +1 -1
  69. package/dist/types/src/extensions/command/index.d.ts +3 -0
  70. package/dist/types/src/extensions/command/index.d.ts.map +1 -1
  71. package/dist/types/src/extensions/command/menu.d.ts +6 -11
  72. package/dist/types/src/extensions/command/menu.d.ts.map +1 -1
  73. package/dist/types/src/extensions/command/state.d.ts +9 -11
  74. package/dist/types/src/extensions/command/state.d.ts.map +1 -1
  75. package/dist/types/src/extensions/command/typeahead.d.ts +17 -0
  76. package/dist/types/src/extensions/command/typeahead.d.ts.map +1 -0
  77. package/dist/types/src/extensions/comments.d.ts +9 -17
  78. package/dist/types/src/extensions/comments.d.ts.map +1 -1
  79. package/dist/types/src/extensions/debug.d.ts.map +1 -1
  80. package/dist/types/src/extensions/dnd.d.ts.map +1 -1
  81. package/dist/types/src/extensions/factories.d.ts +4 -0
  82. package/dist/types/src/extensions/factories.d.ts.map +1 -1
  83. package/dist/types/src/extensions/folding.d.ts.map +1 -1
  84. package/dist/types/src/extensions/index.d.ts +3 -0
  85. package/dist/types/src/extensions/index.d.ts.map +1 -1
  86. package/dist/types/src/extensions/json.d.ts +7 -0
  87. package/dist/types/src/extensions/json.d.ts.map +1 -0
  88. package/dist/types/src/extensions/listener.d.ts.map +1 -1
  89. package/dist/types/src/extensions/markdown/{editorAction.d.ts → action.d.ts} +1 -1
  90. package/dist/types/src/extensions/markdown/action.d.ts.map +1 -0
  91. package/dist/types/src/extensions/markdown/bundle.d.ts +2 -1
  92. package/dist/types/src/extensions/markdown/bundle.d.ts.map +1 -1
  93. package/dist/types/src/extensions/markdown/changes.d.ts.map +1 -1
  94. package/dist/types/src/extensions/markdown/debug.d.ts.map +1 -1
  95. package/dist/types/src/extensions/markdown/decorate.d.ts +5 -1
  96. package/dist/types/src/extensions/markdown/decorate.d.ts.map +1 -1
  97. package/dist/types/src/extensions/markdown/formatting.d.ts +3 -3
  98. package/dist/types/src/extensions/markdown/formatting.d.ts.map +1 -1
  99. package/dist/types/src/extensions/markdown/highlight.d.ts.map +1 -1
  100. package/dist/types/src/extensions/markdown/image.d.ts.map +1 -1
  101. package/dist/types/src/extensions/markdown/index.d.ts +1 -1
  102. package/dist/types/src/extensions/markdown/index.d.ts.map +1 -1
  103. package/dist/types/src/extensions/markdown/link.d.ts +4 -1
  104. package/dist/types/src/extensions/markdown/link.d.ts.map +1 -1
  105. package/dist/types/src/extensions/markdown/styles.d.ts.map +1 -1
  106. package/dist/types/src/extensions/markdown/table.d.ts.map +1 -1
  107. package/dist/types/src/extensions/mention.d.ts.map +1 -1
  108. package/dist/types/src/extensions/modes.d.ts +5 -5
  109. package/dist/types/src/extensions/modes.d.ts.map +1 -1
  110. package/dist/types/src/extensions/outliner/commands.d.ts +10 -0
  111. package/dist/types/src/extensions/outliner/commands.d.ts.map +1 -0
  112. package/dist/types/src/extensions/outliner/editor.d.ts +5 -0
  113. package/dist/types/src/extensions/outliner/editor.d.ts.map +1 -0
  114. package/dist/types/src/extensions/outliner/editor.test.d.ts +2 -0
  115. package/dist/types/src/extensions/outliner/editor.test.d.ts.map +1 -0
  116. package/dist/types/src/extensions/outliner/index.d.ts +4 -0
  117. package/dist/types/src/extensions/outliner/index.d.ts.map +1 -0
  118. package/dist/types/src/extensions/outliner/outliner.d.ts +13 -0
  119. package/dist/types/src/extensions/outliner/outliner.d.ts.map +1 -0
  120. package/dist/types/src/extensions/outliner/outliner.test.d.ts +2 -0
  121. package/dist/types/src/extensions/outliner/outliner.test.d.ts.map +1 -0
  122. package/dist/types/src/extensions/outliner/selection.d.ts +12 -0
  123. package/dist/types/src/extensions/outliner/selection.d.ts.map +1 -0
  124. package/dist/types/src/extensions/outliner/tree.d.ts +79 -0
  125. package/dist/types/src/extensions/outliner/tree.d.ts.map +1 -0
  126. package/dist/types/src/extensions/outliner/tree.test.d.ts +2 -0
  127. package/dist/types/src/extensions/outliner/tree.test.d.ts.map +1 -0
  128. package/dist/types/src/extensions/preview/index.d.ts +2 -0
  129. package/dist/types/src/extensions/preview/index.d.ts.map +1 -0
  130. package/dist/types/src/extensions/preview/preview.d.ts +39 -0
  131. package/dist/types/src/extensions/preview/preview.d.ts.map +1 -0
  132. package/dist/types/src/extensions/selection.d.ts.map +1 -1
  133. package/dist/types/src/extensions/typewriter.d.ts.map +1 -1
  134. package/dist/types/src/hooks/index.d.ts +0 -1
  135. package/dist/types/src/hooks/index.d.ts.map +1 -1
  136. package/dist/types/src/hooks/useTextEditor.d.ts +2 -1
  137. package/dist/types/src/hooks/useTextEditor.d.ts.map +1 -1
  138. package/dist/types/src/stories/Command.stories.d.ts +7 -0
  139. package/dist/types/src/stories/Command.stories.d.ts.map +1 -0
  140. package/dist/types/src/stories/Comments.stories.d.ts +13 -0
  141. package/dist/types/src/stories/Comments.stories.d.ts.map +1 -0
  142. package/dist/types/src/stories/EditorToolbar.stories.d.ts +12 -0
  143. package/dist/types/src/stories/EditorToolbar.stories.d.ts.map +1 -0
  144. package/dist/types/src/stories/Experimental.stories.d.ts +16 -0
  145. package/dist/types/src/stories/Experimental.stories.d.ts.map +1 -0
  146. package/dist/types/src/stories/Markdown.stories.d.ts +46 -0
  147. package/dist/types/src/stories/Markdown.stories.d.ts.map +1 -0
  148. package/dist/types/src/stories/Outliner.stories.d.ts +26 -0
  149. package/dist/types/src/stories/Outliner.stories.d.ts.map +1 -0
  150. package/dist/types/src/stories/Preview.stories.d.ts +10 -0
  151. package/dist/types/src/stories/Preview.stories.d.ts.map +1 -0
  152. package/dist/types/src/stories/TextEditor.stories.d.ts +55 -0
  153. package/dist/types/src/stories/TextEditor.stories.d.ts.map +1 -0
  154. package/dist/types/src/stories/util.d.ts +53 -0
  155. package/dist/types/src/stories/util.d.ts.map +1 -0
  156. package/dist/types/src/styles/theme.d.ts.map +1 -1
  157. package/dist/types/src/styles/tokens.d.ts.map +1 -1
  158. package/dist/types/src/testing/index.d.ts +2 -0
  159. package/dist/types/src/testing/index.d.ts.map +1 -0
  160. package/dist/types/src/testing/util.d.ts +2 -0
  161. package/dist/types/src/testing/util.d.ts.map +1 -0
  162. package/dist/types/src/types.d.ts +5 -0
  163. package/dist/types/src/types.d.ts.map +1 -1
  164. package/dist/types/src/util/cursor.d.ts.map +1 -1
  165. package/dist/types/src/util/debug.d.ts.map +1 -1
  166. package/dist/types/src/util/dom.d.ts.map +1 -1
  167. package/dist/types/src/util/facet.d.ts.map +1 -1
  168. package/dist/types/src/util/react.d.ts +6 -1
  169. package/dist/types/src/util/react.d.ts.map +1 -1
  170. package/dist/types/tsconfig.tsbuildinfo +1 -1
  171. package/package.json +46 -30
  172. package/src/components/EditorToolbar/EditorToolbar.tsx +95 -72
  173. package/src/components/EditorToolbar/blocks.ts +27 -6
  174. package/src/components/EditorToolbar/formatting.ts +34 -7
  175. package/src/components/EditorToolbar/headings.ts +9 -8
  176. package/src/components/EditorToolbar/image.ts +16 -0
  177. package/src/components/EditorToolbar/index.ts +7 -1
  178. package/src/components/EditorToolbar/lists.ts +26 -7
  179. package/src/components/EditorToolbar/search.ts +19 -0
  180. package/src/components/EditorToolbar/util.ts +19 -20
  181. package/src/components/EditorToolbar/{viewMode.ts → view-mode.ts} +9 -8
  182. package/src/components/Popover/RefDropdownMenu.tsx +77 -0
  183. package/src/components/Popover/RefPopover.tsx +75 -0
  184. package/src/components/Popover/index.ts +6 -0
  185. package/src/components/index.ts +1 -0
  186. package/src/defaults.ts +12 -13
  187. package/src/extensions/annotations.ts +41 -64
  188. package/src/extensions/autocomplete.ts +5 -6
  189. package/src/extensions/automerge/automerge.stories.tsx +13 -24
  190. package/src/extensions/automerge/automerge.test.tsx +6 -5
  191. package/src/extensions/automerge/automerge.ts +2 -2
  192. package/src/extensions/automerge/defs.ts +1 -2
  193. package/src/extensions/automerge/sync.ts +7 -7
  194. package/src/extensions/automerge/update-automerge.ts +1 -1
  195. package/src/extensions/automerge/update-codemirror.ts +3 -4
  196. package/src/extensions/awareness/awareness-provider.ts +4 -4
  197. package/src/extensions/awareness/awareness.ts +7 -7
  198. package/src/extensions/blast.ts +9 -9
  199. package/src/extensions/command/action.ts +49 -0
  200. package/src/extensions/command/command.ts +7 -27
  201. package/src/extensions/command/hint.ts +36 -33
  202. package/src/extensions/command/index.ts +3 -0
  203. package/src/extensions/command/menu.ts +79 -51
  204. package/src/extensions/command/state.ts +41 -61
  205. package/src/extensions/command/typeahead.ts +116 -0
  206. package/src/extensions/comments.ts +11 -76
  207. package/src/extensions/factories.ts +13 -0
  208. package/src/extensions/folding.tsx +1 -1
  209. package/src/extensions/index.ts +3 -0
  210. package/src/extensions/json.ts +56 -0
  211. package/src/extensions/markdown/bundle.ts +13 -9
  212. package/src/extensions/markdown/changes.ts +3 -2
  213. package/src/extensions/markdown/decorate.ts +19 -17
  214. package/src/extensions/markdown/formatting.ts +6 -6
  215. package/src/extensions/markdown/image.ts +14 -13
  216. package/src/extensions/markdown/index.ts +1 -1
  217. package/src/extensions/markdown/link.ts +33 -24
  218. package/src/extensions/markdown/styles.ts +4 -3
  219. package/src/extensions/markdown/table.ts +3 -3
  220. package/src/extensions/modes.ts +5 -6
  221. package/src/extensions/outliner/commands.ts +270 -0
  222. package/src/extensions/outliner/editor.test.ts +33 -0
  223. package/src/extensions/outliner/editor.ts +184 -0
  224. package/src/extensions/outliner/index.ts +7 -0
  225. package/src/extensions/outliner/outliner.test.ts +99 -0
  226. package/src/extensions/outliner/outliner.ts +168 -0
  227. package/src/extensions/outliner/selection.ts +50 -0
  228. package/src/extensions/outliner/tree.test.ts +164 -0
  229. package/src/extensions/outliner/tree.ts +315 -0
  230. package/src/extensions/preview/index.ts +5 -0
  231. package/src/extensions/preview/preview.ts +271 -0
  232. package/src/hooks/index.ts +0 -1
  233. package/src/hooks/useTextEditor.ts +4 -3
  234. package/src/stories/Command.stories.tsx +97 -0
  235. package/src/stories/Comments.stories.tsx +98 -0
  236. package/src/stories/EditorToolbar.stories.tsx +96 -0
  237. package/src/stories/Experimental.stories.tsx +86 -0
  238. package/src/stories/Markdown.stories.tsx +121 -0
  239. package/src/stories/Outliner.stories.tsx +108 -0
  240. package/src/stories/Preview.stories.tsx +149 -0
  241. package/src/stories/TextEditor.stories.tsx +256 -0
  242. package/src/stories/util.tsx +326 -0
  243. package/src/styles/theme.ts +15 -5
  244. package/src/styles/tokens.ts +1 -2
  245. package/src/testing/index.ts +5 -0
  246. package/src/testing/util.ts +5 -0
  247. package/src/types.ts +7 -0
  248. package/src/util/react.tsx +20 -2
  249. package/dist/types/src/InputMode.stories.d.ts +0 -57
  250. package/dist/types/src/InputMode.stories.d.ts.map +0 -1
  251. package/dist/types/src/TextEditor.stories.d.ts +0 -115
  252. package/dist/types/src/TextEditor.stories.d.ts.map +0 -1
  253. package/dist/types/src/components/EditorToolbar/comment.d.ts.map +0 -1
  254. package/dist/types/src/components/EditorToolbar/viewMode.d.ts.map +0 -1
  255. package/dist/types/src/extensions/command/preview.d.ts +0 -12
  256. package/dist/types/src/extensions/command/preview.d.ts.map +0 -1
  257. package/dist/types/src/extensions/markdown/editorAction.d.ts.map +0 -1
  258. package/dist/types/src/fragments.d.ts +0 -3
  259. package/dist/types/src/fragments.d.ts.map +0 -1
  260. package/dist/types/src/hooks/useActionHandler.d.ts +0 -4
  261. package/dist/types/src/hooks/useActionHandler.d.ts.map +0 -1
  262. package/src/InputMode.stories.tsx +0 -124
  263. package/src/TextEditor.stories.tsx +0 -856
  264. package/src/components/EditorToolbar/comment.ts +0 -23
  265. package/src/extensions/command/preview.ts +0 -79
  266. package/src/fragments.ts +0 -19
  267. package/src/hooks/useActionHandler.ts +0 -12
  268. /package/src/extensions/markdown/{editorAction.ts → action.ts} +0 -0
@@ -0,0 +1,315 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { syntaxTree } from '@codemirror/language';
6
+ import { StateField, type Transaction, type Extension, type EditorState } from '@codemirror/state';
7
+ import { Facet } from '@codemirror/state';
8
+ import { type SyntaxNode } from '@lezer/common';
9
+
10
+ import { invariant } from '@dxos/invariant';
11
+
12
+ import { type Range } from '../../types';
13
+
14
+ /**
15
+ * Represents a single item in the tree.
16
+ */
17
+ export interface Item {
18
+ type: 'root' | 'bullet' | 'task' | 'unknown';
19
+ index: number;
20
+ level: number;
21
+ node: SyntaxNode;
22
+ parent?: Item;
23
+ nextSibling?: Item;
24
+ prevSibling?: Item;
25
+ children: Item[];
26
+ /**
27
+ * Actual range.
28
+ * Starts at the start of the line containing the item and ends at the end of the line before the
29
+ * first child or next sibling.
30
+ */
31
+ lineRange: Range;
32
+ /**
33
+ * Range of the editable content.
34
+ * This doesn't include the list or task marker or indentation.
35
+ */
36
+ contentRange: Range;
37
+ }
38
+
39
+ export const itemToJSON = ({ type, index, level, lineRange, contentRange, children }: Item): any => {
40
+ return { type, index, level, lineRange, contentRange, children: children.map(itemToJSON) };
41
+ };
42
+
43
+ /**
44
+ * Tree assumes the entire document is a single contiguous well-formed hierarchy of markdown LiteItem nodes.
45
+ */
46
+ export class Tree implements Item {
47
+ type: Item['type'] = 'root';
48
+ index = -1;
49
+ level = -1;
50
+ node: Item['node'];
51
+ lineRange: Item['lineRange'];
52
+ contentRange: Item['contentRange'];
53
+ children: Item['children'] = [];
54
+
55
+ constructor(node: SyntaxNode) {
56
+ this.node = node;
57
+ this.lineRange = { from: node.from, to: node.to };
58
+ this.contentRange = this.lineRange;
59
+ }
60
+
61
+ toJSON() {
62
+ return itemToJSON(this);
63
+ }
64
+
65
+ get root(): Item {
66
+ return this;
67
+ }
68
+
69
+ traverse<T = any>(cb: (item: Item, level: number) => T | void): T | undefined;
70
+ traverse<T = any>(item: Item, cb: (item: Item, level: number) => T | void): T | undefined;
71
+ traverse<T = any>(
72
+ itemOrCb: Item | ((item: Item, level: number) => T | void),
73
+ maybeCb?: (item: Item, level: number) => T | void,
74
+ ): T | undefined {
75
+ if (typeof itemOrCb === 'function') {
76
+ return traverse<T>(this, itemOrCb);
77
+ } else {
78
+ return traverse<T>(itemOrCb, maybeCb!);
79
+ }
80
+ }
81
+
82
+ /**
83
+ * Return the closest item.
84
+ */
85
+ find(pos: number): Item | undefined {
86
+ return this.traverse((item) => (item.lineRange.from <= pos && item.lineRange.to >= pos ? item : undefined));
87
+ }
88
+
89
+ /**
90
+ * Return the first child, next sibling, or parent's next sibling.
91
+ */
92
+ next(item: Item, enter = true): Item | undefined {
93
+ if (enter && item.children.length > 0) {
94
+ return item.children[0];
95
+ }
96
+
97
+ if (item.nextSibling) {
98
+ return item.nextSibling;
99
+ }
100
+
101
+ if (item.parent) {
102
+ return this.next(item.parent, false);
103
+ }
104
+
105
+ return undefined;
106
+ }
107
+
108
+ /**
109
+ * Return the previous sibling, or parent.
110
+ */
111
+ prev(item: Item): Item | undefined {
112
+ if (item.prevSibling) {
113
+ return this.lastDescendant(item.prevSibling);
114
+ }
115
+
116
+ return item.parent?.type === 'root' ? undefined : item.parent;
117
+ }
118
+
119
+ /**
120
+ * Return the last descendant of the item, or the item itself if it has no children.
121
+ */
122
+ lastDescendant(item: Item): Item {
123
+ return item.children.length > 0 ? this.lastDescendant(item.children.at(-1)!) : item;
124
+ }
125
+ }
126
+
127
+ export const getRange = (tree: Tree, item: Item): [number, number] => {
128
+ const lastDescendant = tree.lastDescendant(item);
129
+ return [item.lineRange.from, lastDescendant.lineRange.to];
130
+ };
131
+
132
+ /**
133
+ * Traverse the tree, calling the callback for each item.
134
+ * If the callback returns a value, the traversal is stopped and the value is returned.
135
+ */
136
+ export const traverse = <T = any>(root: Item, cb: (item: Item, level: number) => T | void): T | undefined => {
137
+ const t = (item: Item, level: number): T | undefined => {
138
+ if (item.type !== 'root') {
139
+ const value = cb(item, level);
140
+ if (value != null) {
141
+ return value;
142
+ }
143
+ }
144
+
145
+ for (const child of item.children) {
146
+ const value = t(child, level + 1);
147
+ if (value != null) {
148
+ return value;
149
+ }
150
+ }
151
+ };
152
+
153
+ return t(root, root.type === 'root' ? -1 : 0);
154
+ };
155
+
156
+ export const getListItemContent = (state: EditorState, item: Item): string => {
157
+ return state.doc.sliceString(item.contentRange.from, item.contentRange.to);
158
+ };
159
+
160
+ export const listItemToString = (item: Item, level = 0) => {
161
+ const indent = ' '.repeat(level);
162
+ const data = {
163
+ i: item.index,
164
+ n: item.nextSibling?.index ?? '∅',
165
+ p: item.prevSibling?.index ?? '∅',
166
+ level: item.level,
167
+ node: format([item.node.from, item.node.to]),
168
+ line: format([item.lineRange.from, item.lineRange.to]),
169
+ content: format([item.contentRange.from, item.contentRange.to]),
170
+ };
171
+
172
+ return `${indent}${item.type[0].toUpperCase()}(${Object.entries(data)
173
+ .map(([k, v]) => `${k}=${v}`)
174
+ .join(', ')})`;
175
+ };
176
+
177
+ const format = (value: any) =>
178
+ JSON.stringify(value, (key: string, value: any) => {
179
+ if (typeof value === 'number') {
180
+ return value.toString().padStart(3, ' ');
181
+ }
182
+ return value;
183
+ }).replaceAll('"', '');
184
+
185
+ export const treeFacet = Facet.define<Tree, Tree>({
186
+ combine: (values) => values[0],
187
+ });
188
+
189
+ export type TreeOptions = {};
190
+
191
+ /**
192
+ * Creates a shadow tree of `ListItem` nodes whenever the document changes.
193
+ * This adds overhead relative to the markdown AST, but allows for efficient traversal of the list items.
194
+ * NOTE: Requires markdown parser to be enabled.
195
+ */
196
+ export const outlinerTree = (options: TreeOptions = {}): Extension => {
197
+ const buildTree = (state: EditorState): Tree => {
198
+ let tree: Tree | undefined;
199
+ let parent: Item | undefined;
200
+ let current: Item | undefined;
201
+ let prev: Item | undefined;
202
+ let level = -1;
203
+ let index = -1;
204
+
205
+ // Array to track previous siblings at each level.
206
+ const prevSiblings: (Item | undefined)[] = [];
207
+
208
+ syntaxTree(state).iterate({
209
+ enter: (node) => {
210
+ switch (node.name) {
211
+ case 'Document': {
212
+ tree = new Tree(node.node);
213
+ current = tree;
214
+ break;
215
+ }
216
+ case 'BulletList': {
217
+ invariant(current);
218
+ parent = current;
219
+ if (current) {
220
+ current.lineRange.to = current.node.from;
221
+ }
222
+ prevSiblings[++level] = undefined;
223
+ break;
224
+ }
225
+ case 'ListItem': {
226
+ invariant(parent);
227
+
228
+ // Include all content up to the next sibling or the end of the document.
229
+ const nextSibling = node.node.nextSibling ?? node.node.parent?.nextSibling;
230
+ const docRange: Range = {
231
+ from: state.doc.lineAt(node.from).from,
232
+ to: nextSibling ? nextSibling.from - 1 : state.doc.length,
233
+ };
234
+
235
+ current = {
236
+ type: 'unknown',
237
+ index: ++index,
238
+ level,
239
+ node: node.node,
240
+ lineRange: docRange,
241
+ contentRange: { ...docRange },
242
+ parent,
243
+ prevSibling: prevSiblings[level],
244
+ children: [],
245
+ };
246
+
247
+ // Update sibling refs.
248
+ if (current.prevSibling) {
249
+ current.prevSibling.nextSibling = current;
250
+ }
251
+
252
+ // Update previous siblings array at current level.
253
+ prevSiblings[level] = current;
254
+
255
+ // Update previous item (not sibling).
256
+ if (prev) {
257
+ prev.lineRange.to = prev.contentRange.to = current.lineRange.from - 1;
258
+ }
259
+ prev = current;
260
+
261
+ // Update parent.
262
+ parent.children.push(current);
263
+ if (parent.lineRange.to === parent.node.from) {
264
+ parent.lineRange.to = parent.contentRange.to = current.lineRange.from - 1;
265
+ }
266
+
267
+ break;
268
+ }
269
+ case 'ListMark': {
270
+ invariant(current);
271
+ current.type = 'bullet';
272
+ current.contentRange.from = node.from + '- '.length;
273
+ break;
274
+ }
275
+ case 'Task': {
276
+ invariant(current);
277
+ current.type = 'task';
278
+ break;
279
+ }
280
+ case 'TaskMarker': {
281
+ invariant(current);
282
+ current.contentRange.from = node.from + '[ ] '.length;
283
+ break;
284
+ }
285
+ }
286
+ },
287
+ leave: (node) => {
288
+ if (node.name === 'BulletList') {
289
+ invariant(parent);
290
+ prevSiblings[level--] = undefined;
291
+ parent = parent.parent;
292
+ }
293
+ },
294
+ });
295
+
296
+ invariant(tree);
297
+ return tree;
298
+ };
299
+
300
+ return [
301
+ StateField.define<Tree | undefined>({
302
+ create: (state) => {
303
+ return buildTree(state);
304
+ },
305
+ update: (value: Tree | undefined, tr: Transaction) => {
306
+ if (!tr.docChanged) {
307
+ return value;
308
+ }
309
+
310
+ return buildTree(tr.state);
311
+ },
312
+ provide: (field) => treeFacet.from(field),
313
+ }),
314
+ ];
315
+ };
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ export * from './preview';
@@ -0,0 +1,271 @@
1
+ //
2
+ // Copyright 2023 DXOS.org
3
+ //
4
+
5
+ import '@dxos/lit-ui/dx-ref-tag.pcss';
6
+
7
+ import { syntaxTree } from '@codemirror/language';
8
+ import {
9
+ type EditorState,
10
+ type Extension,
11
+ type RangeSet,
12
+ RangeSetBuilder,
13
+ StateField,
14
+ type Transaction,
15
+ } from '@codemirror/state';
16
+ import { Decoration, type DecorationSet, EditorView, WidgetType } from '@codemirror/view';
17
+ import { type SyntaxNode } from '@lezer/common';
18
+
19
+ import { type RenderCallback } from '../../types';
20
+
21
+ export type PreviewLinkRef = {
22
+ suggest?: boolean;
23
+ block?: boolean;
24
+ label: string;
25
+ ref: string;
26
+ };
27
+
28
+ export type PreviewLinkTarget = {
29
+ label: string;
30
+ text?: string;
31
+ object?: any;
32
+ };
33
+
34
+ export type PreviewAction =
35
+ | {
36
+ type: 'insert';
37
+ link: PreviewLinkRef;
38
+ target: PreviewLinkTarget;
39
+ }
40
+ | {
41
+ type: 'delete';
42
+ link: PreviewLinkRef;
43
+ };
44
+
45
+ // TODO(burdon): Handle error.
46
+ export type PreviewLookup = (link: PreviewLinkRef) => Promise<PreviewLinkTarget | null | undefined>;
47
+
48
+ export type PreviewActionHandler = (action: PreviewAction) => void;
49
+
50
+ export type PreviewRenderProps = {
51
+ readonly: boolean;
52
+ link: PreviewLinkRef;
53
+ onAction: PreviewActionHandler;
54
+ onLookup?: PreviewLookup;
55
+ };
56
+
57
+ export type PreviewOptions = {
58
+ renderBlock?: RenderCallback<PreviewRenderProps>;
59
+ onLookup?: PreviewLookup;
60
+ };
61
+
62
+ /**
63
+ * Create preview decorations.
64
+ */
65
+ export const preview = (options: PreviewOptions = {}): Extension => {
66
+ return [
67
+ // NOTE: Atomic block decorations must be created from a state field, now a widget, otherwise it results in the following error:
68
+ // "Block decorations may not be specified via plugins"
69
+ StateField.define<DecorationSet>({
70
+ create: (state) => buildDecorations(state, options),
71
+ update: (_: RangeSet<Decoration>, tr: Transaction) => buildDecorations(tr.state, options),
72
+ provide: (field) => [
73
+ EditorView.decorations.from(field),
74
+ EditorView.atomicRanges.of((view) => view.state.field(field)),
75
+ ],
76
+ }),
77
+
78
+ EditorView.theme({
79
+ '.cm-preview-block': {
80
+ marginLeft: '-1rem',
81
+ marginRight: '-1rem',
82
+ padding: '1rem',
83
+ borderRadius: '0.5rem',
84
+ background: 'var(--dx-modalSurface)',
85
+ border: '1px solid var(--dx-separator)',
86
+ },
87
+ }),
88
+ ];
89
+ };
90
+
91
+ /**
92
+ * Link references.
93
+ *
94
+ * [Label][dxn:echo:123] Inline reference
95
+ * ![Label][dxn:echo:123] Block reference
96
+ * ![Label][?dxn:echo:123] Suggestion
97
+ */
98
+ const getLinkRef = (state: EditorState, node: SyntaxNode): PreviewLinkRef | undefined => {
99
+ const mark = node.getChild('LinkMark');
100
+ const label = node.getChild('LinkLabel');
101
+ if (mark && label) {
102
+ const ref = state.sliceDoc(label.from + 1, label.to - 1);
103
+ return {
104
+ suggest: ref.startsWith('?'),
105
+ block: state.sliceDoc(mark.from, mark.from + 1) === '!',
106
+ label: state.sliceDoc(mark.to, label.from - 1),
107
+ ref: ref.startsWith('?') ? ref.slice(1) : ref,
108
+ };
109
+ }
110
+ };
111
+
112
+ /**
113
+ * Echo references are represented as markdown reference links.
114
+ * https://www.markdownguide.org/basic-syntax/#reference-style-links
115
+ * [Label|block][dxn:echo:123]
116
+ * [Label|inline][dxn:echo:123]
117
+ */
118
+ const buildDecorations = (state: EditorState, options: PreviewOptions) => {
119
+ const builder = new RangeSetBuilder<Decoration>();
120
+
121
+ syntaxTree(state).iterate({
122
+ enter: (node) => {
123
+ switch (node.name) {
124
+ //
125
+ // Decoration.
126
+ // [Label][dxn:echo:123]
127
+ //
128
+ case 'Link': {
129
+ const link = getLinkRef(state, node.node);
130
+ if (link) {
131
+ builder.add(
132
+ node.from,
133
+ node.to,
134
+ Decoration.replace({
135
+ widget: new PreviewInlineWidget(options, link),
136
+ }),
137
+ );
138
+ }
139
+ break;
140
+ }
141
+ //
142
+ // Block widget.
143
+ // ![Label][dxn:echo:123]
144
+ //
145
+ case 'Image': {
146
+ const link = getLinkRef(state, node.node);
147
+ if (options.renderBlock && link) {
148
+ builder.add(
149
+ node.from,
150
+ node.to,
151
+ Decoration.replace({
152
+ block: true,
153
+ // atomic: true,
154
+ widget: new PreviewBlockWidget(options, link),
155
+ }),
156
+ );
157
+ }
158
+ break;
159
+ }
160
+ }
161
+ },
162
+ });
163
+
164
+ return builder.finish();
165
+ };
166
+
167
+ /**
168
+ * Inline widget.
169
+ * [Label][dxn:echo:123]
170
+ */
171
+ class PreviewInlineWidget extends WidgetType {
172
+ constructor(
173
+ readonly _options: PreviewOptions,
174
+ readonly _link: PreviewLinkRef,
175
+ ) {
176
+ super();
177
+ }
178
+
179
+ // override ignoreEvent() {
180
+ // return false;
181
+ // }
182
+
183
+ override eq(other: this): boolean {
184
+ return this._link.ref === other._link.ref && this._link.label === other._link.label;
185
+ }
186
+
187
+ override toDOM(view: EditorView): HTMLElement {
188
+ const root = document.createElement('dx-ref-tag');
189
+ root.textContent = this._link.label;
190
+ root.setAttribute('refId', this._link.ref);
191
+ return root;
192
+ }
193
+ }
194
+
195
+ /**
196
+ * Block widget.
197
+ * ![Label][dxn:echo:123]
198
+ */
199
+ class PreviewBlockWidget extends WidgetType {
200
+ constructor(
201
+ readonly _options: PreviewOptions,
202
+ readonly _link: PreviewLinkRef,
203
+ ) {
204
+ super();
205
+ }
206
+
207
+ // override ignoreEvent() {
208
+ // return true;
209
+ // }
210
+
211
+ override eq(other: this): boolean {
212
+ return this._link.ref === other._link.ref;
213
+ }
214
+
215
+ override toDOM(view: EditorView): HTMLDivElement {
216
+ const root = document.createElement('div');
217
+ root.classList.add('cm-preview-block');
218
+
219
+ // TODO(burdon): Inject handler.
220
+ const handleAction: PreviewActionHandler = (action) => {
221
+ const pos = view.posAtDOM(root);
222
+ const node = syntaxTree(view.state).resolve(pos + 1).node.parent;
223
+ if (!node) {
224
+ return;
225
+ }
226
+
227
+ const link = getLinkRef(view.state, node);
228
+ if (link?.ref !== action.link.ref) {
229
+ return;
230
+ }
231
+
232
+ switch (action.type) {
233
+ // TODO(burdon): Should we dispatch to the view or mutate the document? (i.e., handle externally?)
234
+ // Insert ref text.
235
+ case 'insert': {
236
+ view.dispatch({
237
+ changes: {
238
+ from: node.from,
239
+ to: node.to,
240
+ insert: action.target.text,
241
+ },
242
+ });
243
+ break;
244
+ }
245
+ // Remove ref.
246
+ case 'delete': {
247
+ view.dispatch({
248
+ changes: {
249
+ from: node.from,
250
+ to: node.to,
251
+ },
252
+ });
253
+ break;
254
+ }
255
+ }
256
+ };
257
+
258
+ this._options.renderBlock!(
259
+ root,
260
+ {
261
+ readonly: view.state.readOnly,
262
+ link: this._link,
263
+ onAction: handleAction,
264
+ onLookup: this._options.onLookup,
265
+ },
266
+ view,
267
+ );
268
+
269
+ return root;
270
+ }
271
+ }
@@ -2,5 +2,4 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- export * from './useActionHandler';
6
5
  export * from './useTextEditor';
@@ -2,7 +2,7 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import { EditorState, type EditorStateConfig } from '@codemirror/state';
5
+ import { EditorState, type EditorStateConfig, type Text } from '@codemirror/state';
6
6
  import { EditorView } from '@codemirror/view';
7
7
  import { useFocusableGroup, type TabsterTypes } from '@fluentui/react-tabster';
8
8
  import {
@@ -43,6 +43,7 @@ export type CursorInfo = {
43
43
 
44
44
  export type UseTextEditorProps = Pick<EditorStateConfig, 'extensions'> & {
45
45
  id?: string;
46
+ doc?: Text;
46
47
  initialValue?: string;
47
48
  className?: string;
48
49
  autoFocus?: boolean;
@@ -61,7 +62,7 @@ export const useTextEditor = (
61
62
  props: MaybeProvider<UseTextEditorProps> = {},
62
63
  deps: DependencyList = [],
63
64
  ): UseTextEditor => {
64
- const { id, initialValue, extensions, autoFocus, scrollTo, selection, moveToEndOfLine, debug } =
65
+ const { id, doc, initialValue, extensions, autoFocus, scrollTo, selection, moveToEndOfLine, debug } =
65
66
  useMemo<UseTextEditorProps>(() => getProviderValue(props), deps ?? []);
66
67
 
67
68
  // NOTE: Increments by 2 in strict mode.
@@ -87,7 +88,7 @@ export const useTextEditor = (
87
88
 
88
89
  // https://codemirror.net/docs/ref/#state.EditorStateConfig
89
90
  const state = EditorState.create({
90
- doc: initialValue,
91
+ doc: doc ?? initialValue,
91
92
  // selection: initialSelection,
92
93
  extensions: [
93
94
  id && documentId.of(id),