@dxos/react-ui-editor 0.8.3 → 0.8.4-main.1f223c7

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 (269) hide show
  1. package/dist/lib/browser/chunk-22UMM3QJ.mjs +22 -0
  2. package/dist/lib/browser/chunk-22UMM3QJ.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +2498 -1380
  4. package/dist/lib/browser/index.mjs.map +4 -4
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/testing/index.mjs +72 -2
  7. package/dist/lib/browser/testing/index.mjs.map +4 -4
  8. package/dist/lib/browser/types/index.mjs +13 -0
  9. package/dist/lib/browser/types/index.mjs.map +7 -0
  10. package/dist/lib/node-esm/chunk-YXYQPV6R.mjs +24 -0
  11. package/dist/lib/node-esm/chunk-YXYQPV6R.mjs.map +7 -0
  12. package/dist/lib/node-esm/index.mjs +2500 -1383
  13. package/dist/lib/node-esm/index.mjs.map +4 -4
  14. package/dist/lib/node-esm/meta.json +1 -1
  15. package/dist/lib/node-esm/testing/index.mjs +72 -2
  16. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  17. package/dist/lib/node-esm/types/index.mjs +14 -0
  18. package/dist/lib/node-esm/types/index.mjs.map +7 -0
  19. package/dist/types/src/components/{Popover → CommandMenu}/CommandMenu.d.ts +10 -6
  20. package/dist/types/src/components/CommandMenu/CommandMenu.d.ts.map +1 -0
  21. package/dist/types/src/components/CommandMenu/index.d.ts +2 -0
  22. package/dist/types/src/components/CommandMenu/index.d.ts.map +1 -0
  23. package/dist/types/src/components/Editor/Editor.d.ts +19 -0
  24. package/dist/types/src/components/Editor/Editor.d.ts.map +1 -0
  25. package/dist/types/src/components/Editor/index.d.ts +2 -0
  26. package/dist/types/src/components/Editor/index.d.ts.map +1 -0
  27. package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts.map +1 -1
  28. package/dist/types/src/components/EditorToolbar/blocks.d.ts.map +1 -1
  29. package/dist/types/src/components/EditorToolbar/formatting.d.ts.map +1 -1
  30. package/dist/types/src/components/EditorToolbar/headings.d.ts.map +1 -1
  31. package/dist/types/src/components/EditorToolbar/image.d.ts.map +1 -1
  32. package/dist/types/src/components/EditorToolbar/lists.d.ts.map +1 -1
  33. package/dist/types/src/components/EditorToolbar/search.d.ts.map +1 -1
  34. package/dist/types/src/components/EditorToolbar/util.d.ts +6 -5
  35. package/dist/types/src/components/EditorToolbar/util.d.ts.map +1 -1
  36. package/dist/types/src/components/EditorToolbar/view-mode.d.ts +1 -1
  37. package/dist/types/src/components/EditorToolbar/view-mode.d.ts.map +1 -1
  38. package/dist/types/src/components/index.d.ts +2 -1
  39. package/dist/types/src/components/index.d.ts.map +1 -1
  40. package/dist/types/src/defaults.d.ts.map +1 -1
  41. package/dist/types/src/extensions/autocomplete.d.ts +20 -7
  42. package/dist/types/src/extensions/autocomplete.d.ts.map +1 -1
  43. package/dist/types/src/extensions/automerge/automerge.d.ts.map +1 -1
  44. package/dist/types/src/extensions/automerge/automerge.stories.d.ts +36 -45
  45. package/dist/types/src/extensions/automerge/automerge.stories.d.ts.map +1 -1
  46. package/dist/types/src/extensions/automerge/defs.d.ts +1 -1
  47. package/dist/types/src/extensions/automerge/defs.d.ts.map +1 -1
  48. package/dist/types/src/extensions/automerge/sync.d.ts.map +1 -1
  49. package/dist/types/src/extensions/automerge/update-automerge.d.ts.map +1 -1
  50. package/dist/types/src/extensions/autoscroll.d.ts +10 -0
  51. package/dist/types/src/extensions/autoscroll.d.ts.map +1 -0
  52. package/dist/types/src/extensions/blast.d.ts.map +1 -1
  53. package/dist/types/src/extensions/command/action.d.ts +1 -1
  54. package/dist/types/src/extensions/command/action.d.ts.map +1 -1
  55. package/dist/types/src/extensions/command/command-menu.d.ts +1 -1
  56. package/dist/types/src/extensions/command/command-menu.d.ts.map +1 -1
  57. package/dist/types/src/extensions/command/command.d.ts.map +1 -1
  58. package/dist/types/src/extensions/command/floating-menu.d.ts.map +1 -1
  59. package/dist/types/src/extensions/command/hint.d.ts +2 -7
  60. package/dist/types/src/extensions/command/hint.d.ts.map +1 -1
  61. package/dist/types/src/extensions/command/index.d.ts +1 -1
  62. package/dist/types/src/extensions/command/index.d.ts.map +1 -1
  63. package/dist/types/src/extensions/command/state.d.ts +1 -1
  64. package/dist/types/src/extensions/command/state.d.ts.map +1 -1
  65. package/dist/types/src/extensions/command/typeahead.d.ts +7 -2
  66. package/dist/types/src/extensions/command/typeahead.d.ts.map +1 -1
  67. package/dist/types/src/extensions/command/useCommandMenu.d.ts +3 -4
  68. package/dist/types/src/extensions/command/useCommandMenu.d.ts.map +1 -1
  69. package/dist/types/src/extensions/comments.d.ts +1 -1
  70. package/dist/types/src/extensions/comments.d.ts.map +1 -1
  71. package/dist/types/src/extensions/dnd.d.ts.map +1 -1
  72. package/dist/types/src/extensions/factories.d.ts +15 -1
  73. package/dist/types/src/extensions/factories.d.ts.map +1 -1
  74. package/dist/types/src/extensions/index.d.ts +2 -0
  75. package/dist/types/src/extensions/index.d.ts.map +1 -1
  76. package/dist/types/src/extensions/markdown/action.d.ts.map +1 -1
  77. package/dist/types/src/extensions/markdown/bundle.d.ts +8 -2
  78. package/dist/types/src/extensions/markdown/bundle.d.ts.map +1 -1
  79. package/dist/types/src/extensions/markdown/changes.d.ts +1 -1
  80. package/dist/types/src/extensions/markdown/changes.d.ts.map +1 -1
  81. package/dist/types/src/extensions/markdown/decorate.d.ts +9 -1
  82. package/dist/types/src/extensions/markdown/decorate.d.ts.map +1 -1
  83. package/dist/types/src/extensions/markdown/formatting.d.ts +1 -1
  84. package/dist/types/src/extensions/markdown/formatting.d.ts.map +1 -1
  85. package/dist/types/src/extensions/markdown/formatting.test.d.ts.map +1 -1
  86. package/dist/types/src/extensions/markdown/highlight.d.ts.map +1 -1
  87. package/dist/types/src/extensions/markdown/image.d.ts.map +1 -1
  88. package/dist/types/src/extensions/markdown/link.d.ts.map +1 -1
  89. package/dist/types/src/extensions/modes.d.ts +0 -7
  90. package/dist/types/src/extensions/modes.d.ts.map +1 -1
  91. package/dist/types/src/extensions/outliner/outliner.d.ts +1 -1
  92. package/dist/types/src/extensions/outliner/outliner.d.ts.map +1 -1
  93. package/dist/types/src/extensions/outliner/selection.d.ts.map +1 -1
  94. package/dist/types/src/extensions/outliner/tree.d.ts +2 -2
  95. package/dist/types/src/extensions/outliner/tree.d.ts.map +1 -1
  96. package/dist/types/src/extensions/preview/preview.d.ts +3 -6
  97. package/dist/types/src/extensions/preview/preview.d.ts.map +1 -1
  98. package/dist/types/src/extensions/tags/extended-markdown.d.ts +10 -0
  99. package/dist/types/src/extensions/tags/extended-markdown.d.ts.map +1 -0
  100. package/dist/types/src/extensions/tags/extended-markdown.test.d.ts +2 -0
  101. package/dist/types/src/extensions/tags/extended-markdown.test.d.ts.map +1 -0
  102. package/dist/types/src/extensions/tags/index.d.ts +4 -0
  103. package/dist/types/src/extensions/tags/index.d.ts.map +1 -0
  104. package/dist/types/src/extensions/tags/streamer.d.ts +12 -0
  105. package/dist/types/src/extensions/tags/streamer.d.ts.map +1 -0
  106. package/dist/types/src/extensions/tags/xml-tags.d.ts +71 -0
  107. package/dist/types/src/extensions/tags/xml-tags.d.ts.map +1 -0
  108. package/dist/types/src/extensions/tags/xml-util.d.ts +10 -0
  109. package/dist/types/src/extensions/tags/xml-util.d.ts.map +1 -0
  110. package/dist/types/src/hooks/useTextEditor.d.ts.map +1 -1
  111. package/dist/types/src/index.d.ts +1 -1
  112. package/dist/types/src/index.d.ts.map +1 -1
  113. package/dist/types/src/stories/Command.stories.d.ts +12 -4
  114. package/dist/types/src/stories/Command.stories.d.ts.map +1 -1
  115. package/dist/types/src/stories/CommandMenu.stories.d.ts +11 -4
  116. package/dist/types/src/stories/CommandMenu.stories.d.ts.map +1 -1
  117. package/dist/types/src/stories/Comments.stories.d.ts +21 -9
  118. package/dist/types/src/stories/Comments.stories.d.ts.map +1 -1
  119. package/dist/types/src/stories/EditorToolbar.stories.d.ts +40 -3
  120. package/dist/types/src/stories/EditorToolbar.stories.d.ts.map +1 -1
  121. package/dist/types/src/stories/Experimental.stories.d.ts +22 -12
  122. package/dist/types/src/stories/Experimental.stories.d.ts.map +1 -1
  123. package/dist/types/src/stories/Markdown.stories.d.ts +32 -42
  124. package/dist/types/src/stories/Markdown.stories.d.ts.map +1 -1
  125. package/dist/types/src/stories/Outliner.stories.d.ts +15 -20
  126. package/dist/types/src/stories/Outliner.stories.d.ts.map +1 -1
  127. package/dist/types/src/stories/Preview.stories.d.ts +21 -6
  128. package/dist/types/src/stories/Preview.stories.d.ts.map +1 -1
  129. package/dist/types/src/stories/Tags.stories.d.ts +17 -0
  130. package/dist/types/src/stories/Tags.stories.d.ts.map +1 -0
  131. package/dist/types/src/stories/TextEditor.stories.d.ts +38 -51
  132. package/dist/types/src/stories/TextEditor.stories.d.ts.map +1 -1
  133. package/dist/types/src/stories/components/EditorStory.d.ts +3 -6
  134. package/dist/types/src/stories/components/EditorStory.d.ts.map +1 -1
  135. package/dist/types/src/styles/theme.d.ts.map +1 -1
  136. package/dist/types/src/testing/PreviewPopover.d.ts +20 -0
  137. package/dist/types/src/testing/PreviewPopover.d.ts.map +1 -0
  138. package/dist/types/src/testing/index.d.ts +1 -0
  139. package/dist/types/src/testing/index.d.ts.map +1 -1
  140. package/dist/types/src/testing/util.d.ts +1 -0
  141. package/dist/types/src/testing/util.d.ts.map +1 -1
  142. package/dist/types/src/translations.d.ts +28 -29
  143. package/dist/types/src/translations.d.ts.map +1 -1
  144. package/dist/types/src/types/index.d.ts +2 -0
  145. package/dist/types/src/types/index.d.ts.map +1 -0
  146. package/dist/types/src/types/types.d.ts +21 -0
  147. package/dist/types/src/types/types.d.ts.map +1 -0
  148. package/dist/types/src/util/cursor.d.ts.map +1 -1
  149. package/dist/types/src/util/debug.d.ts +1 -1
  150. package/dist/types/src/util/debug.d.ts.map +1 -1
  151. package/dist/types/src/util/decorations.d.ts +4 -0
  152. package/dist/types/src/util/decorations.d.ts.map +1 -0
  153. package/dist/types/src/util/dom.d.ts +2 -12
  154. package/dist/types/src/util/dom.d.ts.map +1 -1
  155. package/dist/types/src/util/domino.d.ts +18 -0
  156. package/dist/types/src/util/domino.d.ts.map +1 -0
  157. package/dist/types/src/util/index.d.ts +2 -0
  158. package/dist/types/src/util/index.d.ts.map +1 -1
  159. package/dist/types/src/util/react.d.ts +1 -1
  160. package/dist/types/src/util/react.d.ts.map +1 -1
  161. package/dist/types/tsconfig.tsbuildinfo +1 -1
  162. package/package.json +64 -55
  163. package/src/components/{Popover → CommandMenu}/CommandMenu.tsx +93 -26
  164. package/src/components/{Popover → CommandMenu}/index.ts +0 -2
  165. package/src/components/Editor/Editor.tsx +39 -0
  166. package/src/components/Editor/index.ts +5 -0
  167. package/src/components/EditorToolbar/EditorToolbar.tsx +40 -30
  168. package/src/components/EditorToolbar/blocks.ts +22 -25
  169. package/src/components/EditorToolbar/formatting.ts +22 -25
  170. package/src/components/EditorToolbar/headings.ts +10 -5
  171. package/src/components/EditorToolbar/image.ts +8 -4
  172. package/src/components/EditorToolbar/lists.ts +16 -19
  173. package/src/components/EditorToolbar/search.ts +8 -4
  174. package/src/components/EditorToolbar/util.ts +21 -9
  175. package/src/components/EditorToolbar/view-mode.ts +12 -7
  176. package/src/components/index.ts +2 -1
  177. package/src/defaults.ts +5 -2
  178. package/src/extensions/autocomplete.ts +204 -54
  179. package/src/extensions/automerge/automerge.stories.tsx +26 -17
  180. package/src/extensions/automerge/automerge.ts +4 -3
  181. package/src/extensions/automerge/defs.ts +1 -1
  182. package/src/extensions/automerge/sync.ts +1 -1
  183. package/src/extensions/automerge/update-automerge.ts +1 -1
  184. package/src/extensions/autoscroll.ts +157 -0
  185. package/src/extensions/awareness/awareness.ts +2 -2
  186. package/src/extensions/blast.ts +3 -16
  187. package/src/extensions/command/action.ts +1 -2
  188. package/src/extensions/command/command-menu.ts +7 -6
  189. package/src/extensions/command/command.ts +3 -3
  190. package/src/extensions/command/floating-menu.ts +10 -15
  191. package/src/extensions/command/hint.ts +2 -1
  192. package/src/extensions/command/index.ts +1 -1
  193. package/src/extensions/command/placeholder.ts +1 -1
  194. package/src/extensions/command/state.ts +4 -3
  195. package/src/extensions/command/typeahead.ts +28 -15
  196. package/src/extensions/command/useCommandMenu.ts +6 -9
  197. package/src/extensions/comments.ts +18 -13
  198. package/src/extensions/dnd.ts +1 -1
  199. package/src/extensions/factories.ts +22 -15
  200. package/src/extensions/folding.tsx +2 -2
  201. package/src/extensions/index.ts +2 -0
  202. package/src/extensions/markdown/action.ts +2 -1
  203. package/src/extensions/markdown/bundle.ts +25 -3
  204. package/src/extensions/markdown/changes.ts +1 -1
  205. package/src/extensions/markdown/decorate.ts +23 -14
  206. package/src/extensions/markdown/formatting.test.ts +7 -7
  207. package/src/extensions/markdown/formatting.ts +16 -14
  208. package/src/extensions/markdown/highlight.ts +1 -1
  209. package/src/extensions/markdown/image.ts +3 -4
  210. package/src/extensions/markdown/link.ts +3 -0
  211. package/src/extensions/markdown/table.ts +7 -1
  212. package/src/extensions/mention.ts +1 -1
  213. package/src/extensions/modes.ts +0 -9
  214. package/src/extensions/outliner/outliner.test.ts +3 -2
  215. package/src/extensions/outliner/outliner.ts +6 -5
  216. package/src/extensions/outliner/selection.ts +1 -1
  217. package/src/extensions/outliner/tree.test.ts +2 -1
  218. package/src/extensions/outliner/tree.ts +2 -2
  219. package/src/extensions/preview/preview.ts +59 -62
  220. package/src/extensions/tags/extended-markdown.test.ts +261 -0
  221. package/src/extensions/tags/extended-markdown.ts +78 -0
  222. package/src/extensions/tags/index.ts +7 -0
  223. package/src/extensions/tags/streamer.ts +244 -0
  224. package/src/extensions/tags/xml-tags.ts +335 -0
  225. package/src/extensions/tags/xml-util.ts +94 -0
  226. package/src/hooks/useTextEditor.ts +3 -15
  227. package/src/index.ts +1 -1
  228. package/src/stories/Command.stories.tsx +24 -31
  229. package/src/stories/CommandMenu.stories.tsx +29 -30
  230. package/src/stories/Comments.stories.tsx +10 -6
  231. package/src/stories/EditorToolbar.stories.tsx +10 -11
  232. package/src/stories/Experimental.stories.tsx +12 -8
  233. package/src/stories/Markdown.stories.tsx +21 -17
  234. package/src/stories/Outliner.stories.tsx +42 -30
  235. package/src/stories/Preview.stories.tsx +34 -33
  236. package/src/stories/Tags.stories.tsx +81 -0
  237. package/src/stories/TextEditor.stories.tsx +41 -35
  238. package/src/stories/components/EditorStory.tsx +9 -10
  239. package/src/styles/theme.ts +11 -10
  240. package/src/testing/PreviewPopover.tsx +78 -0
  241. package/src/testing/index.ts +1 -0
  242. package/src/testing/util.ts +2 -0
  243. package/src/translations.ts +5 -3
  244. package/src/types/index.ts +5 -0
  245. package/src/types/types.ts +32 -0
  246. package/src/util/cursor.ts +2 -1
  247. package/src/util/debug.ts +2 -2
  248. package/src/util/decorations.ts +21 -0
  249. package/src/util/dom.ts +5 -27
  250. package/src/util/domino.ts +51 -0
  251. package/src/util/index.ts +2 -0
  252. package/src/util/react.tsx +1 -1
  253. package/dist/lib/node/index.cjs +0 -7754
  254. package/dist/lib/node/index.cjs.map +0 -7
  255. package/dist/lib/node/meta.json +0 -1
  256. package/dist/lib/node/testing/index.cjs +0 -29
  257. package/dist/lib/node/testing/index.cjs.map +0 -7
  258. package/dist/types/src/components/Popover/CommandMenu.d.ts.map +0 -1
  259. package/dist/types/src/components/Popover/RefDropdownMenu.d.ts +0 -21
  260. package/dist/types/src/components/Popover/RefDropdownMenu.d.ts.map +0 -1
  261. package/dist/types/src/components/Popover/RefPopover.d.ts +0 -34
  262. package/dist/types/src/components/Popover/RefPopover.d.ts.map +0 -1
  263. package/dist/types/src/components/Popover/index.d.ts +0 -4
  264. package/dist/types/src/components/Popover/index.d.ts.map +0 -1
  265. package/dist/types/src/types.d.ts +0 -14
  266. package/dist/types/src/types.d.ts.map +0 -1
  267. package/src/components/Popover/RefDropdownMenu.tsx +0 -79
  268. package/src/components/Popover/RefPopover.tsx +0 -99
  269. package/src/types.ts +0 -23
@@ -2,21 +2,14 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos/lit-ui/dx-ref-tag.pcss';
6
-
7
5
  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';
6
+ import { type EditorState, type Extension, RangeSetBuilder, StateField } from '@codemirror/state';
16
7
  import { Decoration, type DecorationSet, EditorView, WidgetType } from '@codemirror/view';
17
8
  import { type SyntaxNode } from '@lezer/common';
18
9
 
19
10
  export type PreviewLinkRef = {
11
+ /** @deprecated */
12
+ // TODO(burdon): Remove?
20
13
  suggest?: boolean;
21
14
  block?: boolean;
22
15
  label: string;
@@ -29,10 +22,6 @@ export type PreviewLinkTarget = {
29
22
  object?: any;
30
23
  };
31
24
 
32
- // TODO(wittjosiah): Remove.
33
- // TODO(burdon): Handle error.
34
- export type PreviewLookup = (link: PreviewLinkRef) => Promise<PreviewLinkTarget | null | undefined>;
35
-
36
25
  export type PreviewOptions = {
37
26
  addBlockContainer?: (link: PreviewLinkRef, el: HTMLElement) => void;
38
27
  removeBlockContainer?: (link: PreviewLinkRef) => void;
@@ -44,10 +33,16 @@ export type PreviewOptions = {
44
33
  export const preview = (options: PreviewOptions = {}): Extension => {
45
34
  return [
46
35
  // NOTE: Atomic block decorations must be created from a state field, now a widget, otherwise it results in the following error:
47
- // "Block decorations may not be specified via plugins"
36
+ // "Block decorations may not be specified via plugins".
48
37
  StateField.define<DecorationSet>({
49
38
  create: (state) => buildDecorations(state, options),
50
- update: (_: RangeSet<Decoration>, tr: Transaction) => buildDecorations(tr.state, options),
39
+ update: (decorations, tr) => {
40
+ if (tr.docChanged) {
41
+ return buildDecorations(tr.state, options);
42
+ }
43
+
44
+ return decorations.map(tr.changes);
45
+ },
51
46
  provide: (field) => [
52
47
  EditorView.decorations.from(field),
53
48
  EditorView.atomicRanges.of((view) => view.state.field(field)),
@@ -56,42 +51,19 @@ export const preview = (options: PreviewOptions = {}): Extension => {
56
51
  ];
57
52
  };
58
53
 
59
- /**
60
- * Link references.
61
- *
62
- * [Label][dxn:echo:123] Inline reference
63
- * ![Label][dxn:echo:123] Block reference
64
- * ![Label][?dxn:echo:123] Suggestion
65
- */
66
- export const getLinkRef = (state: EditorState, node: SyntaxNode): PreviewLinkRef | undefined => {
67
- const mark = node.getChild('LinkMark');
68
- const label = node.getChild('LinkLabel');
69
- if (mark && label) {
70
- const ref = state.sliceDoc(label.from + 1, label.to - 1);
71
- return {
72
- suggest: ref.startsWith('?'),
73
- block: state.sliceDoc(mark.from, mark.from + 1) === '!',
74
- label: state.sliceDoc(mark.to, label.from - 1),
75
- ref: ref.startsWith('?') ? ref.slice(1) : ref,
76
- };
77
- }
78
- };
79
-
80
54
  /**
81
55
  * Echo references are represented as markdown reference links.
82
56
  * https://www.markdownguide.org/basic-syntax/#reference-style-links
83
- * [Label|block][dxn:echo:123]
84
- * [Label|inline][dxn:echo:123]
85
57
  */
86
- const buildDecorations = (state: EditorState, options: PreviewOptions) => {
58
+ const buildDecorations = (state: EditorState, options: PreviewOptions): DecorationSet => {
87
59
  const builder = new RangeSetBuilder<Decoration>();
88
60
 
89
61
  syntaxTree(state).iterate({
90
62
  enter: (node) => {
91
63
  switch (node.name) {
92
64
  //
93
- // Decoration.
94
- // [Label][dxn:echo:123]
65
+ // Inline widget.
66
+ // [Label](dxn:echo:123)
95
67
  //
96
68
  case 'Link': {
97
69
  const link = getLinkRef(state, node.node);
@@ -101,29 +73,32 @@ const buildDecorations = (state: EditorState, options: PreviewOptions) => {
101
73
  node.to,
102
74
  Decoration.replace({
103
75
  widget: new PreviewInlineWidget(options, link),
76
+ side: 1,
104
77
  }),
105
78
  );
106
79
  }
107
- break;
80
+ return false;
108
81
  }
82
+
109
83
  //
110
- // Block widget.
111
- // ![Label][dxn:echo:123]
84
+ // Block widget (transclusion).
85
+ // ![Label](dxn:echo:123)
112
86
  //
113
87
  case 'Image': {
114
- const link = getLinkRef(state, node.node);
115
- if (options.addBlockContainer && options.removeBlockContainer && link) {
116
- builder.add(
117
- node.from,
118
- node.to,
119
- Decoration.replace({
120
- block: true,
121
- // atomic: true,
122
- widget: new PreviewBlockWidget(options, link),
123
- }),
124
- );
88
+ if (options.addBlockContainer && options.removeBlockContainer) {
89
+ const link = getLinkRef(state, node.node);
90
+ if (link) {
91
+ builder.add(
92
+ node.from,
93
+ node.to,
94
+ Decoration.replace({
95
+ block: true,
96
+ widget: new PreviewBlockWidget(options, link),
97
+ }),
98
+ );
99
+ }
125
100
  }
126
- break;
101
+ return false;
127
102
  }
128
103
  }
129
104
  },
@@ -132,9 +107,30 @@ const buildDecorations = (state: EditorState, options: PreviewOptions) => {
132
107
  return builder.finish();
133
108
  };
134
109
 
110
+ /**
111
+ * Link references.
112
+ * [Label](dxn:echo:123) Inline reference
113
+ * ![Label](dxn:echo:123) Block reference
114
+ */
115
+ export const getLinkRef = (state: EditorState, node: SyntaxNode): PreviewLinkRef | undefined => {
116
+ const mark = node.getChildren('LinkMark');
117
+ const urlNode = node.getChild('URL');
118
+ if (mark && urlNode) {
119
+ const url = state.sliceDoc(urlNode.from, urlNode.to);
120
+ if (url.startsWith('dxn:')) {
121
+ const label = state.sliceDoc(mark[0].to, mark[1].from);
122
+ return {
123
+ block: state.sliceDoc(mark[0].from, mark[0].from + 1) === '!',
124
+ label,
125
+ ref: url,
126
+ };
127
+ }
128
+ }
129
+ };
130
+
135
131
  /**
136
132
  * Inline widget.
137
- * [Label][dxn:echo:123]
133
+ * [Label](dxn:echo:123)
138
134
  */
139
135
  class PreviewInlineWidget extends WidgetType {
140
136
  constructor(
@@ -152,8 +148,9 @@ class PreviewInlineWidget extends WidgetType {
152
148
  return this._link.ref === other._link.ref && this._link.label === other._link.label;
153
149
  }
154
150
 
155
- override toDOM(view: EditorView): HTMLElement {
156
- const root = document.createElement('dx-ref-tag');
151
+ override toDOM(_view: EditorView): HTMLElement {
152
+ const root = document.createElement('dx-anchor');
153
+ root.classList.add('dx-tag--anchor');
157
154
  root.textContent = this._link.label;
158
155
  root.setAttribute('refId', this._link.ref);
159
156
  return root;
@@ -161,7 +158,7 @@ class PreviewInlineWidget extends WidgetType {
161
158
  }
162
159
 
163
160
  /**
164
- * Block widget.
161
+ * Block widget (e.g., for surfaces).
165
162
  * ![Label][dxn:echo:123]
166
163
  */
167
164
  class PreviewBlockWidget extends WidgetType {
@@ -180,7 +177,7 @@ class PreviewBlockWidget extends WidgetType {
180
177
  return this._link.ref === other._link.ref;
181
178
  }
182
179
 
183
- override toDOM(view: EditorView): HTMLDivElement {
180
+ override toDOM(_view: EditorView): HTMLDivElement {
184
181
  const root = document.createElement('div');
185
182
  root.classList.add('cm-preview-block', 'density-coarse');
186
183
  this._options.addBlockContainer?.(this._link, root);
@@ -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';