@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
@@ -11,20 +11,23 @@ import { clientRectsFor, flattenRect } from '../../util';
11
11
  type Content = string | HTMLElement | ((view: EditorView) => HTMLElement);
12
12
 
13
13
  export type PlaceholderOptions = {
14
- delay?: number;
15
14
  content: Content;
15
+ delay?: number;
16
16
  };
17
17
 
18
- export const placeholder = ({ delay = 3_000, content }: PlaceholderOptions): Extension => {
18
+ /**
19
+ * Shows a transient placeholder at the current cursor position.
20
+ */
21
+ export const placeholder = ({ content, delay = 3_000 }: PlaceholderOptions): Extension => {
19
22
  const plugin = ViewPlugin.fromClass(
20
23
  class {
21
- decorations = Decoration.none;
22
- timeout: ReturnType<typeof setTimeout> | undefined;
24
+ _timeout: ReturnType<typeof setTimeout> | undefined;
25
+ _decorations = Decoration.none;
23
26
 
24
27
  update(update: ViewUpdate) {
25
- if (this.timeout) {
26
- window.clearTimeout(this.timeout);
27
- this.timeout = undefined;
28
+ if (this._timeout) {
29
+ window.clearTimeout(this._timeout);
30
+ this._timeout = undefined;
28
31
  }
29
32
 
30
33
  // Check if the active line (where cursor is) is empty.
@@ -33,10 +36,10 @@ export const placeholder = ({ delay = 3_000, content }: PlaceholderOptions): Ext
33
36
  if (isEmpty) {
34
37
  // Create widget decoration at the start of the current line.
35
38
  const lineStart = activeLine.from;
36
- this.timeout = setTimeout(() => {
37
- this.decorations = Decoration.set([
39
+ this._timeout = setTimeout(() => {
40
+ this._decorations = Decoration.set([
38
41
  Decoration.widget({
39
- widget: new Placeholder(content),
42
+ widget: new PlaceholderWidget(content),
40
43
  side: 1,
41
44
  }).range(lineStart),
42
45
  ]);
@@ -45,18 +48,18 @@ export const placeholder = ({ delay = 3_000, content }: PlaceholderOptions): Ext
45
48
  }, delay);
46
49
  }
47
50
 
48
- this.decorations = Decoration.none;
51
+ this._decorations = Decoration.none;
49
52
  }
50
53
 
51
54
  destroy() {
52
- if (this.timeout) {
53
- clearTimeout(this.timeout);
55
+ if (this._timeout) {
56
+ clearTimeout(this._timeout);
54
57
  }
55
58
  }
56
59
  },
57
60
  {
58
61
  provide: (plugin) => {
59
- return [EditorView.decorations.of((view) => view.plugin(plugin)?.decorations ?? Decoration.none)];
62
+ return [EditorView.decorations.of((view) => view.plugin(plugin)?._decorations ?? Decoration.none)];
60
63
  },
61
64
  },
62
65
  );
@@ -66,7 +69,7 @@ export const placeholder = ({ delay = 3_000, content }: PlaceholderOptions): Ext
66
69
  : plugin;
67
70
  };
68
71
 
69
- class Placeholder extends WidgetType {
72
+ export class PlaceholderWidget extends WidgetType {
70
73
  constructor(readonly content: Content) {
71
74
  super();
72
75
  }
@@ -75,6 +78,7 @@ class Placeholder extends WidgetType {
75
78
  const wrap = document.createElement('span');
76
79
  wrap.className = 'cm-placeholder';
77
80
  wrap.style.pointerEvents = 'none';
81
+ wrap.setAttribute('aria-hidden', 'true');
78
82
  wrap.appendChild(
79
83
  typeof this.content === 'string'
80
84
  ? document.createTextNode(this.content)
@@ -82,7 +86,7 @@ class Placeholder extends WidgetType {
82
86
  ? this.content(view)
83
87
  : this.content.cloneNode(true),
84
88
  );
85
- wrap.setAttribute('aria-hidden', 'true');
89
+
86
90
  return wrap;
87
91
  }
88
92
 
@@ -92,7 +96,7 @@ class Placeholder extends WidgetType {
92
96
  return null;
93
97
  }
94
98
 
95
- const style = window.getComputedStyle(dom.parentNode as HTMLElement);
99
+ const style = getComputedStyle(dom.parentNode as HTMLElement);
96
100
  const rect = flattenRect(rects[0], style.direction !== 'rtl');
97
101
  const lineHeight = parseInt(style.lineHeight);
98
102
  if (rect.bottom - rect.top > lineHeight * 1.5) {
@@ -13,17 +13,16 @@ import {
13
13
  keymap,
14
14
  } from '@codemirror/view';
15
15
 
16
- import { Hint } from './hint';
16
+ import { type CompoetionContext } from './match';
17
+ import { PlaceholderWidget } from './placeholder';
17
18
 
18
- export type TypeaheadContext = { line: string };
19
-
20
- // TODO(burdon): Option to complete only at end of line?
19
+ // TODO(burdon): Option to complete only at end of line.
21
20
  export type TypeaheadOptions = {
22
- onComplete?: (context: TypeaheadContext) => string | undefined;
21
+ onComplete?: (context: CompoetionContext) => string | undefined;
23
22
  };
24
23
 
25
24
  /**
26
- * CodeMirror extension for typeahead completion.
25
+ * Shows a completion placeholder.
27
26
  */
28
27
  export const typeahead = ({ onComplete }: TypeaheadOptions = {}): Extension => {
29
28
  let hint: string | undefined;
@@ -57,7 +56,7 @@ export const typeahead = ({ onComplete }: TypeaheadOptions = {}): Extension => {
57
56
  const str = update.state.sliceDoc(line.from, selection.from);
58
57
  hint = onComplete?.({ line: str });
59
58
  if (hint) {
60
- builder.add(selection.from, selection.to, Decoration.widget({ widget: new Hint(hint) }));
59
+ builder.add(selection.from, selection.to, Decoration.widget({ widget: new PlaceholderWidget(hint) }));
61
60
  }
62
61
  }
63
62
 
@@ -86,44 +85,3 @@ export const typeahead = ({ onComplete }: TypeaheadOptions = {}): Extension => {
86
85
  ),
87
86
  ];
88
87
  };
89
-
90
- type CompletionOptions = {
91
- default?: string;
92
- minLength?: number;
93
- };
94
-
95
- /**
96
- * Util to match current line to a static list of completions.
97
- */
98
- export const staticCompletion =
99
- (completions: string[], options: CompletionOptions = {}) =>
100
- ({ line }: TypeaheadContext) => {
101
- if (line.length === 0 && options.default) {
102
- return options.default;
103
- }
104
-
105
- const parts = line.split(/\s+/).filter(Boolean);
106
- if (parts.length) {
107
- const str = parts.at(-1)!;
108
- if (str.length >= (options.minLength ?? 0)) {
109
- for (const completion of completions) {
110
- const match = matchCompletion(completion, str);
111
- if (match) {
112
- return match;
113
- }
114
- }
115
- }
116
- }
117
- };
118
-
119
- export const matchCompletion = (completion: string, str: string, minLength = 0): string | undefined => {
120
- if (
121
- str.length >= minLength &&
122
- completion.length > str.length &&
123
- completion.startsWith(str)
124
- // TODO(burdon): If case insensitive, need to replace existing chars.
125
- // completion.toLowerCase().startsWith(str.toLowerCase())
126
- ) {
127
- return completion.slice(str.length);
128
- }
129
- };
@@ -2,8 +2,6 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
7
5
  import '@preact/signals-react';
8
6
 
9
7
  import { Repo } from '@automerge/automerge-repo';
@@ -16,7 +14,8 @@ import { DocAccessor, Query, type Space, createDocAccessor, useQuery, useSpace }
16
14
  import { type Identity, useIdentity } from '@dxos/react-client/halo';
17
15
  import { type ClientRepeatedComponentProps, ClientRepeater } from '@dxos/react-client/testing';
18
16
  import { useThemeContext } from '@dxos/react-ui';
19
- import { render, withLayout, withTheme } from '@dxos/storybook-utils';
17
+ import { withTheme } from '@dxos/react-ui/testing';
18
+ import { render } from '@dxos/storybook-utils';
20
19
 
21
20
  import { editorSlots } from '../../defaults';
22
21
  import { useTextEditor } from '../../hooks';
@@ -51,7 +50,7 @@ const Editor = ({ source, autoFocus, space, identity }: EditorProps) => {
51
50
  [source, themeMode],
52
51
  );
53
52
 
54
- return <div ref={parentRef} className='flex w-full' />;
53
+ return <div ref={parentRef} className='flex is-full' />;
55
54
  };
56
55
 
57
56
  const DefaultStory = () => {
@@ -96,8 +95,9 @@ const EchoStory = ({ spaceKey }: ClientRepeatedComponentProps) => {
96
95
  const objects = useQuery(space, Query.type(Type.Expando, { type: 'test' }));
97
96
 
98
97
  useEffect(() => {
99
- if (!source && objects.length) {
100
- const source = createDocAccessor(objects[0].content, ['content']);
98
+ const content = objects[0]?.content.target;
99
+ if (!source && content) {
100
+ const source = createDocAccessor(content, ['content']);
101
101
  setSource(source);
102
102
  }
103
103
  }, [objects, source]);
@@ -113,8 +113,9 @@ const meta = {
113
113
  title: 'ui/react-ui-editor/Automerge',
114
114
  component: Editor as any,
115
115
  render: render(DefaultStory),
116
- decorators: [withTheme, withLayout({ fullscreen: true })],
116
+ decorators: [withTheme],
117
117
  parameters: {
118
+ layout: 'fullscreen',
118
119
  translations,
119
120
  },
120
121
  } satisfies Meta<typeof DefaultStory>;
@@ -128,14 +129,13 @@ export const Default: Story = {
128
129
  };
129
130
 
130
131
  export const WithEcho: Story = {
131
- decorators: [withTheme],
132
132
  render: () => {
133
133
  return (
134
134
  <ClientRepeater
135
135
  count={2}
136
136
  component={EchoStory}
137
137
  createSpace
138
- onSpaceCreated={async ({ space }) => {
138
+ onCreateSpace={async ({ space }) => {
139
139
  space.db.add(
140
140
  Obj.make(Type.Expando, {
141
141
  type: 'test',
@@ -5,12 +5,13 @@
5
5
  //
6
6
 
7
7
  import { next as A } from '@automerge/automerge';
8
- import { type Extension, StateField } from '@codemirror/state';
8
+ import { type Extension, StateField, Transaction } from '@codemirror/state';
9
9
  import { EditorView, ViewPlugin } from '@codemirror/view';
10
10
 
11
- import { type DocAccessor } from '@dxos/react-client/echo';
11
+ import { DocAccessor } from '@dxos/client/echo';
12
12
 
13
13
  import { Cursor } from '../../util';
14
+ import { initialSync } from '../state';
14
15
 
15
16
  import { cursorConverter } from './cursor';
16
17
  import { type State, isReconcile, updateHeadsEffect } from './defs';
@@ -18,11 +19,13 @@ import { Syncer } from './sync';
18
19
 
19
20
  export const automerge = (accessor: DocAccessor): Extension => {
20
21
  const syncState = StateField.define<State>({
21
- create: () => ({
22
- path: accessor.path.slice(),
23
- lastHeads: A.getHeads(accessor.handle.doc()!),
24
- unreconciledTransactions: [],
25
- }),
22
+ create: () => {
23
+ return {
24
+ path: accessor.path.slice(),
25
+ lastHeads: A.getHeads(accessor.handle.doc()!),
26
+ unreconciledTransactions: [],
27
+ };
28
+ },
26
29
 
27
30
  update: (value, tr) => {
28
31
  const result: State = {
@@ -64,6 +67,18 @@ export const automerge = (accessor: DocAccessor): Extension => {
64
67
  class {
65
68
  constructor(private readonly _view: EditorView) {
66
69
  accessor.handle.addListener('change', this._handleChange);
70
+
71
+ requestAnimationFrame(() => {
72
+ const value = DocAccessor.getValue<string>(accessor);
73
+ const current = this._view.state.doc.toString();
74
+ if (value !== current) {
75
+ // TODO(burdon): This attempts to set the initial state, but creates problems.
76
+ // this._view.dispatch({
77
+ // changes: { from: 0, to: this._view.state.doc.length, insert: value },
78
+ // annotations: initialSync,
79
+ // });
80
+ }
81
+ });
67
82
  }
68
83
 
69
84
  destroy() {
@@ -77,9 +92,13 @@ export const automerge = (accessor: DocAccessor): Extension => {
77
92
  ),
78
93
 
79
94
  // Reconcile local updates.
80
- EditorView.updateListener.of(({ view, changes }) => {
95
+ EditorView.updateListener.of(({ view, changes, transactions }) => {
81
96
  if (!changes.empty) {
82
- syncer.reconcile(view, true);
97
+ // Only reconcile if it's not an initial sync (to avoid loops)
98
+ const isInitialSync = transactions.some((tr) => tr.annotation(Transaction.userEvent) === initialSync.value);
99
+ if (!isInitialSync) {
100
+ syncer.reconcile(view, true);
101
+ }
83
102
  }
84
103
  }),
85
104
  ];
@@ -2,8 +2,8 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
+ import { type DocAccessor, fromCursor, toCursor } from '@dxos/client/echo';
5
6
  import { log } from '@dxos/log';
6
- import { type DocAccessor, fromCursor, toCursor } from '@dxos/react-client/echo';
7
7
 
8
8
  import { type CursorConverter } from '../../util';
9
9
 
@@ -8,7 +8,8 @@ import { next as A } from '@automerge/automerge';
8
8
  import { type StateField } from '@codemirror/state';
9
9
  import { type EditorView } from '@codemirror/view';
10
10
 
11
- import { type IDocHandle } from '@dxos/react-client/echo';
11
+ import { type IDocHandle } from '@dxos/client/echo';
12
+ import { log } from '@dxos/log';
12
13
 
13
14
  import { type State, getLastHeads, getPath, isReconcile, reconcileAnnotation, updateHeads } from './defs';
14
15
  import { updateAutomerge } from './update-automerge';
@@ -27,7 +28,6 @@ export class Syncer {
27
28
  ) {}
28
29
 
29
30
  reconcile(view: EditorView, editor: boolean): void {
30
- // TODO(burdon): Better way to do mutex?
31
31
  if (this._pending) {
32
32
  return;
33
33
  }
@@ -41,7 +41,9 @@ export class Syncer {
41
41
  this._pending = false;
42
42
  }
43
43
 
44
- onEditorChange(view: EditorView): void {
44
+ private onEditorChange(view: EditorView): void {
45
+ log('onEditorChange');
46
+
45
47
  // Apply the unreconciled transactions to the document.
46
48
  const transactions = view.state.field(this._state).unreconciledTransactions.filter((tx) => !isReconcile(tx));
47
49
  const newHeads = updateAutomerge(this._state, this._handle, transactions, view.state);
@@ -54,7 +56,9 @@ export class Syncer {
54
56
  }
55
57
  }
56
58
 
57
- onAutomergeChange(view: EditorView): void {
59
+ private onAutomergeChange(view: EditorView): void {
60
+ log('onAutomergeChange');
61
+
58
62
  // Get the diff between the updated state of the document and the heads and apply that to the codemirror doc.
59
63
  const oldHeads = getLastHeads(view.state, this._state);
60
64
  const newHeads = A.getHeads(this._handle.doc()!);
@@ -7,7 +7,7 @@
7
7
  import { next as A, type Heads } from '@automerge/automerge';
8
8
  import { type EditorState, type StateField, type Text, type Transaction } from '@codemirror/state';
9
9
 
10
- import { type IDocHandle } from '@dxos/react-client/echo';
10
+ import { type IDocHandle } from '@dxos/client/echo';
11
11
 
12
12
  import { type State } from './defs';
13
13
 
@@ -5,130 +5,146 @@
5
5
  import { StateEffect } from '@codemirror/state';
6
6
  import { EditorView, ViewPlugin } from '@codemirror/view';
7
7
 
8
- import { Domino } from '../util';
8
+ import { debounce } from '@dxos/async';
9
+ import { Domino } from '@dxos/react-ui';
9
10
 
10
- const lineHeight = 24;
11
+ import { scrollToLineEffect } from './scrolling';
11
12
 
12
- export const scrollToBottomEffect = StateEffect.define<any>();
13
+ // TODO(burdon): Reconcile with scrollToLineEffect (scrolling).
14
+ export const scrollToBottomEffect = StateEffect.define<ScrollBehavior | undefined>();
13
15
 
14
16
  export type AutoScrollOptions = {
15
- overscroll: number;
16
- throttle: number;
17
+ /** Auto-scroll when reaches the bottom. */
18
+ autoScroll?: boolean;
19
+ /** Threshold in px to trigger scroll from bottom. */
20
+ threshold?: number;
21
+ /** Throttle time in ms. */
22
+ throttleDelay?: number;
23
+ /** Callback when auto-scrolling. */
24
+ onAutoScroll?: (props: { view: EditorView; distanceFromBottom: number }) => boolean | void;
17
25
  };
18
26
 
19
27
  /**
20
28
  * Extension that supports pinning the scroll position and automatically scrolls to the bottom when content is added.
21
29
  */
22
30
  // TODO(burdon): Reconcile with transcript-extension.
23
- export const autoScroll = ({ overscroll = 4 * lineHeight, throttle = 2_000 }: Partial<AutoScrollOptions> = {}) => {
24
- let isThrottled = false;
25
- let isPinned = true;
26
- let timeout: NodeJS.Timeout | undefined;
27
- let buttonContainer: HTMLDivElement;
31
+ export const autoScroll = ({
32
+ autoScroll = true,
33
+ threshold = 100,
34
+ throttleDelay = 1_000,
35
+ onAutoScroll,
36
+ }: Partial<AutoScrollOptions> = {}) => {
37
+ let buttonContainer: HTMLDivElement | undefined;
38
+ let hideTimeout: NodeJS.Timeout | undefined;
28
39
  let lastScrollTop = 0;
29
- let scrollCounter = 0;
40
+ let isPinned = true;
41
+
42
+ const setPinned = (pin: boolean) => {
43
+ isPinned = pin;
44
+ buttonContainer?.classList.toggle('opacity-0', pin);
45
+ };
30
46
 
47
+ // Temporarily hide the scrollbar while auto-scrolling.
31
48
  const hideScrollbar = (view: EditorView) => {
32
49
  view.scrollDOM.classList.add('cm-hide-scrollbar');
33
- clearTimeout(timeout);
34
- timeout = setTimeout(() => {
50
+ clearTimeout(hideTimeout);
51
+ hideTimeout = setTimeout(() => {
35
52
  view.scrollDOM.classList.remove('cm-hide-scrollbar');
36
53
  }, 1_000);
37
54
  };
38
55
 
39
- const scrollToBottom = (view: EditorView) => {
40
- isPinned = true;
41
- scrollCounter = 0;
42
- buttonContainer?.classList.add('opacity-0');
43
- requestAnimationFrame(() => {
44
- hideScrollbar(view);
45
- view.scrollDOM.scrollTo({
46
- top: view.scrollDOM.scrollHeight,
47
- behavior: 'smooth',
48
- });
56
+ // Throttled scroll to bottom.
57
+ const scrollToBottom = (view: EditorView, behavior?: ScrollBehavior) => {
58
+ setPinned(true);
59
+ hideScrollbar(view);
60
+ const line = view.state.doc.lineAt(view.state.doc.length);
61
+ view.dispatch({
62
+ selection: { anchor: line.to, head: line.to },
63
+ effects: scrollToLineEffect.of({ line: line.number, options: { position: 'end', offset: threshold, behavior } }),
49
64
  });
50
65
  };
51
66
 
52
- return [
53
- // Scroll button.
54
- ViewPlugin.fromClass(
55
- class {
56
- constructor(view: EditorView) {
57
- const scroller = view.scrollDOM.parentElement;
58
- buttonContainer = Domino.of('div')
59
- .classNames(true && 'cm-scroll-button transition-opacity duration-300 opacity-0')
60
- .child(
61
- Domino.of('button')
62
- .classNames('dx-button bg-accentSurface')
63
- .data('density', 'fine')
64
- .child(Domino.of<any>('dx-icon').attr('icon', 'ph--arrow-down--regular'))
65
- .on('click', () => {
66
- scrollToBottom(view);
67
- }),
68
- )
69
- .build();
70
- scroller?.appendChild(buttonContainer);
71
- }
72
- },
73
- ),
67
+ // Throttled check for distance from bottom (for downward scrolls only).
68
+ const checkDistance = debounce((view: EditorView) => {
69
+ const scrollerRect = view.scrollDOM.getBoundingClientRect();
70
+ const coords = view.coordsAtPos(view.state.doc.length);
71
+ const distanceFromBottom = coords ? coords.bottom - scrollerRect.bottom : 0;
72
+ setPinned(distanceFromBottom < 0);
73
+ }, 1_000);
74
74
 
75
+ // Debounce scroll updates so rapid edits don't cause clunky scrolling.
76
+ const triggerUpdate = debounce((view: EditorView) => scrollToBottom(view), throttleDelay);
77
+
78
+ return [
75
79
  // Update listener for logging when scrolling is needed.
76
- EditorView.updateListener.of((update) => {
77
- // Listen for effects.
78
- update.transactions.forEach((transaction) => {
80
+ EditorView.updateListener.of(({ view, transactions, heightChanged }) => {
81
+ // TODO(burdon): Remove and use scrollToLineEffect instead.
82
+ transactions.forEach((transaction) => {
79
83
  for (const effect of transaction.effects) {
80
84
  if (effect.is(scrollToBottomEffect)) {
81
- scrollToBottom(update.view);
85
+ scrollToBottom(view, effect.value);
82
86
  }
83
87
  }
84
88
  });
85
89
 
86
- if (update.docChanged && isPinned && !isThrottled) {
87
- const distanceFromBottom = calcDistance(update.view.scrollDOM);
88
-
89
- // Keep pinned.
90
- if (distanceFromBottom >= overscroll) {
91
- isThrottled = true;
92
- requestAnimationFrame(() => {
93
- scrollToBottom(update.view);
94
- });
95
-
96
- // Reset throttle.
97
- setTimeout(() => {
98
- isThrottled = false;
99
- }, throttle);
90
+ // Maybe scroll if doc changed and pinned.
91
+ // NOTE: Geometry changed is triggered when widgets change height (e.g., toggle tool block).
92
+ if (heightChanged && isPinned) {
93
+ const coords = view.coordsAtPos(view.state.doc.length);
94
+ const scrollerRect = view.scrollDOM.getBoundingClientRect();
95
+ const distanceFromBottom = coords ? scrollerRect.bottom - coords.bottom : 0;
96
+ if (autoScroll && distanceFromBottom < threshold) {
97
+ const shouldScroll = onAutoScroll?.({ view, distanceFromBottom }) ?? true;
98
+ if (shouldScroll) {
99
+ triggerUpdate(view);
100
+ }
101
+ } else if (distanceFromBottom < 0) {
102
+ setPinned(false);
100
103
  }
101
104
  }
102
105
  }),
103
106
 
104
107
  // Detect user scroll.
105
- // NOTE: Multiple scroll events are triggered during programmatic smooth scrolling.
106
108
  EditorView.domEventHandlers({
107
109
  scroll: (event, view) => {
108
- const scroller = view.scrollDOM;
109
- // Suspect delta goes positive when rendering widgets, so count positive deltas.
110
- // TODO(burdon): Detect user scroll directly (wheel, touch, keys, etc.)
111
- if (lastScrollTop > scroller.scrollTop) {
112
- scrollCounter++;
113
- }
114
- lastScrollTop = scroller.scrollTop;
115
- const distanceFromBottom = calcDistance(scroller);
116
- if (distanceFromBottom === 0) {
117
- // Pin to bottom.
118
- isPinned = true;
119
- buttonContainer?.classList.add('opacity-0');
120
- scrollCounter = 0;
121
- } else if (scrollCounter > 3) {
122
- // Break pin if user scrolls up.
123
- isPinned = false;
124
- buttonContainer?.classList.remove('opacity-0');
110
+ const currentScrollTop = view.scrollDOM.scrollTop;
111
+ const scrollingUp = currentScrollTop < lastScrollTop;
112
+ lastScrollTop = currentScrollTop;
113
+
114
+ // If user scrolls up, immediately unpin auto-scroll.
115
+ if (scrollingUp) {
116
+ setPinned(false);
117
+ } else {
118
+ checkDistance(view);
125
119
  }
126
120
  },
127
121
  }),
128
122
 
123
+ // Scroll button.
124
+ ViewPlugin.fromClass(
125
+ class {
126
+ constructor(view: EditorView) {
127
+ buttonContainer = Domino.of('div')
128
+ .classNames(true && 'cm-scroll-button transition-opacity duration-300 opacity-0')
129
+ .children(
130
+ Domino.of('button')
131
+ .classNames('dx-button bg-accentSurface')
132
+ .data('density', 'fine')
133
+ .children(Domino.of<any>('dx-icon').attributes({ icon: 'ph--arrow-down--regular' }))
134
+ .on('click', () => {
135
+ scrollToBottom(view);
136
+ }),
137
+ )
138
+ .build();
139
+
140
+ view.scrollDOM.parentElement!.appendChild(buttonContainer);
141
+ }
142
+ },
143
+ ),
144
+
145
+ // Styles.
129
146
  EditorView.theme({
130
147
  '.cm-scroller': {
131
- paddingBottom: `${overscroll}px`,
132
148
  scrollbarWidth: 'thin',
133
149
  },
134
150
  '.cm-scroller.cm-hide-scrollbar': {
@@ -137,8 +153,6 @@ export const autoScroll = ({ overscroll = 4 * lineHeight, throttle = 2_000 }: Pa
137
153
  '.cm-scroller.cm-hide-scrollbar::-webkit-scrollbar': {
138
154
  display: 'none',
139
155
  },
140
-
141
- // TODO(burdon): IconButton.
142
156
  '.cm-scroll-button': {
143
157
  position: 'absolute',
144
158
  bottom: '0.5rem',
@@ -147,11 +161,3 @@ export const autoScroll = ({ overscroll = 4 * lineHeight, throttle = 2_000 }: Pa
147
161
  }),
148
162
  ];
149
163
  };
150
-
151
- const calcDistance = (scroller: HTMLElement) => {
152
- const scrollTop = scroller.scrollTop;
153
- const scrollHeight = scroller.scrollHeight;
154
- const clientHeight = scroller.clientHeight;
155
- const distanceFromBottom = scrollHeight - (scrollTop + clientHeight);
156
- return distanceFromBottom;
157
- };
@@ -3,11 +3,11 @@
3
3
  //
4
4
 
5
5
  import { DeferredTask, Event, sleep } from '@dxos/async';
6
+ import { type Space } from '@dxos/client/echo';
7
+ import { type GossipMessage } from '@dxos/client/mesh';
6
8
  import { Context } from '@dxos/context';
7
9
  import { invariant } from '@dxos/invariant';
8
10
  import { log } from '@dxos/log';
9
- import { type Space } from '@dxos/react-client/echo';
10
- import { type GossipMessage } from '@dxos/react-client/mesh';
11
11
 
12
12
  import { type AwarenessInfo, type AwarenessPosition, type AwarenessProvider, type AwarenessState } from './awareness';
13
13