@dxos/plugin-sheet 0.6.11 → 0.6.12-main.568932b

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 (277) hide show
  1. package/dist/lib/browser/SheetContainer-T2QWJOFD.mjs +262 -0
  2. package/dist/lib/browser/SheetContainer-T2QWJOFD.mjs.map +7 -0
  3. package/dist/lib/browser/{chunk-FUAGSXA4.mjs → chunk-5ZMVZYGB.mjs} +24 -19
  4. package/dist/lib/browser/chunk-5ZMVZYGB.mjs.map +7 -0
  5. package/dist/lib/browser/chunk-GSV5QNLD.mjs +2966 -0
  6. package/dist/lib/browser/chunk-GSV5QNLD.mjs.map +7 -0
  7. package/dist/lib/browser/{chunk-JRL5LGCE.mjs → chunk-QILRZNE5.mjs} +2 -5
  8. package/dist/lib/browser/chunk-QILRZNE5.mjs.map +7 -0
  9. package/dist/lib/browser/{SheetContainer-U4H5D34A.mjs → chunk-ZL2V5UJR.mjs} +1182 -249
  10. package/dist/lib/browser/chunk-ZL2V5UJR.mjs.map +7 -0
  11. package/dist/lib/browser/graph-M4IQ76QX.mjs +33 -0
  12. package/dist/lib/browser/graph-M4IQ76QX.mjs.map +7 -0
  13. package/dist/lib/browser/index.mjs +96 -60
  14. package/dist/lib/browser/index.mjs.map +4 -4
  15. package/dist/lib/browser/meta.json +1 -1
  16. package/dist/lib/browser/meta.mjs +1 -1
  17. package/dist/lib/browser/types.mjs +4 -6
  18. package/dist/lib/node/SheetContainer-PV5ET4UJ.cjs +280 -0
  19. package/dist/lib/node/SheetContainer-PV5ET4UJ.cjs.map +7 -0
  20. package/dist/lib/node/{SheetContainer-AXQV3ZT5.cjs → chunk-2K53Z2TU.cjs} +1212 -287
  21. package/dist/lib/node/chunk-2K53Z2TU.cjs.map +7 -0
  22. package/dist/lib/node/{chunk-5KKJ4NPP.cjs → chunk-5XPK2V4A.cjs} +418 -678
  23. package/dist/lib/node/chunk-5XPK2V4A.cjs.map +7 -0
  24. package/dist/lib/node/{chunk-BJ6ZD7MN.cjs → chunk-BNARJ5GM.cjs} +5 -18
  25. package/dist/lib/node/chunk-BNARJ5GM.cjs.map +7 -0
  26. package/dist/lib/node/{chunk-DSYKOI4E.cjs → chunk-STAVQ2JE.cjs} +28 -24
  27. package/dist/lib/node/chunk-STAVQ2JE.cjs.map +7 -0
  28. package/dist/lib/node/graph-Q3N2X26H.cjs +55 -0
  29. package/dist/lib/node/graph-Q3N2X26H.cjs.map +7 -0
  30. package/dist/lib/node/index.cjs +106 -66
  31. package/dist/lib/node/index.cjs.map +4 -4
  32. package/dist/lib/node/meta.cjs +3 -3
  33. package/dist/lib/node/meta.cjs.map +1 -1
  34. package/dist/lib/node/meta.json +1 -1
  35. package/dist/lib/node/types.cjs +10 -12
  36. package/dist/lib/node/types.cjs.map +2 -2
  37. package/dist/lib/node-esm/SheetContainer-FOZD2WLT.mjs +263 -0
  38. package/dist/lib/node-esm/SheetContainer-FOZD2WLT.mjs.map +7 -0
  39. package/dist/lib/node-esm/chunk-2HAM45RC.mjs +88 -0
  40. package/dist/lib/node-esm/chunk-2HAM45RC.mjs.map +7 -0
  41. package/dist/lib/{browser/chunk-D5AGLXJP.mjs → node-esm/chunk-5WPZCXNS.mjs} +411 -678
  42. package/dist/lib/node-esm/chunk-5WPZCXNS.mjs.map +7 -0
  43. package/dist/lib/node-esm/chunk-IU2L277A.mjs +17 -0
  44. package/dist/lib/node-esm/chunk-IU2L277A.mjs.map +7 -0
  45. package/dist/lib/node-esm/chunk-QEUCIHIN.mjs +2706 -0
  46. package/dist/lib/node-esm/chunk-QEUCIHIN.mjs.map +7 -0
  47. package/dist/lib/node-esm/graph-SMPUMOV2.mjs +34 -0
  48. package/dist/lib/node-esm/graph-SMPUMOV2.mjs.map +7 -0
  49. package/dist/lib/node-esm/index.mjs +285 -0
  50. package/dist/lib/node-esm/index.mjs.map +7 -0
  51. package/dist/lib/node-esm/meta.json +1 -0
  52. package/dist/lib/node-esm/meta.mjs +10 -0
  53. package/dist/lib/node-esm/meta.mjs.map +7 -0
  54. package/dist/lib/node-esm/types.mjs +21 -0
  55. package/dist/lib/node-esm/types.mjs.map +7 -0
  56. package/dist/types/src/SheetPlugin.d.ts.map +1 -1
  57. package/dist/types/src/components/CellEditor/CellEditor.d.ts +23 -3
  58. package/dist/types/src/components/CellEditor/CellEditor.d.ts.map +1 -1
  59. package/dist/types/src/components/CellEditor/CellEditor.stories.d.ts +2 -2
  60. package/dist/types/src/components/CellEditor/CellEditor.stories.d.ts.map +1 -1
  61. package/dist/types/src/components/CellEditor/extension.d.ts +1 -1
  62. package/dist/types/src/components/CellEditor/extension.d.ts.map +1 -1
  63. package/dist/types/src/components/ComputeGraph/ComputeGraphContextProvider.d.ts +11 -0
  64. package/dist/types/src/components/ComputeGraph/ComputeGraphContextProvider.d.ts.map +1 -0
  65. package/dist/types/src/components/ComputeGraph/index.d.ts +1 -3
  66. package/dist/types/src/components/ComputeGraph/index.d.ts.map +1 -1
  67. package/dist/types/src/components/GridSheet/GridSheet.d.ts +10 -0
  68. package/dist/types/src/components/GridSheet/GridSheet.d.ts.map +1 -0
  69. package/dist/types/src/components/GridSheet/GridSheet.stories.d.ts +9 -0
  70. package/dist/types/src/components/GridSheet/GridSheet.stories.d.ts.map +1 -0
  71. package/dist/types/src/components/GridSheet/util.d.ts +16 -0
  72. package/dist/types/src/components/GridSheet/util.d.ts.map +1 -0
  73. package/dist/types/src/components/Sheet/Sheet.d.ts +1 -1
  74. package/dist/types/src/components/Sheet/Sheet.d.ts.map +1 -1
  75. package/dist/types/src/components/Sheet/Sheet.stories.d.ts +5 -6
  76. package/dist/types/src/components/Sheet/Sheet.stories.d.ts.map +1 -1
  77. package/dist/types/src/components/Sheet/decorations.d.ts +24 -0
  78. package/dist/types/src/components/Sheet/decorations.d.ts.map +1 -0
  79. package/dist/types/src/components/Sheet/grid.d.ts +2 -2
  80. package/dist/types/src/components/Sheet/grid.d.ts.map +1 -1
  81. package/dist/types/src/components/Sheet/nav.d.ts +3 -3
  82. package/dist/types/src/components/Sheet/nav.d.ts.map +1 -1
  83. package/dist/types/src/components/Sheet/sheet-context.d.ts +8 -7
  84. package/dist/types/src/components/Sheet/sheet-context.d.ts.map +1 -1
  85. package/dist/types/src/components/Sheet/threads.d.ts +2 -0
  86. package/dist/types/src/components/Sheet/threads.d.ts.map +1 -0
  87. package/dist/types/src/components/SheetContainer.d.ts +2 -3
  88. package/dist/types/src/components/SheetContainer.d.ts.map +1 -1
  89. package/dist/types/src/components/Toolbar/Toolbar.d.ts +19 -3
  90. package/dist/types/src/components/Toolbar/Toolbar.d.ts.map +1 -1
  91. package/dist/types/src/components/Toolbar/Toolbar.stories.d.ts +18 -13
  92. package/dist/types/src/components/Toolbar/Toolbar.stories.d.ts.map +1 -1
  93. package/dist/types/src/components/index.d.ts +2 -2
  94. package/dist/types/src/components/index.d.ts.map +1 -1
  95. package/dist/types/src/defs/index.d.ts +3 -0
  96. package/dist/types/src/defs/index.d.ts.map +1 -0
  97. package/dist/types/src/{model → defs}/types.d.ts +8 -3
  98. package/dist/types/src/defs/types.d.ts.map +1 -0
  99. package/dist/types/src/defs/types.test.d.ts.map +1 -0
  100. package/dist/types/src/defs/util.d.ts +43 -0
  101. package/dist/types/src/defs/util.d.ts.map +1 -0
  102. package/dist/types/src/extensions/compute.d.ts +6 -0
  103. package/dist/types/src/extensions/compute.d.ts.map +1 -0
  104. package/dist/types/src/extensions/compute.stories.d.ts +26 -0
  105. package/dist/types/src/extensions/compute.stories.d.ts.map +1 -0
  106. package/dist/types/src/extensions/index.d.ts +2 -0
  107. package/dist/types/src/extensions/index.d.ts.map +1 -0
  108. package/dist/types/src/graph/compute-graph-registry.d.ts +34 -0
  109. package/dist/types/src/graph/compute-graph-registry.d.ts.map +1 -0
  110. package/dist/types/src/graph/compute-graph.d.ts +64 -0
  111. package/dist/types/src/graph/compute-graph.d.ts.map +1 -0
  112. package/dist/types/src/graph/compute-graph.stories.d.ts +10 -0
  113. package/dist/types/src/graph/compute-graph.stories.d.ts.map +1 -0
  114. package/dist/types/src/graph/compute-graph.test.d.ts +2 -0
  115. package/dist/types/src/graph/compute-graph.test.d.ts.map +1 -0
  116. package/dist/types/src/graph/compute-node.d.ts +26 -0
  117. package/dist/types/src/graph/compute-node.d.ts.map +1 -0
  118. package/dist/types/src/{components/ComputeGraph → graph/functions}/async-function.d.ts +14 -5
  119. package/dist/types/src/graph/functions/async-function.d.ts.map +1 -0
  120. package/dist/types/src/graph/functions/edge-function.d.ts +21 -0
  121. package/dist/types/src/graph/functions/edge-function.d.ts.map +1 -0
  122. package/dist/types/src/{model/functions.d.ts → graph/functions/function-defs.d.ts} +1 -1
  123. package/dist/types/src/graph/functions/function-defs.d.ts.map +1 -0
  124. package/dist/types/src/graph/functions/index.d.ts +4 -0
  125. package/dist/types/src/graph/functions/index.d.ts.map +1 -0
  126. package/dist/types/src/graph/hyperformula.test.d.ts +2 -0
  127. package/dist/types/src/graph/hyperformula.test.d.ts.map +1 -0
  128. package/dist/types/src/graph/index.d.ts +5 -0
  129. package/dist/types/src/graph/index.d.ts.map +1 -0
  130. package/dist/types/src/graph/testing/index.d.ts +3 -0
  131. package/dist/types/src/graph/testing/index.d.ts.map +1 -0
  132. package/dist/types/src/graph/testing/test-builder.d.ts +15 -0
  133. package/dist/types/src/graph/testing/test-builder.d.ts.map +1 -0
  134. package/dist/types/src/graph/testing/test-plugin.d.ts +36 -0
  135. package/dist/types/src/graph/testing/test-plugin.d.ts.map +1 -0
  136. package/dist/types/src/graph/util.d.ts +2 -0
  137. package/dist/types/src/graph/util.d.ts.map +1 -0
  138. package/dist/types/src/hooks/hooks.stories.d.ts +11 -0
  139. package/dist/types/src/hooks/hooks.stories.d.ts.map +1 -0
  140. package/dist/types/src/hooks/index.d.ts +4 -0
  141. package/dist/types/src/hooks/index.d.ts.map +1 -0
  142. package/dist/types/src/hooks/useComputeGraph.d.ts +7 -0
  143. package/dist/types/src/hooks/useComputeGraph.d.ts.map +1 -0
  144. package/dist/types/src/hooks/useFormattingModel.d.ts +3 -0
  145. package/dist/types/src/hooks/useFormattingModel.d.ts.map +1 -0
  146. package/dist/types/src/hooks/useSheetModel.d.ts +8 -0
  147. package/dist/types/src/hooks/useSheetModel.d.ts.map +1 -0
  148. package/dist/types/src/meta.d.ts +1 -4
  149. package/dist/types/src/meta.d.ts.map +1 -1
  150. package/dist/types/src/model/formatting-model.d.ts +16 -0
  151. package/dist/types/src/model/formatting-model.d.ts.map +1 -0
  152. package/dist/types/src/model/index.d.ts +2 -3
  153. package/dist/types/src/model/index.d.ts.map +1 -1
  154. package/dist/types/src/model/{model.d.ts → sheet-model.d.ts} +10 -65
  155. package/dist/types/src/model/sheet-model.d.ts.map +1 -0
  156. package/dist/types/src/model/sheet-model.test.d.ts +2 -0
  157. package/dist/types/src/model/sheet-model.test.d.ts.map +1 -0
  158. package/dist/types/src/sanity.test.d.ts +2 -0
  159. package/dist/types/src/sanity.test.d.ts.map +1 -0
  160. package/dist/types/src/testing/index.d.ts +2 -0
  161. package/dist/types/src/testing/index.d.ts.map +1 -0
  162. package/dist/types/src/testing/testing.d.ts +8 -0
  163. package/dist/types/src/testing/testing.d.ts.map +1 -0
  164. package/dist/types/src/translations.d.ts +17 -12
  165. package/dist/types/src/translations.d.ts.map +1 -1
  166. package/dist/types/src/types.d.ts +86 -5
  167. package/dist/types/src/types.d.ts.map +1 -1
  168. package/dist/vendor/hyperformula.mjs +37145 -0
  169. package/package.json +55 -47
  170. package/src/SheetPlugin.tsx +50 -73
  171. package/src/components/CellEditor/CellEditor.stories.tsx +6 -6
  172. package/src/components/CellEditor/CellEditor.tsx +59 -9
  173. package/src/components/CellEditor/extension.test.ts +4 -6
  174. package/src/components/CellEditor/extension.ts +5 -6
  175. package/src/components/ComputeGraph/ComputeGraphContextProvider.tsx +20 -0
  176. package/src/components/ComputeGraph/index.ts +1 -3
  177. package/src/components/GridSheet/GridSheet.stories.tsx +36 -0
  178. package/src/components/GridSheet/GridSheet.tsx +171 -0
  179. package/src/components/GridSheet/util.ts +148 -0
  180. package/src/components/Sheet/Sheet.stories.tsx +52 -88
  181. package/src/components/Sheet/Sheet.tsx +87 -32
  182. package/src/components/Sheet/decorations.ts +62 -0
  183. package/src/components/Sheet/grid.ts +3 -3
  184. package/src/components/Sheet/nav.ts +19 -19
  185. package/src/components/Sheet/sheet-context.tsx +18 -80
  186. package/src/components/Sheet/threads.tsx +205 -0
  187. package/src/components/SheetContainer.tsx +68 -16
  188. package/src/components/Toolbar/Toolbar.tsx +53 -12
  189. package/src/components/index.ts +1 -0
  190. package/src/defs/index.ts +6 -0
  191. package/src/{model → defs}/types.test.ts +8 -9
  192. package/src/{model → defs}/types.ts +24 -14
  193. package/src/defs/util.ts +151 -0
  194. package/src/extensions/compute.stories.tsx +151 -0
  195. package/src/extensions/compute.ts +147 -0
  196. package/src/extensions/index.ts +5 -0
  197. package/src/graph/compute-graph-registry.ts +90 -0
  198. package/src/graph/compute-graph.stories.tsx +93 -0
  199. package/src/graph/compute-graph.test.ts +87 -0
  200. package/src/graph/compute-graph.ts +242 -0
  201. package/src/graph/compute-node.ts +63 -0
  202. package/src/{components/ComputeGraph → graph/functions}/async-function.ts +25 -15
  203. package/src/{components/ComputeGraph → graph/functions}/edge-function.ts +16 -14
  204. package/src/graph/functions/index.ts +7 -0
  205. package/src/graph/hyperformula.test.ts +14 -0
  206. package/src/graph/index.ts +8 -0
  207. package/src/graph/testing/index.ts +6 -0
  208. package/src/graph/testing/test-builder.ts +54 -0
  209. package/src/{components/ComputeGraph/custom.ts → graph/testing/test-plugin.ts} +44 -14
  210. package/src/graph/util.ts +8 -0
  211. package/src/hooks/hooks.stories.tsx +50 -0
  212. package/src/hooks/index.ts +7 -0
  213. package/src/hooks/useComputeGraph.ts +28 -0
  214. package/src/hooks/useFormattingModel.ts +11 -0
  215. package/src/hooks/useSheetModel.ts +40 -0
  216. package/src/meta.tsx +1 -5
  217. package/src/{components/Sheet/formatting.ts → model/formatting-model.ts} +20 -13
  218. package/src/model/index.ts +2 -3
  219. package/src/model/sheet-model.test.ts +57 -0
  220. package/src/model/sheet-model.ts +416 -0
  221. package/src/sanity.test.ts +40 -0
  222. package/src/testing/index.ts +5 -0
  223. package/src/testing/testing.tsx +68 -0
  224. package/src/translations.ts +6 -1
  225. package/src/types.ts +35 -10
  226. package/dist/lib/browser/SheetContainer-U4H5D34A.mjs.map +0 -7
  227. package/dist/lib/browser/chunk-APHOLYUB.mjs +0 -175
  228. package/dist/lib/browser/chunk-APHOLYUB.mjs.map +0 -7
  229. package/dist/lib/browser/chunk-D5AGLXJP.mjs.map +0 -7
  230. package/dist/lib/browser/chunk-FUAGSXA4.mjs.map +0 -7
  231. package/dist/lib/browser/chunk-JRL5LGCE.mjs.map +0 -7
  232. package/dist/lib/browser/chunk-NU4PBN33.mjs +0 -8
  233. package/dist/lib/browser/chunk-NU4PBN33.mjs.map +0 -7
  234. package/dist/lib/browser/testing.mjs +0 -92
  235. package/dist/lib/browser/testing.mjs.map +0 -7
  236. package/dist/lib/node/SheetContainer-AXQV3ZT5.cjs.map +0 -7
  237. package/dist/lib/node/chunk-5KKJ4NPP.cjs.map +0 -7
  238. package/dist/lib/node/chunk-BJ6ZD7MN.cjs.map +0 -7
  239. package/dist/lib/node/chunk-CN3RPESU.cjs +0 -202
  240. package/dist/lib/node/chunk-CN3RPESU.cjs.map +0 -7
  241. package/dist/lib/node/chunk-DSYKOI4E.cjs.map +0 -7
  242. package/dist/lib/node/chunk-PYXHNAAK.cjs +0 -40
  243. package/dist/lib/node/chunk-PYXHNAAK.cjs.map +0 -7
  244. package/dist/lib/node/testing.cjs +0 -111
  245. package/dist/lib/node/testing.cjs.map +0 -7
  246. package/dist/types/src/components/ComputeGraph/async-function.d.ts.map +0 -1
  247. package/dist/types/src/components/ComputeGraph/custom.d.ts +0 -21
  248. package/dist/types/src/components/ComputeGraph/custom.d.ts.map +0 -1
  249. package/dist/types/src/components/ComputeGraph/edge-function.d.ts +0 -20
  250. package/dist/types/src/components/ComputeGraph/edge-function.d.ts.map +0 -1
  251. package/dist/types/src/components/ComputeGraph/graph-context.d.ts +0 -12
  252. package/dist/types/src/components/ComputeGraph/graph-context.d.ts.map +0 -1
  253. package/dist/types/src/components/ComputeGraph/graph.browser.test.d.ts +0 -2
  254. package/dist/types/src/components/ComputeGraph/graph.browser.test.d.ts.map +0 -1
  255. package/dist/types/src/components/ComputeGraph/graph.d.ts +0 -26
  256. package/dist/types/src/components/ComputeGraph/graph.d.ts.map +0 -1
  257. package/dist/types/src/components/Sheet/formatting.d.ts +0 -14
  258. package/dist/types/src/components/Sheet/formatting.d.ts.map +0 -1
  259. package/dist/types/src/model/functions.d.ts.map +0 -1
  260. package/dist/types/src/model/model.browser.test.d.ts +0 -2
  261. package/dist/types/src/model/model.browser.test.d.ts.map +0 -1
  262. package/dist/types/src/model/model.d.ts.map +0 -1
  263. package/dist/types/src/model/types.d.ts.map +0 -1
  264. package/dist/types/src/model/types.test.d.ts.map +0 -1
  265. package/dist/types/src/model/util.d.ts +0 -15
  266. package/dist/types/src/model/util.d.ts.map +0 -1
  267. package/dist/types/src/testing.d.ts +0 -9
  268. package/dist/types/src/testing.d.ts.map +0 -1
  269. package/src/components/ComputeGraph/graph-context.tsx +0 -50
  270. package/src/components/ComputeGraph/graph.browser.test.ts +0 -50
  271. package/src/components/ComputeGraph/graph.ts +0 -62
  272. package/src/model/model.browser.test.ts +0 -100
  273. package/src/model/model.ts +0 -550
  274. package/src/model/util.ts +0 -36
  275. package/src/testing.ts +0 -50
  276. /package/dist/types/src/{model → defs}/types.test.d.ts +0 -0
  277. /package/src/{model/functions.ts → graph/functions/function-defs.ts} +0 -0
@@ -0,0 +1,205 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { effect } from '@preact/signals-core';
6
+ import React, { type PropsWithChildren, useCallback, useEffect, useMemo } from 'react';
7
+
8
+ import { type IntentResolver, LayoutAction, useIntentDispatcher, useIntentResolver } from '@dxos/app-framework';
9
+ import { debounce } from '@dxos/async';
10
+ import { fullyQualifiedId } from '@dxos/react-client/echo';
11
+ import { Icon, useTranslation } from '@dxos/react-ui';
12
+
13
+ import { type Decoration } from './decorations';
14
+ import { useSheetContext } from './sheet-context';
15
+ import { addressFromIndex, addressToIndex, type CellAddress, closest } from '../../defs';
16
+ import { SHEET_PLUGIN } from '../../meta';
17
+
18
+ // TODO(burdon): Move into folder; split hooks.
19
+
20
+ const CommentIndicator = () => {
21
+ return (
22
+ <div
23
+ role='none'
24
+ className='absolute top-0 right-0 w-0 h-0 border-t-8 border-l-8 border-t-cmCommentSurface border-l-transparent'
25
+ />
26
+ );
27
+ };
28
+
29
+ const ThreadedCellWrapper = ({ children }: PropsWithChildren) => {
30
+ const dispatch = useIntentDispatcher();
31
+ const [isHovered, setIsHovered] = React.useState(false);
32
+ const { t } = useTranslation(SHEET_PLUGIN);
33
+
34
+ const handleClick = React.useCallback(
35
+ (_event: React.MouseEvent) => {
36
+ void dispatch({ action: LayoutAction.SET_LAYOUT, data: { element: 'complementary', state: true } });
37
+ },
38
+ [dispatch],
39
+ );
40
+
41
+ return (
42
+ <div
43
+ role='none'
44
+ className='relative h-full is-full'
45
+ onMouseEnter={() => {
46
+ setIsHovered(true);
47
+ }}
48
+ onMouseLeave={() => {
49
+ setIsHovered(false);
50
+ }}
51
+ >
52
+ <CommentIndicator />
53
+ {isHovered && (
54
+ <div className='absolute inset-0 flex items-center justify-end pr-1'>
55
+ <button
56
+ className='ch-button text-xs min-bs-0 p-1'
57
+ onClick={handleClick}
58
+ aria-label={t('open comment for sheet cell')}
59
+ >
60
+ <Icon icon='ph--chat--regular' aria-hidden={true} />
61
+ </button>
62
+ </div>
63
+ )}
64
+ {children}
65
+ </div>
66
+ );
67
+ };
68
+
69
+ const createThreadDecoration = (cellIndex: string, threadId: string, sheetId: string): Decoration => {
70
+ return {
71
+ type: 'comment',
72
+ cellIndex,
73
+ decorate: (props) => <ThreadedCellWrapper {...props} />,
74
+ };
75
+ };
76
+
77
+ // TODO(burdon): Factor out hooks.
78
+
79
+ const useUpdateCursorOnThreadSelection = () => {
80
+ const { setCursor, model } = useSheetContext();
81
+
82
+ const handleScrollIntoView: IntentResolver = useCallback(
83
+ ({ action, data }) => {
84
+ switch (action) {
85
+ case LayoutAction.SCROLL_INTO_VIEW: {
86
+ if (!data?.id || data?.cursor === undefined || data?.id !== fullyQualifiedId(model.sheet)) {
87
+ return;
88
+ }
89
+
90
+ // TODO(Zan): Everywhere we refer to the cursor in a thread context should change to `anchor`.
91
+ const cellAddress = addressFromIndex(model.sheet, data.cursor);
92
+ setCursor(cellAddress);
93
+ }
94
+ }
95
+ },
96
+ [model.sheet, setCursor],
97
+ );
98
+
99
+ useIntentResolver(SHEET_PLUGIN, handleScrollIntoView);
100
+ };
101
+
102
+ const useSelectThreadOnCursorChange = () => {
103
+ const { cursor, model } = useSheetContext();
104
+ const dispatch = useIntentDispatcher();
105
+
106
+ const activeThreads = useMemo(
107
+ () =>
108
+ model.sheet.threads?.filter(
109
+ (thread): thread is NonNullable<typeof thread> => !!thread && thread.status === 'active',
110
+ ) ?? [],
111
+ [JSON.stringify(model.sheet.threads)],
112
+ );
113
+
114
+ const activeThreadAddresses = useMemo(
115
+ () =>
116
+ activeThreads
117
+ .map((thread) => thread.anchor)
118
+ .filter((anchor): anchor is NonNullable<typeof anchor> => anchor !== undefined)
119
+ .map((anchor) => addressFromIndex(model.sheet, anchor)),
120
+ [activeThreads, model.sheet],
121
+ );
122
+
123
+ const selectClosestThread = useCallback(
124
+ (cellAddress: CellAddress) => {
125
+ if (!cellAddress || !activeThreads) {
126
+ return;
127
+ }
128
+
129
+ const closestThreadAnchor = closest(cellAddress, activeThreadAddresses);
130
+ if (closestThreadAnchor) {
131
+ const closestThread = activeThreads.find(
132
+ (thread) => thread && thread.anchor === addressToIndex(model.sheet, closestThreadAnchor),
133
+ );
134
+
135
+ if (closestThread) {
136
+ void dispatch([
137
+ { action: 'dxos.org/plugin/thread/action/select', data: { current: fullyQualifiedId(closestThread) } },
138
+ ]);
139
+ }
140
+ }
141
+ },
142
+ [dispatch, activeThreads, activeThreadAddresses, model.sheet],
143
+ );
144
+
145
+ const debounced = useMemo(() => {
146
+ return debounce((cursor: CellAddress) => requestAnimationFrame(() => selectClosestThread(cursor)), 50);
147
+ }, [selectClosestThread]);
148
+
149
+ useEffect(() => {
150
+ if (!cursor) {
151
+ return;
152
+ }
153
+ debounced(cursor);
154
+ }, [cursor, selectClosestThread]);
155
+ };
156
+
157
+ const useThreadDecorations = () => {
158
+ const { decorations, model } = useSheetContext();
159
+ const sheet = useMemo(() => model.sheet, [model.sheet]);
160
+ const sheetId = useMemo(() => fullyQualifiedId(sheet), [sheet]);
161
+
162
+ useEffect(() => {
163
+ const unsubscribe = effect(() => {
164
+ const activeThreadAnchors = new Set<string>();
165
+ if (!sheet.threads) {
166
+ return;
167
+ }
168
+
169
+ // Process active threads
170
+ for (const thread of sheet.threads) {
171
+ if (!thread || thread.anchor === undefined || thread.status === 'resolved') {
172
+ continue;
173
+ }
174
+
175
+ activeThreadAnchors.add(thread.anchor);
176
+ const index = thread.anchor;
177
+
178
+ // Add decoration only if it doesn't already exist
179
+ const existingDecorations = decorations.getDecorationsForCell(index);
180
+ if (!existingDecorations || !existingDecorations.some((d) => d.type === 'comment')) {
181
+ decorations.addDecoration(index, createThreadDecoration(index, thread.id, sheetId));
182
+ }
183
+ }
184
+
185
+ // Remove decorations for resolved or deleted threads
186
+ for (const decoration of decorations.getAllDecorations()) {
187
+ if (decoration.type !== 'comment') {
188
+ continue;
189
+ }
190
+
191
+ if (!activeThreadAnchors.has(decoration.cellIndex)) {
192
+ decorations.removeDecoration(decoration.cellIndex, 'comment');
193
+ }
194
+ }
195
+ });
196
+
197
+ return () => unsubscribe();
198
+ });
199
+ };
200
+
201
+ export const useThreads = () => {
202
+ useUpdateCursorOnThreadSelection();
203
+ useSelectThreadOnCursorChange();
204
+ useThreadDecorations();
205
+ };
@@ -2,30 +2,82 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import React from 'react';
5
+ import React, { useCallback } from 'react';
6
6
 
7
- import { type LayoutCoordinate } from '@dxos/app-framework';
8
- import { mx } from '@dxos/react-ui-theme';
7
+ import { useIntentDispatcher } from '@dxos/app-framework';
8
+ import { fullyQualifiedId } from '@dxos/react-client/echo';
9
+ import { useAttendableAttributes, useAttention } from '@dxos/react-ui-attention';
10
+ import { focusRing, mx } from '@dxos/react-ui-theme';
9
11
 
10
12
  import { Sheet, type SheetRootProps } from './Sheet';
13
+ import { Toolbar, type ToolbarAction } from './Toolbar';
14
+
15
+ // TODO(Zan): Factor out, copied this from MarkdownPlugin.
16
+ export const sectionToolbarLayout =
17
+ 'bs-[--rail-action] bg-[--sticky-bg] sticky block-start-0 __-block-start-px transition-opacity';
18
+
19
+ const SheetContainer = ({ graph, sheet, role }: SheetRootProps & { role?: string }) => {
20
+ const dispatch = useIntentDispatcher();
21
+
22
+ const id = fullyQualifiedId(sheet);
23
+ const attendableAttrs = useAttendableAttributes(id);
24
+ const { hasAttention } = useAttention(id);
25
+
26
+ // TODO(Zan): Centralise the toolbar action handler. Current implementation in stories.
27
+ const handleAction = useCallback(
28
+ (action: ToolbarAction) => {
29
+ switch (action.type) {
30
+ case 'comment': {
31
+ // TODO(Zan): We shouldn't hardcode the action ID.
32
+ void dispatch({
33
+ action: 'dxos.org/plugin/thread/action/create',
34
+ data: {
35
+ cursor: action.anchor,
36
+ name: action.cellContent,
37
+ subject: sheet,
38
+ },
39
+ });
40
+ }
41
+ }
42
+ },
43
+ [sheet, dispatch],
44
+ );
11
45
 
12
- const SheetContainer = ({
13
- sheet,
14
- space,
15
- role,
16
- remoteFunctionUrl,
17
- }: SheetRootProps & { role?: string; coordinate?: LayoutCoordinate }) => {
18
46
  return (
19
47
  <div
20
48
  role='none'
21
- className={mx(
22
- 'flex',
23
- role === 'article' && 'row-span-2',
24
- role === 'section' && 'aspect-square border-y border-is border-separator',
25
- )}
49
+ className={role === 'article' ? 'row-span-2 grid grid-rows-subgrid' : undefined}
50
+ {...(role === 'article' && attendableAttrs)}
26
51
  >
27
- <Sheet.Root sheet={sheet} space={space} remoteFunctionUrl={remoteFunctionUrl}>
28
- <Sheet.Main />
52
+ <Sheet.Root graph={graph} sheet={sheet}>
53
+ <div role='none' className={mx('flex flex-0 justify-center overflow-x-auto')}>
54
+ <Toolbar.Root
55
+ onAction={handleAction}
56
+ classNames={mx(
57
+ role === 'section'
58
+ ? ['z-[2] group-focus-within/section:visible', !hasAttention && 'invisible', sectionToolbarLayout]
59
+ : 'attention-surface',
60
+ )}
61
+ >
62
+ {/* TODO(Zan): Restore some of this functionality */}
63
+ {/* <Toolbar.Styles /> */}
64
+ {/* <Toolbar.Format /> */}
65
+ {/* <Toolbar.Alignment /> */}
66
+ <Toolbar.Separator />
67
+ <Toolbar.Actions />
68
+ </Toolbar.Root>
69
+ </div>
70
+ <div
71
+ role='none'
72
+ className={mx(
73
+ role === 'section' && 'aspect-square border-is border-bs border-be border-separator',
74
+ role === 'article' &&
75
+ 'flex is-full overflow-hidden focus-visible:ring-inset row-span-1 data-[toolbar=disabled]:pbs-2 data-[toolbar=disabled]:row-span-2 border-bs border-separator attention-surface',
76
+ focusRing,
77
+ )}
78
+ >
79
+ <Sheet.Main />
80
+ </div>
29
81
  </Sheet.Root>
30
82
  </div>
31
83
  );
@@ -23,22 +23,31 @@ import {
23
23
  type ThemedClassName,
24
24
  useTranslation,
25
25
  } from '@dxos/react-ui';
26
+ import { nonNullable } from '@dxos/util';
26
27
 
27
28
  import { ToolbarButton, ToolbarSeparator, ToolbarToggleButton } from './common';
29
+ import { addressToIndex } from '../../defs';
28
30
  import { SHEET_PLUGIN } from '../../meta';
29
31
  import { type Formatting } from '../../types';
32
+ import { useSheetContext } from '../Sheet/sheet-context';
30
33
 
31
34
  //
32
35
  // Root
33
36
  //
34
37
 
35
- export type ToolbarActionType = 'clear' | 'highlight' | 'left' | 'center' | 'right' | 'date' | 'currency';
38
+ export type ToolbarAction =
39
+ | { type: 'clear' }
40
+ | { type: 'highlight' }
41
+ | { type: 'left' }
42
+ | { type: 'center' }
43
+ | { type: 'right' }
44
+ | { type: 'date' }
45
+ | { type: 'currency' }
46
+ | { type: 'comment'; anchor: string; cellContent?: string };
36
47
 
37
- export type ToolbarAction = {
38
- type: ToolbarActionType;
39
- };
48
+ export type ToolbarActionType = ToolbarAction['type'];
40
49
 
41
- export type ToolbarActionHandler = ({ type }: ToolbarAction) => void;
50
+ export type ToolbarActionHandler = (action: ToolbarAction) => void;
42
51
 
43
52
  export type ToolbarProps = ThemedClassName<
44
53
  PropsWithChildren<{
@@ -96,7 +105,7 @@ const Format = () => {
96
105
  Icon={Icon}
97
106
  // disabled={state?.blockType === 'codeblock'}
98
107
  // onClick={state ? () => onAction?.({ type, data: !getState(state) }) : undefined}
99
- onClick={() => onAction?.({ type })}
108
+ onClick={() => onAction?.({ type: type as Exclude<typeof type, 'comment'> })}
100
109
  >
101
110
  {t(`toolbar ${type} label`)}
102
111
  </ToolbarToggleButton>
@@ -127,7 +136,7 @@ const Alignment = () => {
127
136
  Icon={Icon}
128
137
  // disabled={state?.blockType === 'codeblock'}
129
138
  // onClick={state ? () => onAction?.({ type, data: !getState(state) }) : undefined}
130
- onClick={() => onAction?.({ type })}
139
+ onClick={() => onAction?.({ type: type as Exclude<typeof type, 'comment'> })}
131
140
  >
132
141
  {t(`toolbar ${type} label`)}
133
142
  </ToolbarToggleButton>
@@ -157,7 +166,7 @@ const Styles = () => {
157
166
  Icon={Icon}
158
167
  // disabled={state?.blockType === 'codeblock'}
159
168
  // onClick={state ? () => onAction?.({ type, data: !getState(state) }) : undefined}
160
- onClick={() => onAction?.({ type })}
169
+ onClick={() => onAction?.({ type: type as Exclude<typeof type, 'comment'> })}
161
170
  >
162
171
  {t(`toolbar ${type} label`)}
163
172
  </ToolbarToggleButton>
@@ -171,17 +180,49 @@ const Styles = () => {
171
180
  //
172
181
 
173
182
  const Actions = () => {
174
- // const { onAction } = useToolbarContext('Actions');
183
+ const { onAction } = useToolbarContext('Actions');
184
+ const { cursor, range, model } = useSheetContext();
175
185
  const { t } = useTranslation(SHEET_PLUGIN);
186
+
187
+ const overlapsCommentAnchor = (model.sheet.threads ?? [])
188
+ .filter(nonNullable)
189
+ .filter((thread) => thread.status !== 'resolved')
190
+ .some((thread) => {
191
+ if (!cursor) {
192
+ return false;
193
+ }
194
+ return addressToIndex(model.sheet, cursor) === thread.anchor;
195
+ });
196
+
197
+ const hasCursor = !!cursor;
198
+ const cursorOnly = hasCursor && !range && !overlapsCommentAnchor;
199
+
200
+ const tooltipLabelKey = !hasCursor
201
+ ? 'no cursor label'
202
+ : overlapsCommentAnchor
203
+ ? 'selection overlaps existing comment label'
204
+ : range
205
+ ? 'comment ranges not supported label'
206
+ : 'comment label';
207
+
176
208
  return (
177
209
  <ToolbarButton
178
210
  value='comment'
179
211
  Icon={ChatText}
180
212
  data-testid='editor.toolbar.comment'
181
- // onClick={() => onAction?.({ type: 'comment' })}
182
- // disabled={!state || state.comment || !state.selection}
213
+ onClick={() => {
214
+ if (!cursor) {
215
+ return;
216
+ }
217
+ return onAction?.({
218
+ type: 'comment',
219
+ anchor: addressToIndex(model.sheet, cursor),
220
+ cellContent: model.getCellText(cursor),
221
+ });
222
+ }}
223
+ disabled={!cursorOnly || overlapsCommentAnchor}
183
224
  >
184
- {t('comment label')}
225
+ {t(tooltipLabelKey)}
185
226
  </ToolbarButton>
186
227
  );
187
228
  };
@@ -5,6 +5,7 @@
5
5
  import React from 'react';
6
6
 
7
7
  export * from './ComputeGraph';
8
+ export * from './Sheet';
8
9
 
9
10
  // Lazily load components for content surfaces.
10
11
  export const SheetContainer = React.lazy(() => import('./SheetContainer'));
@@ -0,0 +1,6 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ export * from './types';
6
+ export * from './util';
@@ -3,15 +3,14 @@
3
3
  //
4
4
 
5
5
  import { getIndices, sortByIndex, getIndicesBelow, getIndicesAbove, getIndicesBetween } from '@tldraw/indices';
6
- import { expect } from 'chai';
7
- import { describe, test } from 'vitest';
6
+ import { describe, expect, test } from 'vitest';
8
7
 
9
8
  import { inRange, addressFromA1Notation, addressToA1Notation, rangeFromA1Notation, rangeToA1Notation } from './types';
10
9
 
11
10
  describe('cell', () => {
12
11
  test('posToA1Notation', () => {
13
- expect(addressToA1Notation({ column: 0, row: 0 })).to.eq('A1');
14
- expect(addressFromA1Notation('C2')).to.deep.eq({ column: 2, row: 1 });
12
+ expect(addressToA1Notation({ col: 0, row: 0 })).to.eq('A1');
13
+ expect(addressFromA1Notation('C2')).to.deep.eq({ col: 2, row: 1 });
15
14
  });
16
15
 
17
16
  test('rangeToA1Notation', () => {
@@ -56,19 +55,19 @@ describe('cell', () => {
56
55
  // Values.
57
56
  const cells: Record<string, any> = {};
58
57
  const setCell = (cell: string, value: any) => {
59
- const { column, row } = addressFromA1Notation(cell);
58
+ const { col, row } = addressFromA1Notation(cell);
60
59
  // Reallocate if > current bounds.
61
- if (column >= columns.length) {
62
- insertIndex(columns, column);
60
+ if (col >= columns.length) {
61
+ insertIndex(columns, col);
63
62
  }
64
63
  if (row >= rows.length) {
65
64
  insertIndex(rows, row);
66
65
  }
67
- const index = `${columns[column]}@${rows[row]}`;
66
+ const index = `${columns[col]}@${rows[row]}`;
68
67
  cells[index] = value;
69
68
  };
70
69
 
71
- expect(addressFromA1Notation('A1')).to.deep.eq({ column: 0, row: 0 });
70
+ expect(addressFromA1Notation('A1')).to.deep.eq({ col: 0, row: 0 });
72
71
 
73
72
  expect(columns).to.deep.eq(['a1', 'a2', 'a3', 'a4', 'a5']);
74
73
  insertIndex(columns, 7);
@@ -4,33 +4,43 @@
4
4
 
5
5
  import { invariant } from '@dxos/invariant';
6
6
 
7
- export const MAX_COLUMNS = 26 * 26;
7
+ export const DEFAULT_ROWS = 50;
8
+ export const DEFAULT_COLUMNS = 26;
9
+
10
+ export const MAX_ROWS = 500;
11
+ export const MAX_COLUMNS = 26 * 2;
12
+
13
+ export type CellAddress = { col: number; row: number };
8
14
 
9
- export type CellAddress = { column: number; row: number };
10
15
  export type CellRange = { from: CellAddress; to?: CellAddress };
11
16
 
17
+ export type CellIndex = string;
18
+
19
+ export type CellContentValue = number | string | boolean | null;
20
+
12
21
  export const posEquals = (a: CellAddress | undefined, b: CellAddress | undefined) => {
13
- return a?.column === b?.column && a?.row === b?.row;
22
+ return a?.col === b?.col && a?.row === b?.row;
14
23
  };
15
24
 
16
- export const columnLetter = (column: number): string => {
17
- invariant(column < MAX_COLUMNS, `Invalid column: ${column}`);
25
+ export const columnLetter = (col: number): string => {
26
+ invariant(col < MAX_COLUMNS, `Invalid column: ${col}`);
18
27
  return (
19
- (column >= 26 ? String.fromCharCode('A'.charCodeAt(0) + Math.floor(column / 26) - 1) : '') +
20
- String.fromCharCode('A'.charCodeAt(0) + (column % 26))
28
+ (col >= 26 ? String.fromCharCode('A'.charCodeAt(0) + Math.floor(col / 26) - 1) : '') +
29
+ String.fromCharCode('A'.charCodeAt(0) + (col % 26))
21
30
  );
22
31
  };
23
32
 
24
- export const addressToA1Notation = ({ column, row }: CellAddress): string => {
25
- return `${columnLetter(column)}${row + 1}`;
33
+ export const addressToA1Notation = ({ col, row }: CellAddress): string => {
34
+ return `${columnLetter(col)}${row + 1}`;
26
35
  };
27
36
 
37
+ // TODO(burdon): See simpleCellAddressFromString
28
38
  export const addressFromA1Notation = (ref: string): CellAddress => {
29
39
  const match = ref.match(/([A-Z]+)(\d+)/);
30
40
  invariant(match, `Invalid notation: ${ref}`);
31
41
  return {
32
42
  row: parseInt(match[2], 10) - 1,
33
- column: match[1].split('').reduce((acc, c) => acc * 26 + c.charCodeAt(0) - 'A'.charCodeAt(0) + 1, 0) - 1,
43
+ col: match[1].split('').reduce((acc, c) => acc * 26 + c.charCodeAt(0) - 'A'.charCodeAt(0) + 1, 0) - 1,
34
44
  };
35
45
  };
36
46
 
@@ -59,13 +69,13 @@ export const inRange = (range: CellRange | undefined, cell: CellAddress): boolea
59
69
  return false;
60
70
  }
61
71
 
62
- const { column: c1, row: r1 } = from;
63
- const { column: c2, row: r2 } = to;
72
+ const { col: c1, row: r1 } = from;
73
+ const { col: c2, row: r2 } = to;
64
74
  const cMin = Math.min(c1, c2);
65
75
  const cMax = Math.max(c1, c2);
66
76
  const rMin = Math.min(r1, r2);
67
77
  const rMax = Math.max(r1, r2);
68
78
 
69
- const { column, row } = cell;
70
- return column >= cMin && column <= cMax && row >= rMin && row <= rMax;
79
+ const { col, row } = cell;
80
+ return col >= cMin && col <= cMax && row >= rMin && row <= rMax;
71
81
  };