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

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 (312) hide show
  1. package/dist/lib/browser/{chunk-22UMM3QJ.mjs → chunk-HL3YF6WC.mjs} +2 -2
  2. package/dist/lib/browser/chunk-HL3YF6WC.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +4960 -4414
  4. package/dist/lib/browser/index.mjs.map +4 -4
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/types/index.mjs +1 -1
  7. package/dist/lib/node-esm/{chunk-YXYQPV6R.mjs → chunk-YJZGD3LY.mjs} +2 -2
  8. package/dist/lib/node-esm/chunk-YJZGD3LY.mjs.map +7 -0
  9. package/dist/lib/node-esm/index.mjs +4960 -4414
  10. package/dist/lib/node-esm/index.mjs.map +4 -4
  11. package/dist/lib/node-esm/meta.json +1 -1
  12. package/dist/lib/node-esm/types/index.mjs +1 -1
  13. package/dist/types/src/components/Editor/Editor.d.ts +37 -15
  14. package/dist/types/src/components/Editor/Editor.d.ts.map +1 -1
  15. package/dist/types/src/components/Editor/Editor.stories.d.ts +20 -0
  16. package/dist/types/src/components/Editor/Editor.stories.d.ts.map +1 -0
  17. package/dist/types/src/components/EditorContent/EditorContent.d.ts +29 -0
  18. package/dist/types/src/components/EditorContent/EditorContent.d.ts.map +1 -0
  19. package/dist/types/src/components/EditorContent/EditorContent.stories.d.ts +26 -0
  20. package/dist/types/src/components/EditorContent/EditorContent.stories.d.ts.map +1 -0
  21. package/dist/types/src/components/EditorContent/controller.d.ts +10 -0
  22. package/dist/types/src/components/EditorContent/controller.d.ts.map +1 -0
  23. package/dist/types/src/components/EditorContent/index.d.ts +3 -0
  24. package/dist/types/src/components/EditorContent/index.d.ts.map +1 -0
  25. package/dist/types/src/components/EditorMenuProvider/EditorMenuProvider.d.ts +36 -0
  26. package/dist/types/src/components/EditorMenuProvider/EditorMenuProvider.d.ts.map +1 -0
  27. package/dist/types/src/components/EditorMenuProvider/index.d.ts +7 -0
  28. package/dist/types/src/components/EditorMenuProvider/index.d.ts.map +1 -0
  29. package/dist/types/src/components/EditorMenuProvider/menu-presets.d.ts +4 -0
  30. package/dist/types/src/components/EditorMenuProvider/menu-presets.d.ts.map +1 -0
  31. package/dist/types/src/components/EditorMenuProvider/menu.d.ts +28 -0
  32. package/dist/types/src/components/EditorMenuProvider/menu.d.ts.map +1 -0
  33. package/dist/types/src/components/EditorMenuProvider/popover.d.ts +47 -0
  34. package/dist/types/src/components/EditorMenuProvider/popover.d.ts.map +1 -0
  35. package/dist/types/src/components/EditorMenuProvider/useEditorMenu.d.ts +34 -0
  36. package/dist/types/src/components/EditorMenuProvider/useEditorMenu.d.ts.map +1 -0
  37. package/dist/types/src/components/EditorMenuProvider/util.d.ts +8 -0
  38. package/dist/types/src/components/EditorMenuProvider/util.d.ts.map +1 -0
  39. package/dist/types/src/components/EditorPreviewProvider/EditorPreviewProvider.d.ts +16 -0
  40. package/dist/types/src/components/EditorPreviewProvider/EditorPreviewProvider.d.ts.map +1 -0
  41. package/dist/types/src/components/EditorPreviewProvider/index.d.ts +2 -0
  42. package/dist/types/src/components/EditorPreviewProvider/index.d.ts.map +1 -0
  43. package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts +26 -2
  44. package/dist/types/src/components/EditorToolbar/EditorToolbar.d.ts.map +1 -1
  45. package/dist/types/src/components/EditorToolbar/actions.d.ts +39 -0
  46. package/dist/types/src/components/EditorToolbar/actions.d.ts.map +1 -0
  47. package/dist/types/src/components/EditorToolbar/blocks.d.ts +3 -3
  48. package/dist/types/src/components/EditorToolbar/blocks.d.ts.map +1 -1
  49. package/dist/types/src/components/EditorToolbar/formatting.d.ts +3 -3
  50. package/dist/types/src/components/EditorToolbar/formatting.d.ts.map +1 -1
  51. package/dist/types/src/components/EditorToolbar/headings.d.ts +3 -3
  52. package/dist/types/src/components/EditorToolbar/headings.d.ts.map +1 -1
  53. package/dist/types/src/components/EditorToolbar/index.d.ts +2 -1
  54. package/dist/types/src/components/EditorToolbar/index.d.ts.map +1 -1
  55. package/dist/types/src/components/EditorToolbar/useEditorToolbar.d.ts +11 -0
  56. package/dist/types/src/components/EditorToolbar/useEditorToolbar.d.ts.map +1 -0
  57. package/dist/types/src/components/EditorToolbar/view-mode.d.ts +3 -3
  58. package/dist/types/src/components/EditorToolbar/view-mode.d.ts.map +1 -1
  59. package/dist/types/src/components/index.d.ts +4 -2
  60. package/dist/types/src/components/index.d.ts.map +1 -1
  61. package/dist/types/src/extensions/autocomplete/autocomplete.d.ts +17 -0
  62. package/dist/types/src/extensions/autocomplete/autocomplete.d.ts.map +1 -0
  63. package/dist/types/src/extensions/autocomplete/index.d.ts +5 -0
  64. package/dist/types/src/extensions/autocomplete/index.d.ts.map +1 -0
  65. package/dist/types/src/extensions/autocomplete/match.d.ts +13 -0
  66. package/dist/types/src/extensions/autocomplete/match.d.ts.map +1 -0
  67. package/dist/types/src/extensions/autocomplete/placeholder.d.ts +20 -0
  68. package/dist/types/src/extensions/autocomplete/placeholder.d.ts.map +1 -0
  69. package/dist/types/src/extensions/autocomplete/typeahead.d.ts +10 -0
  70. package/dist/types/src/extensions/autocomplete/typeahead.d.ts.map +1 -0
  71. package/dist/types/src/extensions/automerge/automerge.d.ts +1 -1
  72. package/dist/types/src/extensions/automerge/automerge.d.ts.map +1 -1
  73. package/dist/types/src/extensions/automerge/automerge.stories.d.ts +1 -1
  74. package/dist/types/src/extensions/automerge/automerge.stories.d.ts.map +1 -1
  75. package/dist/types/src/extensions/automerge/cursor.d.ts +1 -1
  76. package/dist/types/src/extensions/automerge/cursor.d.ts.map +1 -1
  77. package/dist/types/src/extensions/automerge/sync.d.ts +3 -3
  78. package/dist/types/src/extensions/automerge/sync.d.ts.map +1 -1
  79. package/dist/types/src/extensions/automerge/update-automerge.d.ts +1 -1
  80. package/dist/types/src/extensions/automerge/update-automerge.d.ts.map +1 -1
  81. package/dist/types/src/extensions/autoscroll.d.ts +14 -4
  82. package/dist/types/src/extensions/autoscroll.d.ts.map +1 -1
  83. package/dist/types/src/extensions/awareness/awareness-provider.d.ts +1 -1
  84. package/dist/types/src/extensions/awareness/awareness-provider.d.ts.map +1 -1
  85. package/dist/types/src/extensions/blocks.d.ts +2 -0
  86. package/dist/types/src/extensions/blocks.d.ts.map +1 -0
  87. package/dist/types/src/extensions/bookmarks.d.ts +12 -0
  88. package/dist/types/src/extensions/bookmarks.d.ts.map +1 -0
  89. package/dist/types/src/extensions/comments.d.ts.map +1 -1
  90. package/dist/types/src/extensions/factories.d.ts +10 -5
  91. package/dist/types/src/extensions/factories.d.ts.map +1 -1
  92. package/dist/types/src/extensions/focus.d.ts.map +1 -1
  93. package/dist/types/src/extensions/folding.d.ts.map +1 -1
  94. package/dist/types/src/extensions/index.d.ts +8 -1
  95. package/dist/types/src/extensions/index.d.ts.map +1 -1
  96. package/dist/types/src/extensions/json.d.ts +1 -1
  97. package/dist/types/src/extensions/json.d.ts.map +1 -1
  98. package/dist/types/src/extensions/listener.d.ts +8 -6
  99. package/dist/types/src/extensions/listener.d.ts.map +1 -1
  100. package/dist/types/src/extensions/markdown/bundle.d.ts.map +1 -1
  101. package/dist/types/src/extensions/markdown/decorate.d.ts.map +1 -1
  102. package/dist/types/src/extensions/markdown/formatting.d.ts +1 -3
  103. package/dist/types/src/extensions/markdown/formatting.d.ts.map +1 -1
  104. package/dist/types/src/extensions/modal.d.ts +7 -0
  105. package/dist/types/src/extensions/modal.d.ts.map +1 -0
  106. package/dist/types/src/extensions/modes.d.ts +1 -1
  107. package/dist/types/src/extensions/modes.d.ts.map +1 -1
  108. package/dist/types/src/extensions/outliner/menu.d.ts +8 -0
  109. package/dist/types/src/extensions/outliner/menu.d.ts.map +1 -0
  110. package/dist/types/src/extensions/preview/preview.d.ts +6 -3
  111. package/dist/types/src/extensions/preview/preview.d.ts.map +1 -1
  112. package/dist/types/src/extensions/replacer.d.ts +21 -0
  113. package/dist/types/src/extensions/replacer.d.ts.map +1 -0
  114. package/dist/types/src/extensions/replacer.test.d.ts +2 -0
  115. package/dist/types/src/extensions/replacer.test.d.ts.map +1 -0
  116. package/dist/types/src/extensions/scrolling.d.ts +78 -0
  117. package/dist/types/src/extensions/scrolling.d.ts.map +1 -0
  118. package/dist/types/src/extensions/state.d.ts +2 -0
  119. package/dist/types/src/extensions/state.d.ts.map +1 -0
  120. package/dist/types/src/extensions/submit.d.ts +10 -0
  121. package/dist/types/src/extensions/submit.d.ts.map +1 -0
  122. package/dist/types/src/extensions/tab.d.ts +4 -0
  123. package/dist/types/src/extensions/tab.d.ts.map +1 -0
  124. package/dist/types/src/extensions/tags/streamer.d.ts.map +1 -1
  125. package/dist/types/src/extensions/tags/xml-tags.d.ts +42 -16
  126. package/dist/types/src/extensions/tags/xml-tags.d.ts.map +1 -1
  127. package/dist/types/src/hooks/useTextEditor.d.ts +5 -9
  128. package/dist/types/src/hooks/useTextEditor.d.ts.map +1 -1
  129. package/dist/types/src/stories/{Command.stories.d.ts → CommandDialog.stories.d.ts} +2 -3
  130. package/dist/types/src/stories/CommandDialog.stories.d.ts.map +1 -0
  131. package/dist/types/src/stories/Comments.stories.d.ts +3 -4
  132. package/dist/types/src/stories/Comments.stories.d.ts.map +1 -1
  133. package/dist/types/src/stories/EditorToolbar.stories.d.ts +1 -2
  134. package/dist/types/src/stories/EditorToolbar.stories.d.ts.map +1 -1
  135. package/dist/types/src/stories/Experimental.stories.d.ts +3 -4
  136. package/dist/types/src/stories/Experimental.stories.d.ts.map +1 -1
  137. package/dist/types/src/stories/Markdown.stories.d.ts +3 -4
  138. package/dist/types/src/stories/Markdown.stories.d.ts.map +1 -1
  139. package/dist/types/src/stories/Outliner.stories.d.ts +0 -1
  140. package/dist/types/src/stories/Outliner.stories.d.ts.map +1 -1
  141. package/dist/types/src/stories/{CommandMenu.stories.d.ts → Popover.stories.d.ts} +6 -6
  142. package/dist/types/src/stories/Popover.stories.d.ts.map +1 -0
  143. package/dist/types/src/stories/Preview.stories.d.ts +4 -4
  144. package/dist/types/src/stories/Preview.stories.d.ts.map +1 -1
  145. package/dist/types/src/stories/Tags.stories.d.ts +0 -1
  146. package/dist/types/src/stories/Tags.stories.d.ts.map +1 -1
  147. package/dist/types/src/stories/TextEditor.stories.d.ts +3 -5
  148. package/dist/types/src/stories/TextEditor.stories.d.ts.map +1 -1
  149. package/dist/types/src/stories/components/EditorStory.d.ts +7 -5
  150. package/dist/types/src/stories/components/EditorStory.d.ts.map +1 -1
  151. package/dist/types/src/stories/components/util.d.ts.map +1 -1
  152. package/dist/types/src/styles/theme.d.ts.map +1 -1
  153. package/dist/types/src/types/types.d.ts +1 -1
  154. package/dist/types/src/types/types.d.ts.map +1 -1
  155. package/dist/types/src/util/debug.d.ts +4 -0
  156. package/dist/types/src/util/debug.d.ts.map +1 -1
  157. package/dist/types/src/util/index.d.ts +0 -1
  158. package/dist/types/src/util/index.d.ts.map +1 -1
  159. package/dist/types/tsconfig.tsbuildinfo +1 -1
  160. package/package.json +57 -58
  161. package/src/components/Editor/Editor.stories.tsx +89 -0
  162. package/src/components/Editor/Editor.tsx +160 -25
  163. package/src/components/EditorContent/EditorContent.stories.tsx +70 -0
  164. package/src/components/EditorContent/EditorContent.tsx +70 -0
  165. package/src/components/EditorContent/controller.ts +50 -0
  166. package/src/components/EditorContent/index.ts +6 -0
  167. package/src/components/EditorMenuProvider/EditorMenuProvider.tsx +233 -0
  168. package/src/components/EditorMenuProvider/index.ts +11 -0
  169. package/src/components/EditorMenuProvider/menu-presets.ts +123 -0
  170. package/src/components/EditorMenuProvider/menu.ts +71 -0
  171. package/src/components/EditorMenuProvider/popover.ts +287 -0
  172. package/src/components/EditorMenuProvider/useEditorMenu.ts +175 -0
  173. package/src/components/EditorMenuProvider/util.ts +31 -0
  174. package/src/components/EditorPreviewProvider/EditorPreviewProvider.tsx +82 -0
  175. package/src/components/EditorPreviewProvider/index.ts +5 -0
  176. package/src/components/EditorToolbar/EditorToolbar.tsx +101 -91
  177. package/src/components/EditorToolbar/{lists.ts → actions.ts} +46 -16
  178. package/src/components/EditorToolbar/blocks.ts +2 -1
  179. package/src/components/EditorToolbar/formatting.ts +2 -1
  180. package/src/components/EditorToolbar/headings.ts +8 -5
  181. package/src/components/EditorToolbar/image.ts +1 -1
  182. package/src/components/EditorToolbar/index.ts +3 -7
  183. package/src/components/EditorToolbar/search.ts +1 -1
  184. package/src/components/EditorToolbar/useEditorToolbar.ts +20 -0
  185. package/src/components/EditorToolbar/view-mode.ts +2 -1
  186. package/src/components/index.ts +8 -2
  187. package/src/extensions/{autocomplete.ts → autocomplete/autocomplete.ts} +8 -76
  188. package/src/extensions/autocomplete/index.ts +8 -0
  189. package/src/extensions/autocomplete/match.ts +46 -0
  190. package/src/extensions/{command → autocomplete}/placeholder.ts +21 -17
  191. package/src/extensions/{command → autocomplete}/typeahead.ts +6 -48
  192. package/src/extensions/automerge/automerge.stories.tsx +9 -9
  193. package/src/extensions/automerge/automerge.ts +28 -9
  194. package/src/extensions/automerge/cursor.ts +1 -1
  195. package/src/extensions/automerge/sync.ts +8 -4
  196. package/src/extensions/automerge/update-automerge.ts +1 -1
  197. package/src/extensions/autoscroll.ts +98 -92
  198. package/src/extensions/awareness/awareness-provider.ts +2 -2
  199. package/src/extensions/blocks.ts +131 -0
  200. package/src/extensions/bookmarks.ts +75 -0
  201. package/src/extensions/comments.ts +2 -1
  202. package/src/extensions/factories.ts +47 -16
  203. package/src/extensions/focus.ts +5 -4
  204. package/src/extensions/folding.tsx +3 -6
  205. package/src/extensions/hashtag.tsx +2 -2
  206. package/src/extensions/index.ts +8 -1
  207. package/src/extensions/json.ts +1 -1
  208. package/src/extensions/listener.ts +14 -20
  209. package/src/extensions/markdown/bundle.ts +16 -4
  210. package/src/extensions/markdown/decorate.ts +9 -8
  211. package/src/extensions/markdown/formatting.ts +8 -10
  212. package/src/extensions/markdown/highlight.ts +1 -1
  213. package/src/extensions/markdown/image.ts +2 -2
  214. package/src/extensions/markdown/table.ts +6 -6
  215. package/src/extensions/modal.ts +24 -0
  216. package/src/extensions/modes.ts +2 -2
  217. package/src/extensions/{command/floating-menu.ts → outliner/menu.ts} +9 -9
  218. package/src/extensions/outliner/outliner.test.ts +1 -1
  219. package/src/extensions/outliner/outliner.ts +3 -3
  220. package/src/extensions/outliner/tree.test.ts +1 -1
  221. package/src/extensions/preview/index.ts +1 -1
  222. package/src/extensions/preview/preview.ts +14 -11
  223. package/src/extensions/replacer.test.ts +75 -0
  224. package/src/extensions/replacer.ts +93 -0
  225. package/src/extensions/scrolling.ts +189 -0
  226. package/src/extensions/selection.ts +3 -3
  227. package/src/extensions/state.ts +7 -0
  228. package/src/extensions/submit.ts +62 -0
  229. package/src/extensions/tab.ts +29 -0
  230. package/src/extensions/tags/extended-markdown.test.ts +2 -1
  231. package/src/extensions/tags/streamer.ts +4 -5
  232. package/src/extensions/tags/xml-tags.ts +320 -155
  233. package/src/extensions/typewriter.ts +1 -1
  234. package/src/hooks/useTextEditor.ts +31 -31
  235. package/src/stories/{Command.stories.tsx → CommandDialog.stories.tsx} +20 -27
  236. package/src/stories/Comments.stories.tsx +8 -9
  237. package/src/stories/EditorToolbar.stories.tsx +14 -13
  238. package/src/stories/Experimental.stories.tsx +7 -7
  239. package/src/stories/Markdown.stories.tsx +6 -6
  240. package/src/stories/Outliner.stories.tsx +20 -20
  241. package/src/stories/Popover.stories.tsx +162 -0
  242. package/src/stories/Preview.stories.tsx +46 -42
  243. package/src/stories/Tags.stories.tsx +24 -10
  244. package/src/stories/TextEditor.stories.tsx +10 -35
  245. package/src/stories/components/EditorStory.tsx +29 -16
  246. package/src/stories/components/util.tsx +40 -8
  247. package/src/styles/markdown.ts +1 -1
  248. package/src/styles/theme.ts +13 -11
  249. package/src/types/types.ts +1 -1
  250. package/src/util/debug.ts +5 -0
  251. package/src/util/index.ts +0 -1
  252. package/dist/lib/browser/chunk-22UMM3QJ.mjs.map +0 -7
  253. package/dist/lib/browser/testing/index.mjs +0 -76
  254. package/dist/lib/browser/testing/index.mjs.map +0 -7
  255. package/dist/lib/node-esm/chunk-YXYQPV6R.mjs.map +0 -7
  256. package/dist/lib/node-esm/testing/index.mjs +0 -78
  257. package/dist/lib/node-esm/testing/index.mjs.map +0 -7
  258. package/dist/types/src/components/CommandMenu/CommandMenu.d.ts +0 -38
  259. package/dist/types/src/components/CommandMenu/CommandMenu.d.ts.map +0 -1
  260. package/dist/types/src/components/CommandMenu/index.d.ts +0 -2
  261. package/dist/types/src/components/CommandMenu/index.d.ts.map +0 -1
  262. package/dist/types/src/components/EditorToolbar/lists.d.ts +0 -19
  263. package/dist/types/src/components/EditorToolbar/lists.d.ts.map +0 -1
  264. package/dist/types/src/components/EditorToolbar/util.d.ts +0 -51
  265. package/dist/types/src/components/EditorToolbar/util.d.ts.map +0 -1
  266. package/dist/types/src/extensions/autocomplete.d.ts +0 -26
  267. package/dist/types/src/extensions/autocomplete.d.ts.map +0 -1
  268. package/dist/types/src/extensions/command/action.d.ts +0 -17
  269. package/dist/types/src/extensions/command/action.d.ts.map +0 -1
  270. package/dist/types/src/extensions/command/command-menu.d.ts +0 -20
  271. package/dist/types/src/extensions/command/command-menu.d.ts.map +0 -1
  272. package/dist/types/src/extensions/command/command.d.ts +0 -6
  273. package/dist/types/src/extensions/command/command.d.ts.map +0 -1
  274. package/dist/types/src/extensions/command/floating-menu.d.ts +0 -7
  275. package/dist/types/src/extensions/command/floating-menu.d.ts.map +0 -1
  276. package/dist/types/src/extensions/command/hint.d.ts +0 -19
  277. package/dist/types/src/extensions/command/hint.d.ts.map +0 -1
  278. package/dist/types/src/extensions/command/index.d.ts +0 -7
  279. package/dist/types/src/extensions/command/index.d.ts.map +0 -1
  280. package/dist/types/src/extensions/command/placeholder.d.ts +0 -10
  281. package/dist/types/src/extensions/command/placeholder.d.ts.map +0 -1
  282. package/dist/types/src/extensions/command/state.d.ts +0 -16
  283. package/dist/types/src/extensions/command/state.d.ts.map +0 -1
  284. package/dist/types/src/extensions/command/typeahead.d.ts +0 -22
  285. package/dist/types/src/extensions/command/typeahead.d.ts.map +0 -1
  286. package/dist/types/src/extensions/command/useCommandMenu.d.ts +0 -25
  287. package/dist/types/src/extensions/command/useCommandMenu.d.ts.map +0 -1
  288. package/dist/types/src/stories/Command.stories.d.ts.map +0 -1
  289. package/dist/types/src/stories/CommandMenu.stories.d.ts.map +0 -1
  290. package/dist/types/src/testing/PreviewPopover.d.ts +0 -20
  291. package/dist/types/src/testing/PreviewPopover.d.ts.map +0 -1
  292. package/dist/types/src/testing/index.d.ts +0 -3
  293. package/dist/types/src/testing/index.d.ts.map +0 -1
  294. package/dist/types/src/testing/util.d.ts +0 -3
  295. package/dist/types/src/testing/util.d.ts.map +0 -1
  296. package/dist/types/src/util/domino.d.ts +0 -18
  297. package/dist/types/src/util/domino.d.ts.map +0 -1
  298. package/src/components/CommandMenu/CommandMenu.tsx +0 -346
  299. package/src/components/CommandMenu/index.ts +0 -5
  300. package/src/components/EditorToolbar/util.ts +0 -76
  301. package/src/extensions/command/action.ts +0 -55
  302. package/src/extensions/command/command-menu.ts +0 -211
  303. package/src/extensions/command/command.ts +0 -34
  304. package/src/extensions/command/hint.ts +0 -103
  305. package/src/extensions/command/index.ts +0 -10
  306. package/src/extensions/command/state.ts +0 -90
  307. package/src/extensions/command/useCommandMenu.ts +0 -115
  308. package/src/stories/CommandMenu.stories.tsx +0 -158
  309. package/src/testing/PreviewPopover.tsx +0 -78
  310. package/src/testing/index.ts +0 -6
  311. package/src/testing/util.ts +0 -7
  312. package/src/util/domino.ts +0 -51
@@ -2,11 +2,13 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import { Rx } from '@effect-rx/rx-react';
5
+ import { type EditorView } from '@codemirror/view';
6
+ import { Atom } from '@effect-atom/atom-react';
6
7
  import React, { memo, useMemo } from 'react';
7
8
 
8
- import { rxFromSignal } from '@dxos/app-graph';
9
- import { ElevationProvider } from '@dxos/react-ui';
9
+ import { atomFromSignal } from '@dxos/app-graph';
10
+ import { type Live } from '@dxos/live-object';
11
+ import { ElevationProvider, type ThemedClassName } from '@dxos/react-ui';
10
12
  import {
11
13
  type ActionGraphProps,
12
14
  MenuProvider,
@@ -15,120 +17,128 @@ import {
15
17
  useMenuActions,
16
18
  } from '@dxos/react-ui-menu';
17
19
 
20
+ import { type EditorViewMode } from '../../types';
21
+
22
+ import { createLists } from './actions';
18
23
  import { createBlocks } from './blocks';
19
24
  import { createFormatting } from './formatting';
20
25
  import { createHeadings } from './headings';
21
26
  import { createImageUpload } from './image';
22
- import { createLists } from './lists';
23
27
  import { createSearch } from './search';
24
- import { type EditorToolbarActionGraphProps, type EditorToolbarFeatureFlags, type EditorToolbarProps } from './util';
28
+ import { type EditorToolbarState } from './useEditorToolbar';
25
29
  import { createViewMode } from './view-mode';
26
30
 
31
+ export type EditorToolbarFeatureFlags = Partial<{
32
+ showHeadings: boolean;
33
+ showFormatting: boolean;
34
+ showLists: boolean;
35
+ showBlocks: boolean;
36
+ showSearch: boolean;
37
+
38
+ // TODO(wittjosiah): Factor out (depends on plugin-level capabilities.)
39
+ onImageUpload: () => void;
40
+ onViewModeChange: (mode: EditorViewMode) => void;
41
+ }>;
42
+
43
+ export type EditorToolbarActionGraphProps = {
44
+ state: Live<EditorToolbarState>;
45
+ getView: () => EditorView;
46
+ // TODO(wittjosiah): Control positioning.
47
+ customActions?: Atom.Atom<ActionGraphProps>;
48
+ };
49
+
50
+ export type EditorToolbarProps = ThemedClassName<
51
+ {
52
+ role?: string;
53
+ attendableId?: string;
54
+ } & (EditorToolbarActionGraphProps & EditorToolbarFeatureFlags)
55
+ >;
56
+
57
+ // TODO(burdon): Remove role dependency.
58
+ export const EditorToolbar = memo(({ classNames, role, attendableId, ...props }: EditorToolbarProps) => {
59
+ const menuProps = useEditorToolbarActionGraph(props);
60
+
61
+ return (
62
+ <ElevationProvider elevation={role === 'section' ? 'positioned' : 'base'}>
63
+ <MenuProvider {...menuProps} attendableId={attendableId}>
64
+ <ToolbarMenu classNames={classNames} textBlockWidth />
65
+ </MenuProvider>
66
+ </ElevationProvider>
67
+ );
68
+ });
69
+
70
+ type ToolbarActionsProps = Pick<EditorToolbarActionGraphProps, 'state' | 'getView' | 'customActions'> &
71
+ EditorToolbarFeatureFlags;
72
+
73
+ // TODO(wittjosiah): Toolbar re-rendering is causing this graph to be recreated and breaking reactivity in some cases.
74
+ // E.g. for toolbar dropdowns which use active icon, the icon is not updated when the active item changes.
75
+ // This is currently only happening in the markdown plugin usage and should be reproduced in an editor story.
76
+ const useEditorToolbarActionGraph = ({ state, getView, customActions, ...features }: ToolbarActionsProps) => {
77
+ const menuCreator = useMemo(
78
+ () => createToolbarActions({ state, getView, customActions, ...features }),
79
+ [
80
+ state,
81
+ getView,
82
+ customActions,
83
+ features?.showHeadings,
84
+ features?.showFormatting,
85
+ features?.showLists,
86
+ features?.showBlocks,
87
+ features?.showSearch,
88
+ features?.onImageUpload,
89
+ features?.onViewModeChange,
90
+ ],
91
+ );
92
+
93
+ return useMenuActions(menuCreator);
94
+ };
95
+
27
96
  const createToolbarActions = ({
28
- getView,
29
97
  state,
98
+ getView,
30
99
  customActions,
31
100
  ...features
32
- }: EditorToolbarFeatureFlags &
33
- Pick<EditorToolbarActionGraphProps, 'getView' | 'state' | 'customActions'>): Rx.Rx<ActionGraphProps> => {
34
- return Rx.make((get) => {
101
+ }: ToolbarActionsProps): Atom.Atom<ActionGraphProps> => {
102
+ return Atom.make((get) => {
35
103
  const graph: ActionGraphProps = {
36
104
  nodes: [],
37
105
  edges: [],
38
106
  };
39
107
 
40
- if (features.headings ?? true) {
41
- const headings = get(rxFromSignal(() => createHeadings(state, getView)));
42
- graph.nodes.push(...headings.nodes);
43
- graph.edges.push(...headings.edges);
44
- }
45
- if (features.formatting ?? true) {
46
- const formatting = get(rxFromSignal(() => createFormatting(state, getView)));
47
- graph.nodes.push(...formatting.nodes);
48
- graph.edges.push(...formatting.edges);
108
+ // TODO(burdon): Builder pattern?
109
+ const addSubGraph = (graph: ActionGraphProps, subGraph: ActionGraphProps) => {
110
+ graph.nodes.push(...subGraph.nodes);
111
+ graph.edges.push(...subGraph.edges);
112
+ };
113
+
114
+ if (features?.showHeadings ?? true) {
115
+ addSubGraph(graph, get(atomFromSignal(() => createHeadings(state, getView))));
49
116
  }
50
- if (features.lists ?? true) {
51
- const lists = get(rxFromSignal(() => createLists(state, getView)));
52
- graph.nodes.push(...lists.nodes);
53
- graph.edges.push(...lists.edges);
117
+ if (features?.showFormatting ?? true) {
118
+ addSubGraph(graph, get(atomFromSignal(() => createFormatting(state, getView))));
54
119
  }
55
- if (features.blocks ?? true) {
56
- const blocks = get(rxFromSignal(() => createBlocks(state, getView)));
57
- graph.nodes.push(...blocks.nodes);
58
- graph.edges.push(...blocks.edges);
120
+ if (features?.showLists ?? true) {
121
+ addSubGraph(graph, get(atomFromSignal(() => createLists(state, getView))));
59
122
  }
60
- if (features.image) {
61
- const image = get(rxFromSignal(() => createImageUpload(features.image!)));
62
- graph.nodes.push(...image.nodes);
63
- graph.edges.push(...image.edges);
123
+ if (features?.showBlocks ?? true) {
124
+ addSubGraph(graph, get(atomFromSignal(() => createBlocks(state, getView))));
64
125
  }
65
- {
66
- const gap = createGapSeparator();
67
- graph.nodes.push(...gap.nodes);
68
- graph.edges.push(...gap.edges);
126
+ if (features?.onImageUpload) {
127
+ addSubGraph(graph, get(atomFromSignal(() => createImageUpload(features.onImageUpload!))));
69
128
  }
129
+
130
+ addSubGraph(graph, createGapSeparator());
131
+
70
132
  if (customActions) {
71
- const custom = get(customActions);
72
- graph.nodes.push(...custom.nodes);
73
- graph.edges.push(...custom.edges);
133
+ addSubGraph(graph, get(customActions));
74
134
  }
75
- if (features.search ?? true) {
76
- const search = get(rxFromSignal(() => createSearch(getView)));
77
- graph.nodes.push(...search.nodes);
78
- graph.edges.push(...search.edges);
135
+ if (features?.showSearch ?? true) {
136
+ addSubGraph(graph, get(atomFromSignal(() => createSearch(getView))));
79
137
  }
80
- if (features.viewMode) {
81
- const viewMode = get(rxFromSignal(() => createViewMode(state, features.viewMode!)));
82
- graph.nodes.push(...viewMode.nodes);
83
- graph.edges.push(...viewMode.edges);
138
+ if (features?.onViewModeChange) {
139
+ addSubGraph(graph, get(atomFromSignal(() => createViewMode(state, features.onViewModeChange!))));
84
140
  }
85
141
 
86
142
  return graph;
87
143
  });
88
144
  };
89
-
90
- // TODO(wittjosiah): Toolbar re-rendering is causing this graph to be recreated and breaking reactivity in some cases.
91
- // E.g. for toolbar dropdowns which use active icon, the icon is not updated when the active item changes.
92
- // This is currently only happening in the markdown plugin usage and should be reproduced in an editor story.
93
- const useEditorToolbarActionGraph = (props: EditorToolbarProps) => {
94
- const menuCreator = useMemo(
95
- () =>
96
- createToolbarActions({
97
- getView: props.getView,
98
- state: props.state,
99
- customActions: props.customActions,
100
- headings: props.headings,
101
- formatting: props.formatting,
102
- lists: props.lists,
103
- blocks: props.blocks,
104
- image: props.image,
105
- search: props.search,
106
- viewMode: props.viewMode,
107
- }),
108
- [
109
- props.getView,
110
- props.state,
111
- props.customActions,
112
- props.headings,
113
- props.formatting,
114
- props.lists,
115
- props.blocks,
116
- props.image,
117
- props.search,
118
- props.viewMode,
119
- ],
120
- );
121
-
122
- return useMenuActions(menuCreator);
123
- };
124
-
125
- export const EditorToolbar = memo(({ classNames, attendableId, role, ...props }: EditorToolbarProps) => {
126
- const menuProps = useEditorToolbarActionGraph(props);
127
- return (
128
- <ElevationProvider elevation={role === 'section' ? 'positioned' : 'base'}>
129
- <MenuProvider {...menuProps} attendableId={attendableId}>
130
- <ToolbarMenu classNames={classNames} textBlockWidth />
131
- </MenuProvider>
132
- </ElevationProvider>
133
- );
134
- });
@@ -4,12 +4,18 @@
4
4
 
5
5
  import { type EditorView } from '@codemirror/view';
6
6
 
7
- import { type NodeArg } from '@dxos/app-graph';
8
- import { type ToolbarMenuActionGroupProperties } from '@dxos/react-ui-menu';
7
+ import { type Action, type NodeArg } from '@dxos/app-graph';
8
+ import {
9
+ type MenuActionProperties,
10
+ type ToolbarMenuActionGroupProperties,
11
+ createMenuAction,
12
+ createMenuItemGroup,
13
+ } from '@dxos/react-ui-menu';
9
14
 
10
15
  import { List, addList, removeList } from '../../extensions';
16
+ import { translationKey } from '../../translations';
11
17
 
12
- import { type EditorToolbarState, createEditorAction, createEditorActionGroup } from './util';
18
+ import { type EditorToolbarState } from './useEditorToolbar';
13
19
 
14
20
  const listStyles = {
15
21
  bullet: 'ph--list-bullets--regular',
@@ -17,6 +23,43 @@ const listStyles = {
17
23
  task: 'ph--list-checks--regular',
18
24
  };
19
25
 
26
+ export const createLists = (state: EditorToolbarState, getView: () => EditorView) => {
27
+ const value = state.listStyle ?? '';
28
+ const listGroupAction = createListGroupAction(value);
29
+ const listActionsMap = createListActions(value, getView);
30
+ return {
31
+ nodes: [listGroupAction as NodeArg<any>, ...listActionsMap],
32
+ edges: [
33
+ { source: 'root', target: 'list' },
34
+ ...listActionsMap.map(({ id }) => ({ source: listGroupAction.id, target: id })),
35
+ ],
36
+ };
37
+ };
38
+
39
+ export const createEditorAction = (id: string, props: Partial<MenuActionProperties>, invoke: () => void) => {
40
+ const { label = [`${id} label`, { ns: translationKey }], ...rest } = props;
41
+
42
+ return createMenuAction(id, invoke, {
43
+ label,
44
+ ...rest,
45
+ }) as Action<MenuActionProperties>;
46
+ };
47
+
48
+ export const createEditorActionGroup = (
49
+ id: string,
50
+ props: Omit<ToolbarMenuActionGroupProperties, 'icon'>,
51
+ icon?: string,
52
+ ) => {
53
+ const { label = [`${id} label`, { ns: translationKey }], ...rest } = props;
54
+
55
+ return createMenuItemGroup(id, {
56
+ label,
57
+ icon,
58
+ iconOnly: true,
59
+ ...rest,
60
+ });
61
+ };
62
+
20
63
  const createListGroupAction = (value: string) =>
21
64
  createEditorActionGroup('list', {
22
65
  variant: 'toggleGroup',
@@ -41,16 +84,3 @@ const createListActions = (value: string, getView: () => EditorView) =>
41
84
  }
42
85
  });
43
86
  });
44
-
45
- export const createLists = (state: EditorToolbarState, getView: () => EditorView) => {
46
- const value = state.listStyle ?? '';
47
- const listGroupAction = createListGroupAction(value);
48
- const listActionsMap = createListActions(value, getView);
49
- return {
50
- nodes: [listGroupAction as NodeArg<any>, ...listActionsMap],
51
- edges: [
52
- { source: 'root', target: 'list' },
53
- ...listActionsMap.map(({ id }) => ({ source: listGroupAction.id, target: id })),
54
- ],
55
- };
56
- };
@@ -9,7 +9,8 @@ import { type ToolbarMenuActionGroupProperties } from '@dxos/react-ui-menu';
9
9
 
10
10
  import { addBlockquote, addCodeblock, insertTable, removeBlockquote, removeCodeblock } from '../../extensions';
11
11
 
12
- import { type EditorToolbarState, createEditorAction, createEditorActionGroup } from './util';
12
+ import { createEditorAction, createEditorActionGroup } from './actions';
13
+ import { type EditorToolbarState } from './useEditorToolbar';
13
14
 
14
15
  const createBlockGroupAction = (value: string) =>
15
16
  createEditorActionGroup('block', {
@@ -9,7 +9,8 @@ import { type ToolbarMenuActionGroupProperties } from '@dxos/react-ui-menu';
9
9
 
10
10
  import { type Formatting, Inline, addLink, removeLink, setStyle } from '../../extensions';
11
11
 
12
- import { type EditorToolbarState, createEditorAction, createEditorActionGroup } from './util';
12
+ import { createEditorAction, createEditorActionGroup } from './actions';
13
+ import { type EditorToolbarState } from './useEditorToolbar';
13
14
 
14
15
  const formats = {
15
16
  strong: 'ph--text-b--regular',
@@ -10,7 +10,8 @@ import { type ToolbarMenuActionGroupProperties } from '@dxos/react-ui-menu';
10
10
  import { setHeading } from '../../extensions';
11
11
  import { translationKey } from '../../translations';
12
12
 
13
- import { type EditorToolbarState, createEditorAction, createEditorActionGroup } from './util';
13
+ import { createEditorAction, createEditorActionGroup } from './actions';
14
+ import { type EditorToolbarState } from './useEditorToolbar';
14
15
 
15
16
  const createHeadingGroupAction = (value: string) =>
16
17
  createEditorActionGroup(
@@ -19,12 +20,13 @@ const createHeadingGroupAction = (value: string) =>
19
20
  variant: 'dropdownMenu',
20
21
  applyActive: true,
21
22
  selectCardinality: 'single',
23
+ // TODO(wittjosiah): Remove? Not sure this does anything.
22
24
  value,
23
25
  } as ToolbarMenuActionGroupProperties,
24
26
  'ph--text-h--regular',
25
27
  );
26
28
 
27
- const createHeadingActions = (getView: () => EditorView) =>
29
+ const createHeadingActions = (currentLevel: string, getView: () => EditorView) =>
28
30
  Object.entries({
29
31
  '0': 'ph--paragraph--regular',
30
32
  '1': 'ph--text-h-one--regular',
@@ -40,6 +42,7 @@ const createHeadingActions = (getView: () => EditorView) =>
40
42
  {
41
43
  label: ['heading level label', { count: level, ns: translationKey }],
42
44
  icon,
45
+ checked: levelStr === currentLevel,
43
46
  },
44
47
  () => setHeading(level)(getView()),
45
48
  );
@@ -47,14 +50,14 @@ const createHeadingActions = (getView: () => EditorView) =>
47
50
 
48
51
  const computeHeadingValue = (state: EditorToolbarState) => {
49
52
  const blockType = state ? state.blockType : 'paragraph';
50
- const header = blockType && /heading(\d)/.exec(blockType);
51
- return header ? header[1] : blockType === 'paragraph' || !blockType ? '0' : '';
53
+ const heading = blockType && /heading(\d)/.exec(blockType);
54
+ return heading ? heading[1] : blockType === 'paragraph' || !blockType ? '0' : '';
52
55
  };
53
56
 
54
57
  export const createHeadings = (state: EditorToolbarState, getView: () => EditorView) => {
55
58
  const headingValue = computeHeadingValue(state);
56
59
  const headingGroupAction = createHeadingGroupAction(headingValue);
57
- const headingActions = createHeadingActions(getView);
60
+ const headingActions = createHeadingActions(headingValue, getView);
58
61
  return {
59
62
  nodes: [headingGroupAction as NodeArg<any>, ...headingActions],
60
63
  edges: [
@@ -2,7 +2,7 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { createEditorAction } from './util';
5
+ import { createEditorAction } from './actions';
6
6
 
7
7
  const createImageUploadAction = (onImageUpload: () => void) =>
8
8
  createEditorAction(
@@ -3,10 +3,6 @@
3
3
  //
4
4
 
5
5
  export * from './EditorToolbar';
6
- export {
7
- type EditorToolbarState,
8
- type EditorToolbarActionGraphProps,
9
- useEditorToolbarState,
10
- createEditorAction,
11
- createEditorActionGroup,
12
- } from './util';
6
+
7
+ export { type EditorToolbarState, useEditorToolbar } from './useEditorToolbar';
8
+ export { createEditorAction, createEditorActionGroup } from './actions';
@@ -5,7 +5,7 @@
5
5
  import { openSearchPanel } from '@codemirror/search';
6
6
  import { type EditorView } from '@codemirror/view';
7
7
 
8
- import { createEditorAction } from './util';
8
+ import { createEditorAction } from './actions';
9
9
 
10
10
  const createSearchAction = (getView: () => EditorView) =>
11
11
  createEditorAction(
@@ -0,0 +1,20 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { useMemo } from 'react';
6
+
7
+ import { type Live, live } from '@dxos/live-object';
8
+
9
+ import { type Formatting } from '../../extensions';
10
+ import { type EditorViewMode } from '../../types';
11
+
12
+ // TODO(burdon): Move to extensions.
13
+ export type EditorToolbarState = { viewMode?: EditorViewMode } & Formatting;
14
+
15
+ /**
16
+ * @deprecated Use Editor.Root
17
+ */
18
+ export const useEditorToolbar = (initialState: Partial<EditorToolbarState> = {}): Live<EditorToolbarState> => {
19
+ return useMemo(() => live<EditorToolbarState>(initialState), []);
20
+ };
@@ -8,7 +8,8 @@ import { type ToolbarMenuActionGroupProperties } from '@dxos/react-ui-menu';
8
8
  import { translationKey } from '../../translations';
9
9
  import { type EditorViewMode } from '../../types';
10
10
 
11
- import { type EditorToolbarState, createEditorAction, createEditorActionGroup } from './util';
11
+ import { createEditorAction, createEditorActionGroup } from './actions';
12
+ import { type EditorToolbarState } from './useEditorToolbar';
12
13
 
13
14
  const createViewModeGroupAction = (value: string) =>
14
15
  createEditorActionGroup(
@@ -3,5 +3,11 @@
3
3
  //
4
4
 
5
5
  export * from './Editor';
6
- export * from './EditorToolbar';
7
- export * from './CommandMenu';
6
+
7
+ // export { type GetMenuContext } from './EditorMenuProvider';
8
+
9
+ // TODO(burdon): Remove once Editor is fully migrated.
10
+ export { EditorContent, createEditorController } from './EditorContent';
11
+ export * from './EditorMenuProvider';
12
+ export * from './EditorPreviewProvider';
13
+ export { EditorToolbar, type EditorToolbarProps, type EditorToolbarState, useEditorToolbar } from './EditorToolbar';
@@ -14,33 +14,21 @@ import {
14
14
  } from '@codemirror/view';
15
15
 
16
16
  export type AutocompleteOptions = {
17
- fireIfEmpty?: boolean;
18
-
19
- /**
20
- * Callback triggered when Enter is pressed.
21
- * @param text The current text in the editor
22
- * @returns true if the editor should reset the document.
23
- */
24
- onSubmit?: (text: string) => boolean | void;
25
-
26
17
  /**
27
18
  * Function that returns a list of suggestions based on the current text.
28
19
  * @param text The current text before the cursor
29
20
  * @returns Array of suggestion strings
30
21
  */
31
22
  onSuggest?: (text: string) => string[];
32
-
33
- /**
34
- * ESC pressed.
35
- */
36
- onCancel?: () => void;
37
23
  };
38
24
 
39
25
  /**
40
26
  * Creates an autocomplete extension that shows inline suggestions.
41
27
  * Pressing Tab will complete the suggestion.
28
+ *
29
+ * @deprecated Use typeahead.
42
30
  */
43
- export const autocomplete = ({ fireIfEmpty, onSubmit, onSuggest, onCancel }: AutocompleteOptions): Extension => {
31
+ export const autocomplete = ({ onSuggest }: AutocompleteOptions = {}): Extension => {
44
32
  const suggest = ViewPlugin.fromClass(
45
33
  class {
46
34
  _decorations: DecorationSet;
@@ -140,62 +128,6 @@ export const autocomplete = ({ fireIfEmpty, onSubmit, onSuggest, onCancel }: Aut
140
128
  return plugin?.completeSuggestion(view) ?? false;
141
129
  },
142
130
  },
143
- {
144
- key: 'Enter',
145
- preventDefault: true,
146
- run: (view) => {
147
- const text = view.state.doc.toString().trim();
148
- if (onSubmit && (fireIfEmpty || text.length > 0)) {
149
- const reset = onSubmit(text);
150
-
151
- // Clear the document after calling onEnter.
152
- if (reset) {
153
- view.dispatch({
154
- changes: {
155
- from: 0,
156
- to: view.state.doc.length,
157
- insert: '',
158
- },
159
- });
160
- }
161
- }
162
-
163
- return true;
164
- },
165
- },
166
- {
167
- key: 'Shift-Enter',
168
- preventDefault: true,
169
- run: (view) => {
170
- view.dispatch({
171
- changes: {
172
- from: view.state.selection.main.head,
173
- insert: '\n',
174
- },
175
- selection: {
176
- anchor: view.state.selection.main.head + 1,
177
- head: view.state.selection.main.head + 1,
178
- },
179
- });
180
- return true;
181
- },
182
- },
183
- {
184
- key: 'Escape',
185
- preventDefault: true,
186
- run: (view) => {
187
- // Clear the entire document.
188
- view.dispatch({
189
- changes: {
190
- from: 0,
191
- to: view.state.doc.length,
192
- insert: '',
193
- },
194
- });
195
- onCancel?.();
196
- return true;
197
- },
198
- },
199
131
  ]),
200
132
  ),
201
133
  ];
@@ -206,14 +138,14 @@ class InlineSuggestionWidget extends WidgetType {
206
138
  super();
207
139
  }
208
140
 
209
- override toDOM(): HTMLSpanElement {
141
+ override eq(other: this) {
142
+ return this.suffix === other.suffix;
143
+ }
144
+
145
+ override toDOM() {
210
146
  const span = document.createElement('span');
211
147
  span.textContent = this.suffix;
212
148
  span.className = 'cm-inline-suggestion';
213
149
  return span;
214
150
  }
215
-
216
- override eq(other: InlineSuggestionWidget): boolean {
217
- return other.suffix === this.suffix;
218
- }
219
151
  }
@@ -0,0 +1,8 @@
1
+ //
2
+ // Copyright 2023 DXOS.org
3
+ //
4
+
5
+ export * from './autocomplete';
6
+ export * from './match';
7
+ export * from './placeholder';
8
+ export * from './typeahead';
@@ -0,0 +1,46 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export type CompoetionContext = { line: string };
6
+
7
+ export type CompletionOptions = {
8
+ default?: string;
9
+ minLength?: number;
10
+ };
11
+
12
+ /**
13
+ * Util to match current line to a static list of completions.
14
+ */
15
+ export const staticCompletion =
16
+ (completions: string[], options: CompletionOptions = {}) =>
17
+ ({ line }: CompoetionContext) => {
18
+ if (line.length === 0 && options.default) {
19
+ return options.default;
20
+ }
21
+
22
+ const parts = line.split(/\s+/).filter(Boolean);
23
+ if (parts.length) {
24
+ const str = parts.at(-1)!;
25
+ if (str.length >= (options.minLength ?? 0)) {
26
+ for (const completion of completions) {
27
+ const match = matchCompletion(completion, str);
28
+ if (match) {
29
+ return match;
30
+ }
31
+ }
32
+ }
33
+ }
34
+ };
35
+
36
+ export const matchCompletion = (completion: string, str: string, minLength = 0): string | undefined => {
37
+ if (
38
+ str.length >= minLength &&
39
+ completion.length > str.length &&
40
+ completion.startsWith(str)
41
+ // TODO(burdon): If case insensitive, need to replace existing chars.
42
+ // completion.toLowerCase().startsWith(str.toLowerCase())
43
+ ) {
44
+ return completion.slice(str.length);
45
+ }
46
+ };