@dxos/react-ui-editor 0.8.4-main.c1de068 → 0.8.4-main.e098934

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 (231) hide show
  1. package/dist/lib/browser/index.mjs +2074 -996
  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 +71 -1
  5. package/dist/lib/browser/testing/index.mjs.map +4 -4
  6. package/dist/lib/node-esm/index.mjs +2074 -996
  7. package/dist/lib/node-esm/index.mjs.map +4 -4
  8. package/dist/lib/node-esm/meta.json +1 -1
  9. package/dist/lib/node-esm/testing/index.mjs +71 -1
  10. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  11. package/dist/types/src/components/{Popover → CommandMenu}/CommandMenu.d.ts +10 -6
  12. package/dist/types/src/components/CommandMenu/CommandMenu.d.ts.map +1 -0
  13. package/dist/types/src/components/CommandMenu/index.d.ts +2 -0
  14. package/dist/types/src/components/CommandMenu/index.d.ts.map +1 -0
  15. package/dist/types/src/components/Editor/Editor.d.ts.map +1 -1
  16. package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts.map +1 -1
  17. package/dist/types/src/components/EditorToolbar/blocks.d.ts.map +1 -1
  18. package/dist/types/src/components/EditorToolbar/formatting.d.ts.map +1 -1
  19. package/dist/types/src/components/EditorToolbar/headings.d.ts.map +1 -1
  20. package/dist/types/src/components/EditorToolbar/image.d.ts.map +1 -1
  21. package/dist/types/src/components/EditorToolbar/lists.d.ts.map +1 -1
  22. package/dist/types/src/components/EditorToolbar/search.d.ts.map +1 -1
  23. package/dist/types/src/components/EditorToolbar/util.d.ts +2 -2
  24. package/dist/types/src/components/EditorToolbar/util.d.ts.map +1 -1
  25. package/dist/types/src/components/EditorToolbar/view-mode.d.ts +1 -1
  26. package/dist/types/src/components/EditorToolbar/view-mode.d.ts.map +1 -1
  27. package/dist/types/src/components/index.d.ts +1 -1
  28. package/dist/types/src/components/index.d.ts.map +1 -1
  29. package/dist/types/src/defaults.d.ts.map +1 -1
  30. package/dist/types/src/extensions/autocomplete.d.ts +20 -7
  31. package/dist/types/src/extensions/autocomplete.d.ts.map +1 -1
  32. package/dist/types/src/extensions/automerge/automerge.d.ts.map +1 -1
  33. package/dist/types/src/extensions/automerge/automerge.stories.d.ts +9 -18
  34. package/dist/types/src/extensions/automerge/automerge.stories.d.ts.map +1 -1
  35. package/dist/types/src/extensions/automerge/defs.d.ts +1 -1
  36. package/dist/types/src/extensions/automerge/defs.d.ts.map +1 -1
  37. package/dist/types/src/extensions/automerge/sync.d.ts.map +1 -1
  38. package/dist/types/src/extensions/automerge/update-automerge.d.ts.map +1 -1
  39. package/dist/types/src/extensions/autoscroll.d.ts +10 -0
  40. package/dist/types/src/extensions/autoscroll.d.ts.map +1 -0
  41. package/dist/types/src/extensions/command/action.d.ts +1 -1
  42. package/dist/types/src/extensions/command/action.d.ts.map +1 -1
  43. package/dist/types/src/extensions/command/command-menu.d.ts +1 -1
  44. package/dist/types/src/extensions/command/command-menu.d.ts.map +1 -1
  45. package/dist/types/src/extensions/command/command.d.ts.map +1 -1
  46. package/dist/types/src/extensions/command/floating-menu.d.ts.map +1 -1
  47. package/dist/types/src/extensions/command/hint.d.ts +2 -7
  48. package/dist/types/src/extensions/command/hint.d.ts.map +1 -1
  49. package/dist/types/src/extensions/command/state.d.ts +1 -1
  50. package/dist/types/src/extensions/command/state.d.ts.map +1 -1
  51. package/dist/types/src/extensions/command/typeahead.d.ts.map +1 -1
  52. package/dist/types/src/extensions/command/useCommandMenu.d.ts +3 -4
  53. package/dist/types/src/extensions/command/useCommandMenu.d.ts.map +1 -1
  54. package/dist/types/src/extensions/comments.d.ts +1 -1
  55. package/dist/types/src/extensions/comments.d.ts.map +1 -1
  56. package/dist/types/src/extensions/dnd.d.ts.map +1 -1
  57. package/dist/types/src/extensions/factories.d.ts +2 -7
  58. package/dist/types/src/extensions/factories.d.ts.map +1 -1
  59. package/dist/types/src/extensions/index.d.ts +2 -0
  60. package/dist/types/src/extensions/index.d.ts.map +1 -1
  61. package/dist/types/src/extensions/markdown/action.d.ts.map +1 -1
  62. package/dist/types/src/extensions/markdown/bundle.d.ts +8 -2
  63. package/dist/types/src/extensions/markdown/bundle.d.ts.map +1 -1
  64. package/dist/types/src/extensions/markdown/changes.d.ts +1 -1
  65. package/dist/types/src/extensions/markdown/changes.d.ts.map +1 -1
  66. package/dist/types/src/extensions/markdown/decorate.d.ts +9 -1
  67. package/dist/types/src/extensions/markdown/decorate.d.ts.map +1 -1
  68. package/dist/types/src/extensions/markdown/formatting.d.ts +1 -1
  69. package/dist/types/src/extensions/markdown/formatting.d.ts.map +1 -1
  70. package/dist/types/src/extensions/markdown/formatting.test.d.ts.map +1 -1
  71. package/dist/types/src/extensions/markdown/highlight.d.ts.map +1 -1
  72. package/dist/types/src/extensions/markdown/image.d.ts.map +1 -1
  73. package/dist/types/src/extensions/markdown/link.d.ts.map +1 -1
  74. package/dist/types/src/extensions/outliner/outliner.d.ts +1 -1
  75. package/dist/types/src/extensions/outliner/outliner.d.ts.map +1 -1
  76. package/dist/types/src/extensions/outliner/selection.d.ts.map +1 -1
  77. package/dist/types/src/extensions/outliner/tree.d.ts +2 -2
  78. package/dist/types/src/extensions/outliner/tree.d.ts.map +1 -1
  79. package/dist/types/src/extensions/preview/preview.d.ts +3 -6
  80. package/dist/types/src/extensions/preview/preview.d.ts.map +1 -1
  81. package/dist/types/src/extensions/tags/extended-markdown.d.ts +10 -0
  82. package/dist/types/src/extensions/tags/extended-markdown.d.ts.map +1 -0
  83. package/dist/types/src/extensions/tags/extended-markdown.test.d.ts +2 -0
  84. package/dist/types/src/extensions/tags/extended-markdown.test.d.ts.map +1 -0
  85. package/dist/types/src/extensions/tags/index.d.ts +4 -0
  86. package/dist/types/src/extensions/tags/index.d.ts.map +1 -0
  87. package/dist/types/src/extensions/tags/streamer.d.ts +12 -0
  88. package/dist/types/src/extensions/tags/streamer.d.ts.map +1 -0
  89. package/dist/types/src/extensions/tags/xml-tags.d.ts +71 -0
  90. package/dist/types/src/extensions/tags/xml-tags.d.ts.map +1 -0
  91. package/dist/types/src/extensions/tags/xml-util.d.ts +10 -0
  92. package/dist/types/src/extensions/tags/xml-util.d.ts.map +1 -0
  93. package/dist/types/src/hooks/useTextEditor.d.ts.map +1 -1
  94. package/dist/types/src/stories/Command.stories.d.ts +12 -4
  95. package/dist/types/src/stories/Command.stories.d.ts.map +1 -1
  96. package/dist/types/src/stories/CommandMenu.stories.d.ts +10 -3
  97. package/dist/types/src/stories/CommandMenu.stories.d.ts.map +1 -1
  98. package/dist/types/src/stories/Comments.stories.d.ts +21 -9
  99. package/dist/types/src/stories/Comments.stories.d.ts.map +1 -1
  100. package/dist/types/src/stories/EditorToolbar.stories.d.ts +39 -2
  101. package/dist/types/src/stories/EditorToolbar.stories.d.ts.map +1 -1
  102. package/dist/types/src/stories/Experimental.stories.d.ts +22 -12
  103. package/dist/types/src/stories/Experimental.stories.d.ts.map +1 -1
  104. package/dist/types/src/stories/Markdown.stories.d.ts +32 -42
  105. package/dist/types/src/stories/Markdown.stories.d.ts.map +1 -1
  106. package/dist/types/src/stories/Outliner.stories.d.ts +15 -20
  107. package/dist/types/src/stories/Outliner.stories.d.ts.map +1 -1
  108. package/dist/types/src/stories/Preview.stories.d.ts +21 -6
  109. package/dist/types/src/stories/Preview.stories.d.ts.map +1 -1
  110. package/dist/types/src/stories/Tags.stories.d.ts +17 -0
  111. package/dist/types/src/stories/Tags.stories.d.ts.map +1 -0
  112. package/dist/types/src/stories/TextEditor.stories.d.ts +38 -51
  113. package/dist/types/src/stories/TextEditor.stories.d.ts.map +1 -1
  114. package/dist/types/src/stories/components/EditorStory.d.ts +3 -6
  115. package/dist/types/src/stories/components/EditorStory.d.ts.map +1 -1
  116. package/dist/types/src/styles/theme.d.ts.map +1 -1
  117. package/dist/types/src/testing/PreviewPopover.d.ts +20 -0
  118. package/dist/types/src/testing/PreviewPopover.d.ts.map +1 -0
  119. package/dist/types/src/testing/index.d.ts +1 -0
  120. package/dist/types/src/testing/index.d.ts.map +1 -1
  121. package/dist/types/src/testing/util.d.ts +1 -0
  122. package/dist/types/src/testing/util.d.ts.map +1 -1
  123. package/dist/types/src/translations.d.ts +1 -1
  124. package/dist/types/src/types/types.d.ts +1 -1
  125. package/dist/types/src/util/cursor.d.ts.map +1 -1
  126. package/dist/types/src/util/debug.d.ts +1 -1
  127. package/dist/types/src/util/debug.d.ts.map +1 -1
  128. package/dist/types/src/util/decorations.d.ts +4 -0
  129. package/dist/types/src/util/decorations.d.ts.map +1 -0
  130. package/dist/types/src/util/dom.d.ts +2 -12
  131. package/dist/types/src/util/dom.d.ts.map +1 -1
  132. package/dist/types/src/util/domino.d.ts +18 -0
  133. package/dist/types/src/util/domino.d.ts.map +1 -0
  134. package/dist/types/src/util/index.d.ts +2 -0
  135. package/dist/types/src/util/index.d.ts.map +1 -1
  136. package/dist/types/src/util/react.d.ts +1 -1
  137. package/dist/types/src/util/react.d.ts.map +1 -1
  138. package/dist/types/tsconfig.tsbuildinfo +1 -1
  139. package/package.json +57 -51
  140. package/src/components/{Popover → CommandMenu}/CommandMenu.tsx +93 -26
  141. package/src/components/{Popover → CommandMenu}/index.ts +0 -2
  142. package/src/components/Editor/Editor.tsx +1 -1
  143. package/src/components/EditorToolbar/EditorToolbar.tsx +40 -30
  144. package/src/components/EditorToolbar/blocks.ts +21 -24
  145. package/src/components/EditorToolbar/formatting.ts +22 -25
  146. package/src/components/EditorToolbar/headings.ts +10 -5
  147. package/src/components/EditorToolbar/image.ts +8 -4
  148. package/src/components/EditorToolbar/lists.ts +16 -19
  149. package/src/components/EditorToolbar/search.ts +8 -4
  150. package/src/components/EditorToolbar/util.ts +16 -5
  151. package/src/components/EditorToolbar/view-mode.ts +11 -6
  152. package/src/components/index.ts +1 -1
  153. package/src/defaults.ts +5 -2
  154. package/src/extensions/autocomplete.ts +204 -54
  155. package/src/extensions/automerge/automerge.stories.tsx +25 -18
  156. package/src/extensions/automerge/automerge.ts +4 -3
  157. package/src/extensions/automerge/defs.ts +1 -1
  158. package/src/extensions/automerge/sync.ts +1 -1
  159. package/src/extensions/automerge/update-automerge.ts +1 -1
  160. package/src/extensions/autoscroll.ts +157 -0
  161. package/src/extensions/awareness/awareness.ts +2 -2
  162. package/src/extensions/command/action.ts +1 -2
  163. package/src/extensions/command/command-menu.ts +7 -6
  164. package/src/extensions/command/command.ts +3 -3
  165. package/src/extensions/command/floating-menu.ts +10 -15
  166. package/src/extensions/command/hint.ts +2 -1
  167. package/src/extensions/command/placeholder.ts +1 -1
  168. package/src/extensions/command/state.ts +4 -3
  169. package/src/extensions/command/typeahead.ts +2 -2
  170. package/src/extensions/command/useCommandMenu.ts +6 -9
  171. package/src/extensions/comments.ts +18 -13
  172. package/src/extensions/dnd.ts +1 -1
  173. package/src/extensions/factories.ts +9 -21
  174. package/src/extensions/folding.tsx +2 -2
  175. package/src/extensions/index.ts +2 -0
  176. package/src/extensions/markdown/action.ts +2 -1
  177. package/src/extensions/markdown/bundle.ts +25 -3
  178. package/src/extensions/markdown/changes.ts +1 -1
  179. package/src/extensions/markdown/decorate.ts +23 -14
  180. package/src/extensions/markdown/formatting.test.ts +6 -6
  181. package/src/extensions/markdown/formatting.ts +3 -3
  182. package/src/extensions/markdown/highlight.ts +1 -1
  183. package/src/extensions/markdown/image.ts +3 -4
  184. package/src/extensions/markdown/link.ts +3 -0
  185. package/src/extensions/markdown/table.ts +7 -1
  186. package/src/extensions/mention.ts +1 -1
  187. package/src/extensions/outliner/outliner.test.ts +3 -2
  188. package/src/extensions/outliner/outliner.ts +6 -5
  189. package/src/extensions/outliner/selection.ts +1 -1
  190. package/src/extensions/outliner/tree.test.ts +2 -1
  191. package/src/extensions/outliner/tree.ts +2 -2
  192. package/src/extensions/preview/preview.ts +59 -62
  193. package/src/extensions/tags/extended-markdown.test.ts +261 -0
  194. package/src/extensions/tags/extended-markdown.ts +78 -0
  195. package/src/extensions/tags/index.ts +7 -0
  196. package/src/extensions/tags/streamer.ts +244 -0
  197. package/src/extensions/tags/xml-tags.ts +335 -0
  198. package/src/extensions/tags/xml-util.ts +94 -0
  199. package/src/hooks/useTextEditor.ts +3 -15
  200. package/src/stories/Command.stories.tsx +24 -31
  201. package/src/stories/CommandMenu.stories.tsx +28 -29
  202. package/src/stories/Comments.stories.tsx +10 -6
  203. package/src/stories/EditorToolbar.stories.tsx +8 -8
  204. package/src/stories/Experimental.stories.tsx +12 -8
  205. package/src/stories/Markdown.stories.tsx +21 -17
  206. package/src/stories/Outliner.stories.tsx +42 -30
  207. package/src/stories/Preview.stories.tsx +30 -29
  208. package/src/stories/Tags.stories.tsx +81 -0
  209. package/src/stories/TextEditor.stories.tsx +40 -34
  210. package/src/stories/components/EditorStory.tsx +9 -10
  211. package/src/styles/theme.ts +8 -6
  212. package/src/testing/PreviewPopover.tsx +78 -0
  213. package/src/testing/index.ts +1 -0
  214. package/src/testing/util.ts +2 -0
  215. package/src/translations.ts +1 -1
  216. package/src/util/cursor.ts +2 -1
  217. package/src/util/debug.ts +2 -2
  218. package/src/util/decorations.ts +21 -0
  219. package/src/util/dom.ts +5 -27
  220. package/src/util/domino.ts +51 -0
  221. package/src/util/index.ts +2 -0
  222. package/src/util/react.tsx +1 -1
  223. package/dist/types/src/components/Popover/CommandMenu.d.ts.map +0 -1
  224. package/dist/types/src/components/Popover/RefDropdownMenu.d.ts +0 -21
  225. package/dist/types/src/components/Popover/RefDropdownMenu.d.ts.map +0 -1
  226. package/dist/types/src/components/Popover/RefPopover.d.ts +0 -34
  227. package/dist/types/src/components/Popover/RefPopover.d.ts.map +0 -1
  228. package/dist/types/src/components/Popover/index.d.ts +0 -4
  229. package/dist/types/src/components/Popover/index.d.ts.map +0 -1
  230. package/src/components/Popover/RefDropdownMenu.tsx +0 -85
  231. package/src/components/Popover/RefPopover.tsx +0 -99
@@ -0,0 +1,261 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { syntaxTree } from '@codemirror/language';
6
+ import { EditorState } from '@codemirror/state';
7
+ import { type SyntaxNode } from '@lezer/common';
8
+ import { describe, test } from 'vitest';
9
+
10
+ import { trim } from '@dxos/util';
11
+
12
+ import { extendedMarkdown } from './extended-markdown';
13
+ import { nodeToJson } from './xml-util';
14
+
15
+ describe('extended-markdown', () => {
16
+ const createEditorState = (doc: string) => {
17
+ return EditorState.create({
18
+ doc,
19
+ extensions: [extendedMarkdown()],
20
+ });
21
+ };
22
+
23
+ test('tree', async ({ expect }) => {
24
+ const doc = trim`
25
+ <prompt>
26
+ Hello
27
+ </prompt>
28
+
29
+ Hi there!
30
+
31
+ What can I do for you?
32
+
33
+ <suggestion>Summarize tools</suggestion>
34
+
35
+ <choice>
36
+ <option>Summarize tools</option>
37
+ <option>Retry</option>
38
+ </choice>
39
+
40
+ <toolkit />
41
+ `;
42
+
43
+ const nodes: SyntaxNode[] = [];
44
+ const state = createEditorState(doc);
45
+ const tree = syntaxTree(state);
46
+ tree.iterate({
47
+ enter: (node) => {
48
+ if (node.type.name === 'Element') {
49
+ nodes.push(node.node);
50
+ return false; // Stop traversal.
51
+ }
52
+ },
53
+ });
54
+
55
+ expect(nodes).toHaveLength(4);
56
+ expect(
57
+ nodes.map((node) => ({
58
+ content: doc.slice(node.from, node.to),
59
+ data: nodeToJson(state, node.node),
60
+ })),
61
+ ).toEqual([
62
+ {
63
+ content: '<prompt>\n Hello\n</prompt>',
64
+ data: {
65
+ _tag: 'prompt',
66
+ children: ['Hello'],
67
+ },
68
+ },
69
+ {
70
+ content: '<suggestion>Summarize tools</suggestion>',
71
+ data: {
72
+ _tag: 'suggestion',
73
+ children: ['Summarize tools'],
74
+ },
75
+ },
76
+ {
77
+ content: '<choice>\n <option>Summarize tools</option>\n <option>Retry</option>\n</choice>',
78
+ data: {
79
+ _tag: 'choice',
80
+ children: [
81
+ {
82
+ _tag: 'option',
83
+ children: ['Summarize tools'],
84
+ },
85
+ {
86
+ _tag: 'option',
87
+ children: ['Retry'],
88
+ },
89
+ ],
90
+ },
91
+ },
92
+ {
93
+ content: '<toolkit />',
94
+ data: {
95
+ _tag: 'toolkit',
96
+ },
97
+ },
98
+ ]);
99
+ });
100
+
101
+ test('setext heading disabled', () => {
102
+ const doc = trim`
103
+ This should NOT be a heading
104
+ =
105
+
106
+ Another line that should NOT be a heading
107
+ -
108
+ `;
109
+
110
+ const state = createEditorState(doc);
111
+ const tree = syntaxTree(state);
112
+
113
+ tree.iterate({
114
+ enter: (node) => {
115
+ if (node.type.name === 'SetextHeading') {
116
+ throw new Error('SetextHeading should be disabled!');
117
+ }
118
+ },
119
+ });
120
+ });
121
+
122
+ //
123
+ // TODO(burdon): All tests below should test the tree.
124
+ //
125
+
126
+ test('should parse standard markdown elements', ({ expect }) => {
127
+ const doc = trim`
128
+ # Heading 1
129
+ ## Heading 2
130
+ ### Heading 3
131
+
132
+ This is a paragraph with **bold** and *italic* text.
133
+
134
+ \`\`\`javascript
135
+ const code = 'block';
136
+ \`\`\`
137
+ `;
138
+
139
+ const state = createEditorState(doc);
140
+ const tree = state.sliceDoc(0, state.doc.length);
141
+
142
+ // Verify the document is parsed without errors.
143
+ expect(tree).toBe(doc);
144
+ expect(state.doc.lines).toBeGreaterThan(1);
145
+ });
146
+
147
+ test('should parse custom XML tags', ({ expect }) => {
148
+ const doc = trim`
149
+ # Document with Custom Tags
150
+
151
+ <prompt>This is a custom prompt block</prompt>
152
+
153
+ Regular paragraph text.
154
+
155
+ <prompt />
156
+
157
+ Regular paragraph text.
158
+
159
+ <prompt>
160
+ Multi-line
161
+ prompt content
162
+ </prompt>
163
+ `;
164
+
165
+ const state = createEditorState(doc);
166
+ const tree = state.sliceDoc(0, state.doc.length);
167
+
168
+ // Verify the document contains custom tags.
169
+ expect(tree).toContain('<prompt>');
170
+ expect(tree).toContain('<prompt />');
171
+ expect(tree).toContain('</prompt>');
172
+ });
173
+
174
+ test('should handle mixed markdown and XML content', ({ expect }) => {
175
+ const doc = trim`
176
+ # Mixed Content
177
+
178
+ This is a paragraph with a <prompt>inline prompt</prompt> tag.
179
+
180
+ ## Section 2
181
+
182
+ <prompt>
183
+ This prompt contains **markdown** formatting
184
+ </prompt>
185
+
186
+ \`\`\`
187
+ <prompt>This should not be parsed as XML inside code block</prompt>
188
+ \`\`\`
189
+ `;
190
+
191
+ const state = createEditorState(doc);
192
+ const tree = state.sliceDoc(0, state.doc.length);
193
+
194
+ // Verify mixed content is preserved.
195
+ expect(tree).toBe(doc);
196
+ });
197
+
198
+ test('should not parse XML tags in code blocks', ({ expect }) => {
199
+ const doc = trim`
200
+ \`\`\`xml
201
+ <prompt>This is inside a code block</prompt>
202
+ \`\`\`
203
+
204
+ \`<prompt>inline code</prompt>\`
205
+ `;
206
+
207
+ const state = createEditorState(doc);
208
+ const tree = state.sliceDoc(0, state.doc.length);
209
+
210
+ // Verify code blocks are preserved as-is.
211
+ expect(tree).toContain('```xml');
212
+ expect(tree).toContain('<prompt>This is inside a code block</prompt>');
213
+ });
214
+
215
+ test('should handle nested and complex structures', ({ expect }) => {
216
+ const doc = trim`
217
+ # Complex Document
218
+
219
+ <prompt>
220
+ First prompt with some content
221
+ </prompt>
222
+
223
+ ## Lists and Prompts
224
+
225
+ - Item 1
226
+ - Item 2 with <prompt>embedded prompt</prompt>
227
+ - Item 3
228
+
229
+ <prompt>
230
+ Another prompt after the list
231
+ </prompt>
232
+
233
+ ### Code Example
234
+
235
+ \`\`\`typescript
236
+ const example = '<prompt>not parsed</prompt>';
237
+ \`\`\`
238
+ `;
239
+
240
+ const state = createEditorState(doc);
241
+ const tree = state.sliceDoc(0, state.doc.length);
242
+
243
+ // Verify complex structures are handled.
244
+ expect(tree).toBe(doc);
245
+ expect(tree).toMatch(/Item 2 with <prompt>embedded prompt<\/prompt>/);
246
+ });
247
+
248
+ test('should handle empty and edge cases', ({ expect }) => {
249
+ const emptyDoc = '';
250
+ const emptyState = createEditorState(emptyDoc);
251
+ expect(emptyState.doc.length).toBe(0);
252
+
253
+ const onlyPromptDoc = '<prompt>Only prompt content</prompt>';
254
+ const onlyPromptState = createEditorState(onlyPromptDoc);
255
+ expect(onlyPromptState.sliceDoc(0)).toBe(onlyPromptDoc);
256
+
257
+ const unclosedPromptDoc = '<prompt>Unclosed prompt';
258
+ const unclosedState = createEditorState(unclosedPromptDoc);
259
+ expect(unclosedState.sliceDoc(0)).toBe(unclosedPromptDoc);
260
+ });
261
+ });
@@ -0,0 +1,78 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { xmlLanguage } from '@codemirror/lang-xml';
6
+ import { type Extension } from '@codemirror/state';
7
+ import { type ParseWrapper, parseMixed } from '@lezer/common';
8
+
9
+ import { createMarkdownExtensions } from '../markdown';
10
+
11
+ import { type XmlWidgetRegistry } from './xml-tags';
12
+
13
+ export type ExtendedMarkdownOptions = {
14
+ registry?: XmlWidgetRegistry;
15
+ };
16
+
17
+ /**
18
+ * Extended markdown parser with mixed parser for custom rendering of XML tags.
19
+ */
20
+ export const extendedMarkdown = ({ registry }: ExtendedMarkdownOptions = {}): Extension => {
21
+ return [
22
+ createMarkdownExtensions({
23
+ extensions: [
24
+ {
25
+ wrap: mixedParser(registry),
26
+ parseBlock: [
27
+ // Disable SetextHeading since it causes flickering when parsing/rendering tasks in chunks.
28
+ {
29
+ name: 'SetextHeading',
30
+ parse: () => false,
31
+ },
32
+ ],
33
+ },
34
+ ],
35
+ }),
36
+ ];
37
+ };
38
+
39
+ /**
40
+ * Configure mixed parser to recognize custom tags.
41
+ */
42
+ const mixedParser = (registry?: XmlWidgetRegistry): ParseWrapper => {
43
+ const customTags = Object.keys(registry ?? {});
44
+ const tagPattern = new RegExp(`<(${customTags.join('|')})`);
45
+
46
+ return parseMixed((node, input) => {
47
+ switch (node.name) {
48
+ // Ignore XML inside of fenced and inline code.
49
+ case 'FencedCode':
50
+ case 'InlineCode': {
51
+ return null;
52
+ }
53
+
54
+ // Parse multi-line HTML blocks.
55
+ // case 'XMLBlock':
56
+ case 'HTMLBlock': {
57
+ return {
58
+ parser: xmlLanguage.parser,
59
+ };
60
+ }
61
+
62
+ // Parse paragraphs that contain custom XML tags.
63
+ // TODO(burdon): Entire paragraph should be parsed as XML.
64
+ case 'Paragraph': {
65
+ const content = input.read(node.from, node.to);
66
+ if (tagPattern.test(content)) {
67
+ return {
68
+ parser: xmlLanguage.parser,
69
+ };
70
+ }
71
+
72
+ return null;
73
+ }
74
+ }
75
+
76
+ return null;
77
+ });
78
+ };
@@ -0,0 +1,7 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './extended-markdown';
6
+ export * from './streamer';
7
+ export * from './xml-tags';
@@ -0,0 +1,244 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { type Extension, StateEffect, StateField } from '@codemirror/state';
6
+ import { Decoration, type DecorationSet, EditorView, ViewPlugin, type ViewUpdate, WidgetType } from '@codemirror/view';
7
+
8
+ import { isNotFalsy } from '@dxos/util';
9
+
10
+ import { Domino } from '../../util';
11
+
12
+ const BLINK_RATE = 2_000;
13
+
14
+ export type StreamerOptions = {
15
+ cursor?: boolean;
16
+ // When true, uses defaults. When object, allows configuring removal delay.
17
+ fadeIn?: boolean | { removalDelay?: number };
18
+ };
19
+
20
+ /**
21
+ * Extension that adds a blinking cursor widget at the end of the document.
22
+ */
23
+ export const streamer = (options: StreamerOptions = {}): Extension => {
24
+ return [
25
+ options.cursor && cursor(),
26
+ options.fadeIn && fadeIn(typeof options.fadeIn === 'object' ? options.fadeIn : {}),
27
+ ].filter(isNotFalsy);
28
+ };
29
+
30
+ /**
31
+ * State field to manage the cursor widget decoration.
32
+ */
33
+ const cursor = (): Extension => {
34
+ const hideCursor = StateEffect.define();
35
+
36
+ // State field to track if cursor should be shown.
37
+ const showCursor = StateField.define<boolean>({
38
+ create: () => true,
39
+ update: (value, tr) => {
40
+ for (const effect of tr.effects) {
41
+ if (effect.is(hideCursor)) {
42
+ return false;
43
+ }
44
+ }
45
+ if (tr.docChanged) {
46
+ return true;
47
+ }
48
+
49
+ return value;
50
+ },
51
+ });
52
+
53
+ // View plugin to manage timer and dispatch effects.
54
+ const timerPlugin = ViewPlugin.fromClass(
55
+ class {
56
+ timer: any;
57
+
58
+ constructor(private view: EditorView) {}
59
+
60
+ update(update: ViewUpdate) {
61
+ if (update.docChanged) {
62
+ clearTimeout(this.timer);
63
+ this.timer = setTimeout(() => {
64
+ this.view.dispatch({
65
+ effects: hideCursor.of(null),
66
+ });
67
+ }, BLINK_RATE);
68
+ }
69
+ }
70
+
71
+ destroy() {
72
+ clearTimeout(this.timer);
73
+ }
74
+ },
75
+ );
76
+
77
+ // Decoration field that uses the showCursor state.
78
+ const cursorDecoration = StateField.define<DecorationSet>({
79
+ create: () => Decoration.none,
80
+ update: (_decorations, tr) => {
81
+ const show = tr.state.field(showCursor);
82
+ if (!show) {
83
+ return Decoration.none;
84
+ }
85
+
86
+ // Always place cursor at the end of the document.
87
+ const endPos = tr.state.doc.length;
88
+ return Decoration.set([
89
+ Decoration.widget({
90
+ widget: new CursorWidget(),
91
+ side: 1, // Place after the position.
92
+ }).range(endPos),
93
+ ]);
94
+ },
95
+ provide: (f) => EditorView.decorations.from(f),
96
+ });
97
+
98
+ return [showCursor, timerPlugin, cursorDecoration];
99
+ };
100
+
101
+ /**
102
+ * Widget class for the cursor at the end of the document.
103
+ * Half
104
+ */
105
+ class CursorWidget extends WidgetType {
106
+ toDOM() {
107
+ return Domino.of('span')
108
+ .style({ opacity: '0.8' })
109
+ .child(Domino.of('span').text('\u258F').style({ animation: 'blink 2s infinite' }))
110
+ .build();
111
+ }
112
+ }
113
+
114
+ /**
115
+ * State field to detect and decorate appended text with a fade-in effect.
116
+ * Also schedules removal of the last appended decoration after a delay.
117
+ */
118
+ const fadeIn = (options: { removalDelay?: number } = {}): Extension => {
119
+ const FADE_IN_DURATION = 1_000; // ms.
120
+ const DEFAULT_REMOVAL_DELAY = 5_000; // ms.
121
+ const removalDelay = options.removalDelay ?? DEFAULT_REMOVAL_DELAY;
122
+
123
+ // Effect to remove a specific decoration by range.
124
+ const removeDecoration = StateEffect.define<{ from: number; to: number }>();
125
+
126
+ // Decoration field that adds fade-in marks for appended content and responds to removal effects.
127
+ const fadeField = StateField.define<DecorationSet>({
128
+ create: () => Decoration.none,
129
+ update: (decorations, tr) => {
130
+ let next = decorations;
131
+
132
+ // Apply removals first, if any.
133
+ for (const effect of tr.effects) {
134
+ if (effect.is(removeDecoration)) {
135
+ const target = effect.value;
136
+ next = next.update({
137
+ filter: (from, to) => !(from === target.from && to === target.to),
138
+ });
139
+ }
140
+ }
141
+
142
+ if (!tr.docChanged) {
143
+ return next;
144
+ }
145
+
146
+ // Reset decorations if the entire content was replaced.
147
+ let isReset = tr.state.doc.length === 0;
148
+ if (!isReset) {
149
+ tr.changes.iterChanges((fromA, toA) => {
150
+ if (fromA === 0 && toA === tr.startState.doc.length) {
151
+ isReset = true;
152
+ }
153
+ });
154
+ }
155
+ if (isReset) {
156
+ return Decoration.none;
157
+ }
158
+
159
+ // Add fade-in decorations for appended content at the end only.
160
+ const add: any[] = [];
161
+ tr.changes.iterChanges((fromA, toA, fromB, toB, inserted) => {
162
+ // Don't fade in initial content.
163
+ if (fromB === 0 && toB === inserted.length) {
164
+ return;
165
+ }
166
+ // At-the-end append.
167
+ if (toA === tr.startState.doc.length && inserted.length > 0) {
168
+ add.push(Decoration.mark({ class: 'cm-fade-in' }).range(fromB, toB));
169
+ }
170
+ });
171
+
172
+ return next.update({ add });
173
+ },
174
+ provide: (f) => EditorView.decorations.from(f),
175
+ });
176
+
177
+ // View plugin that tracks appended ranges and schedules their removal.
178
+ const timerPlugin = ViewPlugin.fromClass(
179
+ class {
180
+ // Map a simple key "from-to" to timer id.
181
+ _timers = new Map<string, any>();
182
+
183
+ constructor(private view: EditorView) {}
184
+
185
+ update(update: ViewUpdate) {
186
+ if (!update.docChanged) {
187
+ return;
188
+ }
189
+
190
+ update.changes.iterChanges((fromA, toA, fromB, toB, inserted) => {
191
+ // Only consider appends at the end.
192
+ if (toA !== update.startState.doc.length || inserted.length === 0) {
193
+ return;
194
+ }
195
+
196
+ const key = `${fromB}-${toB}`;
197
+ // Clear any prior timer for this exact range.
198
+ if (this._timers.has(key)) {
199
+ clearTimeout(this._timers.get(key));
200
+ }
201
+
202
+ const totalDelay = FADE_IN_DURATION + removalDelay;
203
+ const id = setTimeout(() => {
204
+ this.view.dispatch({ effects: removeDecoration.of({ from: fromB, to: toB }) });
205
+ this._timers.delete(key);
206
+ }, totalDelay);
207
+
208
+ this._timers.set(key, id);
209
+ });
210
+ }
211
+
212
+ destroy() {
213
+ for (const id of this._timers.values()) {
214
+ clearTimeout(id);
215
+ }
216
+ this._timers.clear();
217
+ }
218
+ },
219
+ );
220
+
221
+ return [
222
+ fadeField,
223
+ timerPlugin,
224
+ EditorView.theme({
225
+ '.cm-line > span': {
226
+ opacity: '0.8',
227
+ },
228
+ '.cm-fade-in': {
229
+ animation: 'fade-in 3s ease-out forwards',
230
+ },
231
+ '@keyframes fade-in': {
232
+ '0%': {
233
+ opacity: '0',
234
+ },
235
+ '80%': {
236
+ opacity: '1',
237
+ },
238
+ '100%': {
239
+ opacity: '0.8',
240
+ },
241
+ },
242
+ }),
243
+ ];
244
+ };