@dxos/ui-editor 0.0.0 → 0.8.4-main.05e74ebcff

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 (263) hide show
  1. package/LICENSE +102 -5
  2. package/README.md +1 -1
  3. package/dist/lib/browser/index.mjs +8657 -0
  4. package/dist/lib/browser/index.mjs.map +7 -0
  5. package/dist/lib/browser/meta.json +1 -0
  6. package/dist/lib/browser/types/index.mjs +33 -0
  7. package/dist/lib/browser/types/index.mjs.map +7 -0
  8. package/dist/lib/node-esm/index.mjs +8659 -0
  9. package/dist/lib/node-esm/index.mjs.map +7 -0
  10. package/dist/lib/node-esm/meta.json +1 -0
  11. package/dist/lib/node-esm/types/index.mjs +35 -0
  12. package/dist/lib/node-esm/types/index.mjs.map +7 -0
  13. package/dist/types/src/defaults.d.ts +6 -0
  14. package/dist/types/src/defaults.d.ts.map +1 -0
  15. package/dist/types/src/extensions/annotations.d.ts +9 -0
  16. package/dist/types/src/extensions/annotations.d.ts.map +1 -0
  17. package/dist/types/src/extensions/autocomplete/autocomplete.d.ts +17 -0
  18. package/dist/types/src/extensions/autocomplete/autocomplete.d.ts.map +1 -0
  19. package/dist/types/src/extensions/autocomplete/index.d.ts +5 -0
  20. package/dist/types/src/extensions/autocomplete/index.d.ts.map +1 -0
  21. package/dist/types/src/extensions/autocomplete/match.d.ts +13 -0
  22. package/dist/types/src/extensions/autocomplete/match.d.ts.map +1 -0
  23. package/dist/types/src/extensions/autocomplete/placeholder.d.ts +23 -0
  24. package/dist/types/src/extensions/autocomplete/placeholder.d.ts.map +1 -0
  25. package/dist/types/src/extensions/autocomplete/typeahead.d.ts +10 -0
  26. package/dist/types/src/extensions/autocomplete/typeahead.d.ts.map +1 -0
  27. package/dist/types/src/extensions/automerge/automerge.d.ts +4 -0
  28. package/dist/types/src/extensions/automerge/automerge.d.ts.map +1 -0
  29. package/dist/types/src/extensions/automerge/automerge.test.d.ts +2 -0
  30. package/dist/types/src/extensions/automerge/automerge.test.d.ts.map +1 -0
  31. package/dist/types/src/extensions/automerge/cursor.d.ts +4 -0
  32. package/dist/types/src/extensions/automerge/cursor.d.ts.map +1 -0
  33. package/dist/types/src/extensions/automerge/defs.d.ts +17 -0
  34. package/dist/types/src/extensions/automerge/defs.d.ts.map +1 -0
  35. package/dist/types/src/extensions/automerge/index.d.ts +2 -0
  36. package/dist/types/src/extensions/automerge/index.d.ts.map +1 -0
  37. package/dist/types/src/extensions/automerge/sync.d.ts +17 -0
  38. package/dist/types/src/extensions/automerge/sync.d.ts.map +1 -0
  39. package/dist/types/src/extensions/automerge/update-automerge.d.ts +6 -0
  40. package/dist/types/src/extensions/automerge/update-automerge.d.ts.map +1 -0
  41. package/dist/types/src/extensions/automerge/update-codemirror.d.ts +5 -0
  42. package/dist/types/src/extensions/automerge/update-codemirror.d.ts.map +1 -0
  43. package/dist/types/src/extensions/awareness/awareness-provider.d.ts +31 -0
  44. package/dist/types/src/extensions/awareness/awareness-provider.d.ts.map +1 -0
  45. package/dist/types/src/extensions/awareness/awareness.d.ts +46 -0
  46. package/dist/types/src/extensions/awareness/awareness.d.ts.map +1 -0
  47. package/dist/types/src/extensions/awareness/index.d.ts +3 -0
  48. package/dist/types/src/extensions/awareness/index.d.ts.map +1 -0
  49. package/dist/types/src/extensions/blast.d.ts +25 -0
  50. package/dist/types/src/extensions/blast.d.ts.map +1 -0
  51. package/dist/types/src/extensions/blocks.d.ts +2 -0
  52. package/dist/types/src/extensions/blocks.d.ts.map +1 -0
  53. package/dist/types/src/extensions/bookmarks.d.ts +12 -0
  54. package/dist/types/src/extensions/bookmarks.d.ts.map +1 -0
  55. package/dist/types/src/extensions/comments.d.ts +90 -0
  56. package/dist/types/src/extensions/comments.d.ts.map +1 -0
  57. package/dist/types/src/extensions/debug.d.ts +3 -0
  58. package/dist/types/src/extensions/debug.d.ts.map +1 -0
  59. package/dist/types/src/extensions/dnd.d.ts +9 -0
  60. package/dist/types/src/extensions/dnd.d.ts.map +1 -0
  61. package/dist/types/src/extensions/factories.d.ts +88 -0
  62. package/dist/types/src/extensions/factories.d.ts.map +1 -0
  63. package/dist/types/src/extensions/factories.test.d.ts +2 -0
  64. package/dist/types/src/extensions/factories.test.d.ts.map +1 -0
  65. package/dist/types/src/extensions/focus.d.ts +7 -0
  66. package/dist/types/src/extensions/focus.d.ts.map +1 -0
  67. package/dist/types/src/extensions/folding.d.ts +6 -0
  68. package/dist/types/src/extensions/folding.d.ts.map +1 -0
  69. package/dist/types/src/extensions/hashtag.d.ts +3 -0
  70. package/dist/types/src/extensions/hashtag.d.ts.map +1 -0
  71. package/dist/types/src/extensions/index.d.ts +30 -0
  72. package/dist/types/src/extensions/index.d.ts.map +1 -0
  73. package/dist/types/src/extensions/json.d.ts +7 -0
  74. package/dist/types/src/extensions/json.d.ts.map +1 -0
  75. package/dist/types/src/extensions/listener.d.ts +13 -0
  76. package/dist/types/src/extensions/listener.d.ts.map +1 -0
  77. package/dist/types/src/extensions/markdown/action.d.ts +12 -0
  78. package/dist/types/src/extensions/markdown/action.d.ts.map +1 -0
  79. package/dist/types/src/extensions/markdown/bundle.d.ts +25 -0
  80. package/dist/types/src/extensions/markdown/bundle.d.ts.map +1 -0
  81. package/dist/types/src/extensions/markdown/changes.d.ts +10 -0
  82. package/dist/types/src/extensions/markdown/changes.d.ts.map +1 -0
  83. package/dist/types/src/extensions/markdown/changes.test.d.ts +2 -0
  84. package/dist/types/src/extensions/markdown/changes.test.d.ts.map +1 -0
  85. package/dist/types/src/extensions/markdown/debug.d.ts +11 -0
  86. package/dist/types/src/extensions/markdown/debug.d.ts.map +1 -0
  87. package/dist/types/src/extensions/markdown/decorate.d.ts +25 -0
  88. package/dist/types/src/extensions/markdown/decorate.d.ts.map +1 -0
  89. package/dist/types/src/extensions/markdown/formatting.d.ts +63 -0
  90. package/dist/types/src/extensions/markdown/formatting.d.ts.map +1 -0
  91. package/dist/types/src/extensions/markdown/formatting.test.d.ts +3 -0
  92. package/dist/types/src/extensions/markdown/formatting.test.d.ts.map +1 -0
  93. package/dist/types/src/extensions/markdown/highlight.d.ts +37 -0
  94. package/dist/types/src/extensions/markdown/highlight.d.ts.map +1 -0
  95. package/dist/types/src/extensions/markdown/image.d.ts +7 -0
  96. package/dist/types/src/extensions/markdown/image.d.ts.map +1 -0
  97. package/dist/types/src/extensions/markdown/index.d.ts +10 -0
  98. package/dist/types/src/extensions/markdown/index.d.ts.map +1 -0
  99. package/dist/types/src/extensions/markdown/link.d.ts +7 -0
  100. package/dist/types/src/extensions/markdown/link.d.ts.map +1 -0
  101. package/dist/types/src/extensions/markdown/parser.test.d.ts +2 -0
  102. package/dist/types/src/extensions/markdown/parser.test.d.ts.map +1 -0
  103. package/dist/types/src/extensions/markdown/styles.d.ts +4 -0
  104. package/dist/types/src/extensions/markdown/styles.d.ts.map +1 -0
  105. package/dist/types/src/extensions/markdown/table.d.ts +8 -0
  106. package/dist/types/src/extensions/markdown/table.d.ts.map +1 -0
  107. package/dist/types/src/extensions/mention.d.ts +7 -0
  108. package/dist/types/src/extensions/mention.d.ts.map +1 -0
  109. package/dist/types/src/extensions/modal.d.ts +7 -0
  110. package/dist/types/src/extensions/modal.d.ts.map +1 -0
  111. package/dist/types/src/extensions/modes.d.ts +10 -0
  112. package/dist/types/src/extensions/modes.d.ts.map +1 -0
  113. package/dist/types/src/extensions/outliner/commands.d.ts +10 -0
  114. package/dist/types/src/extensions/outliner/commands.d.ts.map +1 -0
  115. package/dist/types/src/extensions/outliner/editor.d.ts +5 -0
  116. package/dist/types/src/extensions/outliner/editor.d.ts.map +1 -0
  117. package/dist/types/src/extensions/outliner/editor.test.d.ts +2 -0
  118. package/dist/types/src/extensions/outliner/editor.test.d.ts.map +1 -0
  119. package/dist/types/src/extensions/outliner/index.d.ts +4 -0
  120. package/dist/types/src/extensions/outliner/index.d.ts.map +1 -0
  121. package/dist/types/src/extensions/outliner/menu.d.ts +8 -0
  122. package/dist/types/src/extensions/outliner/menu.d.ts.map +1 -0
  123. package/dist/types/src/extensions/outliner/outliner.d.ts +11 -0
  124. package/dist/types/src/extensions/outliner/outliner.d.ts.map +1 -0
  125. package/dist/types/src/extensions/outliner/outliner.test.d.ts +2 -0
  126. package/dist/types/src/extensions/outliner/outliner.test.d.ts.map +1 -0
  127. package/dist/types/src/extensions/outliner/selection.d.ts +12 -0
  128. package/dist/types/src/extensions/outliner/selection.d.ts.map +1 -0
  129. package/dist/types/src/extensions/outliner/tree.d.ts +79 -0
  130. package/dist/types/src/extensions/outliner/tree.d.ts.map +1 -0
  131. package/dist/types/src/extensions/outliner/tree.test.d.ts +2 -0
  132. package/dist/types/src/extensions/outliner/tree.test.d.ts.map +1 -0
  133. package/dist/types/src/extensions/preview/index.d.ts +2 -0
  134. package/dist/types/src/extensions/preview/index.d.ts.map +1 -0
  135. package/dist/types/src/extensions/preview/preview.d.ts +34 -0
  136. package/dist/types/src/extensions/preview/preview.d.ts.map +1 -0
  137. package/dist/types/src/extensions/replacer.d.ts +21 -0
  138. package/dist/types/src/extensions/replacer.d.ts.map +1 -0
  139. package/dist/types/src/extensions/replacer.test.d.ts +2 -0
  140. package/dist/types/src/extensions/replacer.test.d.ts.map +1 -0
  141. package/dist/types/src/extensions/scrolling/auto-scroll.d.ts +18 -0
  142. package/dist/types/src/extensions/scrolling/auto-scroll.d.ts.map +1 -0
  143. package/dist/types/src/extensions/scrolling/crawler.d.ts +75 -0
  144. package/dist/types/src/extensions/scrolling/crawler.d.ts.map +1 -0
  145. package/dist/types/src/extensions/scrolling/index.d.ts +5 -0
  146. package/dist/types/src/extensions/scrolling/index.d.ts.map +1 -0
  147. package/dist/types/src/extensions/scrolling/scroll-past-end.d.ts +3 -0
  148. package/dist/types/src/extensions/scrolling/scroll-past-end.d.ts.map +1 -0
  149. package/dist/types/src/extensions/scrolling/scroller.d.ts +16 -0
  150. package/dist/types/src/extensions/scrolling/scroller.d.ts.map +1 -0
  151. package/dist/types/src/extensions/selection.d.ts +24 -0
  152. package/dist/types/src/extensions/selection.d.ts.map +1 -0
  153. package/dist/types/src/extensions/snippets.d.ts +10 -0
  154. package/dist/types/src/extensions/snippets.d.ts.map +1 -0
  155. package/dist/types/src/extensions/state.d.ts +2 -0
  156. package/dist/types/src/extensions/state.d.ts.map +1 -0
  157. package/dist/types/src/extensions/submit.d.ts +10 -0
  158. package/dist/types/src/extensions/submit.d.ts.map +1 -0
  159. package/dist/types/src/extensions/tags/extended-markdown.d.ts +10 -0
  160. package/dist/types/src/extensions/tags/extended-markdown.d.ts.map +1 -0
  161. package/dist/types/src/extensions/tags/extended-markdown.test.d.ts +2 -0
  162. package/dist/types/src/extensions/tags/extended-markdown.test.d.ts.map +1 -0
  163. package/dist/types/src/extensions/tags/fader.d.ts +12 -0
  164. package/dist/types/src/extensions/tags/fader.d.ts.map +1 -0
  165. package/dist/types/src/extensions/tags/index.d.ts +7 -0
  166. package/dist/types/src/extensions/tags/index.d.ts.map +1 -0
  167. package/dist/types/src/extensions/tags/typewriter.d.ts +43 -0
  168. package/dist/types/src/extensions/tags/typewriter.d.ts.map +1 -0
  169. package/dist/types/src/extensions/tags/typewriter.test.d.ts +2 -0
  170. package/dist/types/src/extensions/tags/typewriter.test.d.ts.map +1 -0
  171. package/dist/types/src/extensions/tags/xml-block-decoration.d.ts +31 -0
  172. package/dist/types/src/extensions/tags/xml-block-decoration.d.ts.map +1 -0
  173. package/dist/types/src/extensions/tags/xml-formatting.d.ts +24 -0
  174. package/dist/types/src/extensions/tags/xml-formatting.d.ts.map +1 -0
  175. package/dist/types/src/extensions/tags/xml-tags.d.ts +117 -0
  176. package/dist/types/src/extensions/tags/xml-tags.d.ts.map +1 -0
  177. package/dist/types/src/extensions/tags/xml-util.d.ts +10 -0
  178. package/dist/types/src/extensions/tags/xml-util.d.ts.map +1 -0
  179. package/dist/types/src/extensions/tags/xml-util.test.d.ts +2 -0
  180. package/dist/types/src/extensions/tags/xml-util.test.d.ts.map +1 -0
  181. package/dist/types/src/index.d.ts +8 -0
  182. package/dist/types/src/index.d.ts.map +1 -0
  183. package/dist/types/src/styles/index.d.ts +2 -0
  184. package/dist/types/src/styles/index.d.ts.map +1 -0
  185. package/dist/types/src/styles/theme.d.ts +58 -0
  186. package/dist/types/src/styles/theme.d.ts.map +1 -0
  187. package/dist/types/src/types/index.d.ts +2 -0
  188. package/dist/types/src/types/index.d.ts.map +1 -0
  189. package/dist/types/src/types/types.d.ts +21 -0
  190. package/dist/types/src/types/types.d.ts.map +1 -0
  191. package/dist/types/src/util/cursor.d.ts +31 -0
  192. package/dist/types/src/util/cursor.d.ts.map +1 -0
  193. package/dist/types/src/util/debug.d.ts +17 -0
  194. package/dist/types/src/util/debug.d.ts.map +1 -0
  195. package/dist/types/src/util/decorations.d.ts +4 -0
  196. package/dist/types/src/util/decorations.d.ts.map +1 -0
  197. package/dist/types/src/util/dom.d.ts +10 -0
  198. package/dist/types/src/util/dom.d.ts.map +1 -0
  199. package/dist/types/src/util/facet.d.ts +3 -0
  200. package/dist/types/src/util/facet.d.ts.map +1 -0
  201. package/dist/types/src/util/index.d.ts +7 -0
  202. package/dist/types/src/util/index.d.ts.map +1 -0
  203. package/dist/types/src/util/util.d.ts +8 -0
  204. package/dist/types/src/util/util.d.ts.map +1 -0
  205. package/dist/types/tsconfig.tsbuildinfo +1 -0
  206. package/package.json +43 -44
  207. package/src/defaults.ts +33 -20
  208. package/src/extensions/annotations.ts +1 -1
  209. package/src/extensions/autocomplete/placeholder.ts +37 -18
  210. package/src/extensions/automerge/automerge.test.tsx +37 -11
  211. package/src/extensions/automerge/automerge.ts +5 -7
  212. package/src/extensions/blocks.ts +5 -5
  213. package/src/extensions/comments.ts +5 -6
  214. package/src/extensions/dnd.ts +2 -2
  215. package/src/extensions/factories.test.ts +88 -0
  216. package/src/extensions/factories.ts +32 -15
  217. package/src/extensions/folding.ts +5 -22
  218. package/src/extensions/index.ts +2 -3
  219. package/src/extensions/markdown/action.ts +0 -1
  220. package/src/extensions/markdown/bundle.ts +23 -9
  221. package/src/extensions/markdown/decorate.ts +15 -12
  222. package/src/extensions/markdown/formatting.ts +5 -10
  223. package/src/extensions/markdown/highlight.ts +15 -7
  224. package/src/extensions/markdown/link.ts +27 -33
  225. package/src/extensions/markdown/parser.test.ts +0 -1
  226. package/src/extensions/markdown/styles.ts +42 -9
  227. package/src/extensions/markdown/table.ts +24 -2
  228. package/src/extensions/outliner/outliner.test.ts +0 -1
  229. package/src/extensions/outliner/outliner.ts +3 -4
  230. package/src/extensions/outliner/tree.test.ts +0 -1
  231. package/src/extensions/preview/preview.ts +62 -15
  232. package/src/extensions/scrolling/auto-scroll.ts +244 -0
  233. package/src/extensions/scrolling/crawler.ts +263 -0
  234. package/src/extensions/scrolling/index.ts +8 -0
  235. package/src/extensions/scrolling/scroll-past-end.ts +32 -0
  236. package/src/extensions/scrolling/scroller.ts +27 -0
  237. package/src/extensions/selection.ts +1 -1
  238. package/src/extensions/snippets.ts +67 -0
  239. package/src/extensions/tags/extended-markdown.test.ts +120 -2
  240. package/src/extensions/tags/extended-markdown.ts +80 -1
  241. package/src/extensions/tags/fader.ts +195 -0
  242. package/src/extensions/tags/index.ts +4 -1
  243. package/src/extensions/tags/testing/text.md +36 -0
  244. package/src/extensions/tags/testing/text.txt +35 -0
  245. package/src/extensions/tags/typewriter.test.ts +65 -0
  246. package/src/extensions/tags/typewriter.ts +594 -0
  247. package/src/extensions/tags/xml-block-decoration.ts +123 -0
  248. package/src/extensions/tags/xml-formatting.ts +125 -0
  249. package/src/extensions/tags/xml-tags.ts +186 -35
  250. package/src/extensions/tags/xml-util.test.ts +199 -24
  251. package/src/extensions/tags/xml-util.ts +62 -5
  252. package/src/index.ts +0 -1
  253. package/src/styles/index.ts +0 -2
  254. package/src/styles/theme.ts +125 -33
  255. package/src/types/types.ts +10 -2
  256. package/src/typings.d.ts +8 -0
  257. package/src/util/cursor.ts +1 -2
  258. package/src/extensions/autoscroll.ts +0 -165
  259. package/src/extensions/scrolling.ts +0 -189
  260. package/src/extensions/tags/streamer.ts +0 -243
  261. package/src/extensions/typewriter.ts +0 -68
  262. package/src/styles/markdown.ts +0 -26
  263. package/src/styles/tokens.ts +0 -17
package/package.json CHANGED
@@ -1,10 +1,14 @@
1
1
  {
2
2
  "name": "@dxos/ui-editor",
3
- "version": "0.0.0",
3
+ "version": "0.8.4-main.05e74ebcff",
4
4
  "description": "Text editor components.",
5
5
  "homepage": "https://dxos.org",
6
6
  "bugs": "https://github.com/dxos/dxos/issues",
7
- "license": "MIT",
7
+ "repository": {
8
+ "type": "git",
9
+ "url": "https://github.com/dxos/dxos"
10
+ },
11
+ "license": "FSL-1.1-Apache-2.0",
8
12
  "author": "DXOS.org",
9
13
  "sideEffects": false,
10
14
  "type": "module",
@@ -23,15 +27,12 @@
23
27
  }
24
28
  },
25
29
  "types": "dist/types/src/index.d.ts",
26
- "typesVersions": {
27
- "*": {}
28
- },
29
30
  "files": [
30
31
  "dist",
31
32
  "src"
32
33
  ],
33
34
  "dependencies": {
34
- "@automerge/automerge": "3.2.1",
35
+ "@automerge/automerge": "3.2.6",
35
36
  "@codemirror/autocomplete": "^6.19.0",
36
37
  "@codemirror/commands": "^6.8.1",
37
38
  "@codemirror/lang-html": "^6.4.11",
@@ -46,7 +47,7 @@
46
47
  "@codemirror/search": "^6.5.11",
47
48
  "@codemirror/state": "^6.5.2",
48
49
  "@codemirror/theme-one-dark": "^6.1.3",
49
- "@codemirror/view": "^6.38.4",
50
+ "@codemirror/view": "^6.38.5",
50
51
  "@lezer/common": "^1.2.2",
51
52
  "@lezer/generator": "^1.7.1",
52
53
  "@lezer/highlight": "^1.2.1",
@@ -54,36 +55,35 @@
54
55
  "@replit/codemirror-vim": "^6.2.1",
55
56
  "@replit/codemirror-vscode-keymap": "^6.0.2",
56
57
  "@uiw/codemirror-theme-vscode": "^4.25.2",
57
- "ajv": "^8.17.1",
58
+ "ajv": "^8.18.0",
58
59
  "codemirror": "^6.0.1",
59
60
  "lib0": "^0.2.65",
60
61
  "lodash.defaultsdeep": "^4.6.1",
61
62
  "lodash.merge": "^4.6.2",
62
63
  "lodash.sortby": "^4.7.0",
63
64
  "style-mod": "^4.1.0",
64
- "@dxos/app-graph": "0.8.3",
65
- "@dxos/context": "0.8.3",
66
- "@dxos/async": "0.8.3",
67
- "@dxos/client": "0.8.3",
68
- "@dxos/debug": "0.8.3",
69
- "@dxos/invariant": "0.8.3",
70
- "@dxos/echo-db": "0.8.3",
71
- "@dxos/echo": "0.8.3",
72
- "@dxos/live-object": "0.8.3",
73
- "@dxos/log": "0.8.3",
74
- "@dxos/protocols": "0.8.3",
75
- "@dxos/display-name": "0.8.3",
76
- "@dxos/ui-types": "0.0.0",
77
- "@dxos/ui": "0.0.0",
78
- "@dxos/ui-theme": "0.0.0",
79
- "@dxos/lit-ui": "0.8.3",
80
- "@dxos/util": "0.8.3"
65
+ "@dxos/app-graph": "0.8.4-main.05e74ebcff",
66
+ "@dxos/client": "0.8.4-main.05e74ebcff",
67
+ "@dxos/context": "0.8.4-main.05e74ebcff",
68
+ "@dxos/echo": "0.8.4-main.05e74ebcff",
69
+ "@dxos/debug": "0.8.4-main.05e74ebcff",
70
+ "@dxos/display-name": "0.8.4-main.05e74ebcff",
71
+ "@dxos/echo-db": "0.8.4-main.05e74ebcff",
72
+ "@dxos/invariant": "0.8.4-main.05e74ebcff",
73
+ "@dxos/lit-ui": "0.8.4-main.05e74ebcff",
74
+ "@dxos/log": "0.8.4-main.05e74ebcff",
75
+ "@dxos/protocols": "0.8.4-main.05e74ebcff",
76
+ "@dxos/ui": "0.8.4-main.05e74ebcff",
77
+ "@dxos/ui-theme": "0.8.4-main.05e74ebcff",
78
+ "@dxos/ui-types": "0.8.4-main.05e74ebcff",
79
+ "@dxos/util": "0.8.4-main.05e74ebcff",
80
+ "@dxos/async": "0.8.4-main.05e74ebcff"
81
81
  },
82
82
  "devDependencies": {
83
- "@automerge/automerge": "3.2.1",
84
- "@automerge/automerge-repo": "2.5.1",
85
- "@automerge/automerge-repo-network-broadcastchannel": "2.5.1",
86
- "@effect/platform": "0.93.6",
83
+ "@automerge/automerge": "3.2.6",
84
+ "@automerge/automerge-repo": "2.6.0-subduction.17",
85
+ "@automerge/automerge-repo-network-broadcastchannel": "2.6.0-subduction.17",
86
+ "@effect/platform": "0.94.4",
87
87
  "@types/chai": "^4.2.15",
88
88
  "@types/chai-dom": "^1.11.0",
89
89
  "@types/lodash.defaultsdeep": "^4.6.6",
@@ -91,26 +91,25 @@
91
91
  "@types/lodash.sortby": "^4.7.7",
92
92
  "chai": "^4.4.1",
93
93
  "chai-dom": "^1.11.0",
94
- "effect": "3.19.11",
95
- "happy-dom": "^13.3.1",
94
+ "effect": "3.20.0",
95
+ "happy-dom": "^20.0.0",
96
96
  "jsdom": "^27.0.0",
97
97
  "mocha": "^10.6.0",
98
- "vite": "7.1.9",
98
+ "vite": "^8.0.13",
99
99
  "vite-plugin-top-level-await": "^1.6.0",
100
- "vite-plugin-wasm": "^3.5.0",
101
- "@dxos/echo": "0.8.3",
102
- "@dxos/keyboard": "0.8.3",
103
- "@dxos/echo-signals": "0.8.3",
104
- "@dxos/config": "0.8.3",
105
- "@dxos/random": "0.8.3",
106
- "@dxos/storybook-utils": "0.8.3",
107
- "@dxos/ui-theme": "0.0.0",
108
- "@dxos/schema": "0.8.3"
100
+ "vite-plugin-wasm": "^3.6.0",
101
+ "@dxos/config": "0.8.4-main.05e74ebcff",
102
+ "@dxos/echo": "0.8.4-main.05e74ebcff",
103
+ "@dxos/keyboard": "0.8.4-main.05e74ebcff",
104
+ "@dxos/schema": "0.8.4-main.05e74ebcff",
105
+ "@dxos/ui-theme": "0.8.4-main.05e74ebcff",
106
+ "@dxos/storybook-utils": "0.8.4-main.05e74ebcff",
107
+ "@dxos/random": "0.8.4-main.05e74ebcff"
109
108
  },
110
109
  "peerDependencies": {
111
- "@effect/platform": "0.93.6",
112
- "effect": "3.19.11",
113
- "@dxos/ui-theme": "0.0.0"
110
+ "@effect/platform": "0.94.4",
111
+ "effect": "3.20.0",
112
+ "@dxos/ui-theme": "0.8.4-main.05e74ebcff"
114
113
  },
115
114
  "publishConfig": {
116
115
  "access": "public"
package/src/defaults.ts CHANGED
@@ -6,29 +6,42 @@ import { mx } from '@dxos/ui-theme';
6
6
 
7
7
  import { type ThemeExtensionsOptions } from './extensions';
8
8
 
9
- /**
10
- * CodeMirror content width.
11
- * 40rem = 640px. Corresponds to initial plank width (Google docs, Stashpad, etc.)
12
- * 50rem = 800px. Maximum content width for solo mode.
13
- * NOTE: Max width - 4rem = 2rem left/right margin (or 2rem gutter plus 1rem left/right margin).
14
- */
15
- export const editorWidth = '!mli-auto is-full max-is-[min(50rem,100%-4rem)]';
9
+ // NOTE: Padding is added to the editor to account for the focus ring (since otherwise the CM gutter will clip it)
10
+ export const editorClassNames = (role?: string) =>
11
+ mx(
12
+ 'dx-attention-surface p-0.5 data-[toolbar=disabled]:pt-2 dx-focus-ring-inset',
13
+ role === 'section' ? '[&_.cm-scroller]:overflow-hidden [&_.cm-scroller]:min-h-24' : 'dx-container overflow-hidden',
14
+ );
16
15
 
17
- export const editorSlots: ThemeExtensionsOptions['slots'] = {
18
- scroll: {
19
- className: 'pbs-2',
20
- },
16
+ export const documentSlots: ThemeExtensionsOptions['slots'] = {
21
17
  content: {
22
- className: editorWidth,
18
+ /**
19
+ * CodeMirror content width.
20
+ * 40rem = 640px. Corresponds to initial plank width (Google docs, Stashpad, etc.)
21
+ * 50rem = 800px. Maximum content width for solo mode.
22
+ * NOTE: Max width - 4rem = 2rem left/right margin (or 2rem gutter plus 1rem left/right margin).
23
+ */
24
+ className: mx(
25
+ // Inline-size container for widget sizing (children use `max-w-[100cqi]`).
26
+ // NOTE: Use inline-size, not full size containment — `container-type: size` on the
27
+ // editor content breaks CodeMirror's viewport measurement, leaving blank gaps during
28
+ // scroll until a click forces a re-measure.
29
+ 'dx-inline-size-container',
30
+ // Wider margin for web (vs. mobile).
31
+ 'pointer-fine:max-w-[min(50rem,100%-4rem)] pointer-coarse:max-w-[min(50rem,100%-2rem)]',
32
+ 'mx-auto! w-full',
33
+ ),
23
34
  },
24
35
  };
25
36
 
26
- export const editorWithToolbarLayout =
27
- 'grid grid-cols-1 grid-rows-[min-content_1fr] data-[toolbar=disabled]:grid-rows-[1fr] justify-center content-start overflow-hidden';
37
+ export const compactSlots: ThemeExtensionsOptions['slots'] = {
38
+ content: {
39
+ className: 'mx-2!',
40
+ },
41
+ };
28
42
 
29
- // NOTE: Padding is added to the editor to account for the focus ring (since otherwise the CM gutter will clip it)
30
- export const stackItemContentEditorClassNames = (role?: string) =>
31
- mx(
32
- 'p-0.5 dx-focus-ring-inset attention-surface data-[toolbar=disabled]:pbs-2',
33
- role === 'section' ? '[&_.cm-scroller]:overflow-hidden [&_.cm-scroller]:min-bs-24' : 'min-bs-0',
34
- );
43
+ export const mobileSlots: ThemeExtensionsOptions['slots'] = {
44
+ content: {
45
+ className: 'mx-2!',
46
+ },
47
+ };
@@ -48,7 +48,7 @@ export const annotations = ({ match }: AnnotationOptions = {}): Extension => {
48
48
  '.cm-annotation': {
49
49
  textDecoration: 'underline',
50
50
  textDecorationStyle: 'wavy',
51
- textDecorationColor: 'var(--dx-errorText)',
51
+ textDecorationColor: 'var(--color-error-text)',
52
52
  },
53
53
  }),
54
54
  ];
@@ -13,42 +13,61 @@ type Content = string | HTMLElement | ((view: EditorView) => HTMLElement);
13
13
  export type PlaceholderOptions = {
14
14
  content: Content;
15
15
  delay?: number;
16
+ focusOnly?: boolean;
16
17
  };
17
18
 
18
19
  /**
19
- * Shows a transient placeholder at the current cursor position.
20
+ * Shows a transient placeholder at the current cursor position. When
21
+ * `focusOnly` is set the placeholder is suppressed unless the editor has DOM
22
+ * focus, and is hidden again when focus is lost.
20
23
  */
21
- export const placeholder = ({ content, delay = 3_000 }: PlaceholderOptions): Extension => {
24
+ export const placeholder = ({ content, delay = 3_000, focusOnly = false }: PlaceholderOptions): Extension => {
22
25
  const plugin = ViewPlugin.fromClass(
23
26
  class {
24
27
  _timeout: ReturnType<typeof setTimeout> | undefined;
25
28
  _decorations = Decoration.none;
26
29
 
27
30
  update(update: ViewUpdate) {
31
+ // React to actual user activity only. The empty `view.update([])`
32
+ // dispatched from the timeout below carries no doc/selection/focus
33
+ // change, so it falls through here as a no-op — that's how the
34
+ // freshly-set widget survives long enough for the decoration
35
+ // provider to read it. Without this gate, the unconditional reset
36
+ // would clobber the decoration in the same tick and the placeholder
37
+ // would never visibly render.
38
+ if (!update.docChanged && !update.selectionSet && !update.focusChanged) {
39
+ return;
40
+ }
41
+
28
42
  if (this._timeout) {
29
43
  window.clearTimeout(this._timeout);
30
44
  this._timeout = undefined;
31
45
  }
46
+ this._decorations = Decoration.none;
47
+
48
+ // Honour `focusOnly`: when the option is set and the editor isn't
49
+ // focused, leave the placeholder hidden and skip rescheduling. The
50
+ // next `focusChanged` update reschedules once focus returns.
51
+ if (focusOnly && !update.view.hasFocus) {
52
+ return;
53
+ }
32
54
 
33
- // Check if the active line (where cursor is) is empty.
34
55
  const activeLine = update.view.state.doc.lineAt(update.view.state.selection.main.head);
35
- const isEmpty = activeLine.text.trim() === '';
36
- if (isEmpty) {
37
- // Create widget decoration at the start of the current line.
38
- const lineStart = activeLine.from;
39
- this._timeout = setTimeout(() => {
40
- this._decorations = Decoration.set([
41
- Decoration.widget({
42
- widget: new PlaceholderWidget(content),
43
- side: 1,
44
- }).range(lineStart),
45
- ]);
46
-
47
- update.view.update([]);
48
- }, delay);
56
+ if (activeLine.text.trim() !== '') {
57
+ return;
49
58
  }
50
59
 
51
- this._decorations = Decoration.none;
60
+ const lineStart = activeLine.from;
61
+ const view = update.view;
62
+ this._timeout = setTimeout(() => {
63
+ this._decorations = Decoration.set([
64
+ Decoration.widget({
65
+ widget: new PlaceholderWidget(content),
66
+ side: 1,
67
+ }).range(lineStart),
68
+ ]);
69
+ view.update([]);
70
+ }, delay);
52
71
  }
53
72
 
54
73
  destroy() {
@@ -2,23 +2,43 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import { type DocHandle, Repo } from '@automerge/automerge-repo';
5
+ import { type ChangeFn, type ChangeOptions, type Doc, type Heads } from '@automerge/automerge';
6
+ import { type DocHandle, Repo, decodeHeads, encodeHeads, initSubduction } from '@automerge/automerge-repo';
6
7
  import { EditorState } from '@codemirror/state';
7
8
  import { EditorView } from '@codemirror/view';
8
9
  import { render, screen } from '@testing-library/react';
9
10
  import React, { type FC, useEffect, useRef, useState } from 'react';
10
- import { describe, test } from 'vitest';
11
+ import { beforeAll, describe, test } from 'vitest';
11
12
 
12
- import { get } from '@dxos/util';
13
+ import { type IDocHandle } from '@dxos/echo-db';
14
+ import { getDeep } from '@dxos/util';
13
15
 
14
16
  import { automerge } from './automerge';
15
17
 
18
+ // Adapter: `IDocHandle` (used by `DocHandleProxy` in production) takes raw `Heads` (hex)
19
+ // at `changeAt`, but automerge-repo's `DocHandle.changeAt` expects bs58check-encoded
20
+ // `UrlHeads`. Encode on the way in, decode on the way out so the test's repo-backed
21
+ // handle satisfies the extension's `IDocHandle` contract.
22
+ const adaptRepoHandle = <T,>(handle: DocHandle<T>): IDocHandle<T> => ({
23
+ doc: () => handle.doc() as Doc<T> | undefined,
24
+ change: (callback: ChangeFn<T>, options?: ChangeOptions<T>) =>
25
+ options ? handle.change(callback, options) : handle.change(callback),
26
+ changeAt: (heads: Heads, callback: ChangeFn<T>, options?: ChangeOptions<T>): Heads | undefined => {
27
+ const encoded = options
28
+ ? handle.changeAt(encodeHeads(heads), callback, options)
29
+ : handle.changeAt(encodeHeads(heads), callback);
30
+ return encoded ? decodeHeads(encoded) : undefined;
31
+ },
32
+ addListener: (event, listener) => handle.on(event, listener),
33
+ removeListener: (event, listener) => handle.off(event, listener),
34
+ });
35
+
36
+ const path = ['text'];
37
+
16
38
  type TestObject = {
17
39
  text: string;
18
40
  };
19
41
 
20
- const path = ['text'];
21
-
22
42
  class Generator {
23
43
  constructor(private readonly _handle: DocHandle<TestObject>) {}
24
44
  update(text: string): void {
@@ -33,8 +53,7 @@ const Test: FC<{ handle: DocHandle<TestObject>; generator: Generator }> = ({ han
33
53
  const [view, setView] = useState<EditorView>();
34
54
  useEffect(() => {
35
55
  const extensions = [
36
- // TODO(mykola): Fix types.
37
- automerge({ handle: handle as any, path }),
56
+ automerge({ handle: adaptRepoHandle(handle), path }),
38
57
  EditorView.updateListener.of((update) => {
39
58
  if (view.state.doc.toString() === 'hello!') {
40
59
  // Update editor.
@@ -46,7 +65,7 @@ const Test: FC<{ handle: DocHandle<TestObject>; generator: Generator }> = ({ han
46
65
  ];
47
66
 
48
67
  const view = new EditorView({
49
- state: EditorState.create({ doc: get(handle.doc()!, path), extensions }),
68
+ state: EditorState.create({ doc: getDeep(handle.doc()!, path), extensions }),
50
69
  parent: ref.current!,
51
70
  });
52
71
 
@@ -58,19 +77,26 @@ const Test: FC<{ handle: DocHandle<TestObject>; generator: Generator }> = ({ han
58
77
  return <div ref={ref} data-testid='editor' />;
59
78
  };
60
79
 
80
+ // TODO(burdon): Test history/undo.
81
+ // TODO(burdon): https://testing-library.com/docs/react-testing-library/example-intro/
82
+
61
83
  describe('Automerge', () => {
84
+ // Subduction-fork `Repo` constructs a `MemorySigner` internally; WASM must be
85
+ // initialized first or the constructor throws `'set_subduction_logger' of undefined`.
86
+ beforeAll(async () => {
87
+ await initSubduction();
88
+ });
89
+
62
90
  test('basic sync', ({ expect }) => {
63
91
  const repo = new Repo({ network: [] });
64
92
  const handle = repo.create<TestObject>();
65
93
  const generator = new Generator(handle);
66
94
  render(<Test handle={handle} generator={generator} />);
95
+
67
96
  const editor = screen.getByTestId('editor');
68
97
  expect(editor.textContent).toBe('');
69
98
 
70
99
  generator.update('hello!');
71
100
  expect(editor.textContent).toBe('hello world!');
72
101
  });
73
-
74
- // TODO(burdon): Test history/undo.
75
- // TODO(burdon): https://testing-library.com/docs/react-testing-library/example-intro/
76
102
  });
@@ -12,9 +12,8 @@ import { DocAccessor } from '@dxos/echo-db';
12
12
 
13
13
  import { Cursor } from '../../util';
14
14
  import { initialSync } from '../state';
15
-
16
15
  import { cursorConverter } from './cursor';
17
- import { type State, isReconcile, updateHeadsEffect } from './defs';
16
+ import { type State, isReconcile, reconcileAnnotation, updateHeadsEffect } from './defs';
18
17
  import { Syncer } from './sync';
19
18
 
20
19
  export const automerge = (accessor: DocAccessor): Extension => {
@@ -72,11 +71,10 @@ export const automerge = (accessor: DocAccessor): Extension => {
72
71
  const value = DocAccessor.getValue<string>(accessor);
73
72
  const current = this._view.state.doc.toString();
74
73
  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
- // });
74
+ this._view.dispatch({
75
+ changes: { from: 0, to: this._view.state.doc.length, insert: value },
76
+ annotations: [initialSync, reconcileAnnotation.of(true)],
77
+ });
80
78
  }
81
79
  });
82
80
  }
@@ -101,11 +101,11 @@ export const blocks = () => [
101
101
  '.cm-line.block-line': {
102
102
  paddingLeft: '0.75rem',
103
103
  paddingRight: '0.75rem',
104
- borderLeft: '1px solid var(--dx-subduedSeparator)',
105
- borderRight: '1px solid var(--dx-subduedSeparator)',
104
+ borderLeft: '1px solid var(--color-subdued-separator)',
105
+ borderRight: '1px solid var(--color-subdued-separator)',
106
106
  },
107
107
  '.cm-line.block-single': {
108
- border: '1px solid var(--dx-subduedSeparator)',
108
+ border: '1px solid var(--color-subdued-separator)',
109
109
  borderRadius: '6px',
110
110
  paddingTop: '0.5rem',
111
111
  paddingBottom: '0.5rem',
@@ -113,7 +113,7 @@ export const blocks = () => [
113
113
  marginBottom: '0.5rem',
114
114
  },
115
115
  '.cm-line.block-first': {
116
- borderTop: '1px solid var(--dx-subduedSeparator)',
116
+ borderTop: '1px solid var(--color-subdued-separator)',
117
117
  borderTopLeftRadius: '6px',
118
118
  borderTopRightRadius: '6px',
119
119
  paddingTop: '0.5rem',
@@ -121,7 +121,7 @@ export const blocks = () => [
121
121
  },
122
122
  '.cm-line.block-middle': {},
123
123
  '.cm-line.block-last': {
124
- borderBottom: '1px solid var(--dx-subduedSeparator)',
124
+ borderBottom: '1px solid var(--color-subdued-separator)',
125
125
  borderBottomLeftRadius: '6px',
126
126
  borderBottomRightRadius: '6px',
127
127
  paddingBottom: '0.5rem',
@@ -22,7 +22,6 @@ import { isNonNullable } from '@dxos/util';
22
22
 
23
23
  import { type Comment, type Range, type RenderCallback } from '../types';
24
24
  import { Cursor, singleValueFacet, wrapWithCatch } from '../util';
25
-
26
25
  import { documentId } from './selection';
27
26
 
28
27
  //
@@ -103,14 +102,14 @@ export const commentsState = StateField.define<CommentsState>({
103
102
  const styles = EditorView.theme({
104
103
  '.cm-comment, .cm-comment-current': {
105
104
  padding: '3px 0',
106
- color: 'var(--dx-cmCommentText)',
107
- backgroundColor: 'var(--dx-cmCommentSurface)',
105
+ color: 'var(--color-cm-comment-text)',
106
+ backgroundColor: 'var(--color-cm-comment-surface)',
108
107
  },
109
108
  '.cm-comment > span, .cm-comment-current > span': {
110
109
  boxDecorationBreak: 'clone',
111
- boxShadow: '0 0 1px 3px var(--dx-cmCommentSurface)',
112
- backgroundColor: 'var(--dx-cmCommentSurface)',
113
- color: 'var(--dx-cmCommentText)',
110
+ boxShadow: '0 0 1px 3px var(--color-cm-comment-surface)',
111
+ backgroundColor: 'var(--color-cm-comment-surface)',
112
+ color: 'var(--color-cm-comment-text)',
114
113
  cursor: 'pointer',
115
114
  },
116
115
  });
@@ -29,8 +29,8 @@ export const dropFile = (options: DropOptions = {}): Extension => {
29
29
 
30
30
  const styles = EditorView.theme({
31
31
  '.cm-dropCursor': {
32
- borderLeft: '2px solid var(--dx-accentText)',
33
- color: 'var(--dx-accentText)',
32
+ borderLeft: '2px solid var(--color-accent-text)',
33
+ color: 'var(--color-accent-text)',
34
34
  padding: '0 4px',
35
35
  },
36
36
  '.cm-dropCursor:after': {
@@ -0,0 +1,88 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { markdown, markdownLanguage, insertNewlineContinueMarkup } from '@codemirror/lang-markdown';
6
+ import { EditorSelection, EditorState } from '@codemirror/state';
7
+ import { describe, test } from 'vitest';
8
+
9
+ import { createBasicExtensions } from './factories';
10
+
11
+ describe('createBasicExtensions readOnly', () => {
12
+ test('drops user-initiated edits when readOnly is true', ({ expect }) => {
13
+ const state = EditorState.create({
14
+ doc: 'hello',
15
+ extensions: [createBasicExtensions({ readOnly: true })],
16
+ });
17
+ const tr = state.update({
18
+ changes: { from: state.doc.length, insert: ' world' },
19
+ userEvent: 'input.type',
20
+ });
21
+ expect(tr.state.doc.toString()).toBe('hello');
22
+ });
23
+
24
+ for (const userEvent of ['delete.forward', 'undo', 'redo']) {
25
+ test(`drops '${userEvent}' user events when readOnly is true`, ({ expect }) => {
26
+ const state = EditorState.create({
27
+ doc: 'hello',
28
+ extensions: [createBasicExtensions({ readOnly: true })],
29
+ });
30
+ const tr = state.update({
31
+ changes: { from: state.doc.length, insert: ' world' },
32
+ userEvent,
33
+ });
34
+ expect(tr.state.doc.toString()).toBe('hello');
35
+ });
36
+ }
37
+
38
+ test('allows programmatic dispatches (no userEvent) when readOnly is true', ({ expect }) => {
39
+ // Streaming consumers (e.g. MarkdownStream) populate the doc programmatically — those
40
+ // transactions must pass even though the editor is read-only to the user.
41
+ const state = EditorState.create({
42
+ doc: 'hello',
43
+ extensions: [createBasicExtensions({ readOnly: true })],
44
+ });
45
+ const tr = state.update({ changes: { from: state.doc.length, insert: ' world' } });
46
+ expect(tr.state.doc.toString()).toBe('hello world');
47
+ });
48
+
49
+ test('selection-only transactions still apply when readOnly', ({ expect }) => {
50
+ const state = EditorState.create({
51
+ doc: 'hello',
52
+ extensions: [createBasicExtensions({ readOnly: true })],
53
+ });
54
+ const tr = state.update({ selection: EditorSelection.cursor(2) });
55
+ expect(tr.state.selection.main.head).toBe(2);
56
+ });
57
+
58
+ test('doc-changing transactions apply normally when readOnly is false', ({ expect }) => {
59
+ const state = EditorState.create({
60
+ doc: 'hello',
61
+ extensions: [createBasicExtensions({ readOnly: false })],
62
+ });
63
+ const tr = state.update({ changes: { from: state.doc.length, insert: ' world' } });
64
+ expect(tr.state.doc.toString()).toBe('hello world');
65
+ });
66
+
67
+ // Regression: in MarkdownStream `readOnly: true` must also block the markdown extension's
68
+ // Enter handler (`insertNewlineContinueMarkup`), which programmatically dispatches a list
69
+ // continuation regardless of the readOnly facet.
70
+ test('markdown insertNewlineContinueMarkup is suppressed when readOnly', ({ expect }) => {
71
+ const doc = '- one\n- two';
72
+ const state = EditorState.create({
73
+ doc,
74
+ selection: EditorSelection.cursor(doc.length),
75
+ extensions: [createBasicExtensions({ readOnly: true }), markdown({ base: markdownLanguage })],
76
+ });
77
+ let dispatched: any;
78
+ insertNewlineContinueMarkup({
79
+ state,
80
+ dispatch: (tr) => {
81
+ dispatched = tr;
82
+ },
83
+ });
84
+ if (dispatched) {
85
+ expect(dispatched.state.doc.toString()).toBe(doc);
86
+ }
87
+ });
88
+ });