@bendyline/squisq-editor-react 1.4.0 → 1.5.0

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 (446) hide show
  1. package/dist/DocumentSettingsDialog.d.ts +26 -0
  2. package/dist/DocumentSettingsDialog.d.ts.map +1 -0
  3. package/dist/DocumentSettingsDialog.js +115 -0
  4. package/dist/DocumentSettingsDialog.js.map +1 -0
  5. package/dist/EditorContext.d.ts +248 -4
  6. package/dist/EditorContext.d.ts.map +1 -1
  7. package/dist/EditorContext.js +248 -10
  8. package/dist/EditorContext.js.map +1 -1
  9. package/dist/EditorShell.d.ts +173 -4
  10. package/dist/EditorShell.d.ts.map +1 -1
  11. package/dist/EditorShell.js +110 -10
  12. package/dist/EditorShell.js.map +1 -1
  13. package/dist/EmojiPicker.d.ts +50 -0
  14. package/dist/EmojiPicker.d.ts.map +1 -0
  15. package/dist/EmojiPicker.js +182 -0
  16. package/dist/EmojiPicker.js.map +1 -0
  17. package/dist/ImageEditor.d.ts +68 -0
  18. package/dist/ImageEditor.d.ts.map +1 -0
  19. package/dist/ImageEditor.js +166 -0
  20. package/dist/ImageEditor.js.map +1 -0
  21. package/dist/ImageNodeView.d.ts +13 -1
  22. package/dist/ImageNodeView.d.ts.map +1 -1
  23. package/dist/ImageNodeView.js +172 -19
  24. package/dist/ImageNodeView.js.map +1 -1
  25. package/dist/ImageViewer.d.ts +26 -0
  26. package/dist/ImageViewer.d.ts.map +1 -0
  27. package/dist/ImageViewer.js +119 -0
  28. package/dist/ImageViewer.js.map +1 -0
  29. package/dist/InlineIcon.d.ts +17 -0
  30. package/dist/InlineIcon.d.ts.map +1 -0
  31. package/dist/InlineIcon.js +72 -0
  32. package/dist/InlineIcon.js.map +1 -0
  33. package/dist/InlinePreviewGutter.d.ts +52 -0
  34. package/dist/InlinePreviewGutter.d.ts.map +1 -0
  35. package/dist/InlinePreviewGutter.js +397 -0
  36. package/dist/InlinePreviewGutter.js.map +1 -0
  37. package/dist/LinkDialog.d.ts +43 -0
  38. package/dist/LinkDialog.d.ts.map +1 -0
  39. package/dist/LinkDialog.js +102 -0
  40. package/dist/LinkDialog.js.map +1 -0
  41. package/dist/MentionExtension.js +10 -7
  42. package/dist/MentionExtension.js.map +1 -1
  43. package/dist/OutlinePanel.d.ts +17 -0
  44. package/dist/OutlinePanel.d.ts.map +1 -0
  45. package/dist/OutlinePanel.js +167 -0
  46. package/dist/OutlinePanel.js.map +1 -0
  47. package/dist/PlainHtmlPreview.d.ts +50 -0
  48. package/dist/PlainHtmlPreview.d.ts.map +1 -0
  49. package/dist/PlainHtmlPreview.js +155 -0
  50. package/dist/PlainHtmlPreview.js.map +1 -0
  51. package/dist/PreviewControls.d.ts +15 -1
  52. package/dist/PreviewControls.d.ts.map +1 -1
  53. package/dist/PreviewControls.js +75 -18
  54. package/dist/PreviewControls.js.map +1 -1
  55. package/dist/PreviewPanel.d.ts +11 -10
  56. package/dist/PreviewPanel.d.ts.map +1 -1
  57. package/dist/PreviewPanel.js +20 -17
  58. package/dist/PreviewPanel.js.map +1 -1
  59. package/dist/RawEditor.d.ts.map +1 -1
  60. package/dist/RawEditor.js +198 -4
  61. package/dist/RawEditor.js.map +1 -1
  62. package/dist/RecorderEntry.d.ts +24 -0
  63. package/dist/RecorderEntry.d.ts.map +1 -0
  64. package/dist/RecorderEntry.js +139 -0
  65. package/dist/RecorderEntry.js.map +1 -0
  66. package/dist/TemplateAnnotation.d.ts.map +1 -1
  67. package/dist/TemplateAnnotation.js +32 -6
  68. package/dist/TemplateAnnotation.js.map +1 -1
  69. package/dist/TemplatePicker.d.ts +53 -0
  70. package/dist/TemplatePicker.d.ts.map +1 -0
  71. package/dist/TemplatePicker.js +388 -0
  72. package/dist/TemplatePicker.js.map +1 -0
  73. package/dist/ThemeCustomizerPanel.d.ts +32 -0
  74. package/dist/ThemeCustomizerPanel.d.ts.map +1 -0
  75. package/dist/ThemeCustomizerPanel.js +256 -0
  76. package/dist/ThemeCustomizerPanel.js.map +1 -0
  77. package/dist/ThemePicker.d.ts +33 -0
  78. package/dist/ThemePicker.d.ts.map +1 -0
  79. package/dist/ThemePicker.js +148 -0
  80. package/dist/ThemePicker.js.map +1 -0
  81. package/dist/Toolbar.d.ts.map +1 -1
  82. package/dist/Toolbar.js +508 -33
  83. package/dist/Toolbar.js.map +1 -1
  84. package/dist/VersionHistoryPanel.d.ts +14 -0
  85. package/dist/VersionHistoryPanel.d.ts.map +1 -0
  86. package/dist/VersionHistoryPanel.js +147 -0
  87. package/dist/VersionHistoryPanel.js.map +1 -0
  88. package/dist/ViewMenuPanel.d.ts +13 -0
  89. package/dist/ViewMenuPanel.d.ts.map +1 -0
  90. package/dist/ViewMenuPanel.js +58 -0
  91. package/dist/ViewMenuPanel.js.map +1 -0
  92. package/dist/WysiwygEditor.d.ts.map +1 -1
  93. package/dist/WysiwygEditor.js +198 -9
  94. package/dist/WysiwygEditor.js.map +1 -1
  95. package/dist/__tests__/detectMarkdown.test.js +0 -14
  96. package/dist/__tests__/detectMarkdown.test.js.map +1 -1
  97. package/dist/__tests__/documentSettingsDialog.test.d.ts +2 -0
  98. package/dist/__tests__/documentSettingsDialog.test.d.ts.map +1 -0
  99. package/dist/__tests__/documentSettingsDialog.test.js +132 -0
  100. package/dist/__tests__/documentSettingsDialog.test.js.map +1 -0
  101. package/dist/__tests__/emojiPicker.test.d.ts +2 -0
  102. package/dist/__tests__/emojiPicker.test.d.ts.map +1 -0
  103. package/dist/__tests__/emojiPicker.test.js +111 -0
  104. package/dist/__tests__/emojiPicker.test.js.map +1 -0
  105. package/dist/__tests__/fileKind.test.js +13 -0
  106. package/dist/__tests__/fileKind.test.js.map +1 -1
  107. package/dist/__tests__/imageEditAffordance.test.d.ts +2 -0
  108. package/dist/__tests__/imageEditAffordance.test.d.ts.map +1 -0
  109. package/dist/__tests__/imageEditAffordance.test.js +188 -0
  110. package/dist/__tests__/imageEditAffordance.test.js.map +1 -0
  111. package/dist/__tests__/imageEditorShell.test.d.ts +2 -0
  112. package/dist/__tests__/imageEditorShell.test.d.ts.map +1 -0
  113. package/dist/__tests__/imageEditorShell.test.js +52 -0
  114. package/dist/__tests__/imageEditorShell.test.js.map +1 -0
  115. package/dist/__tests__/imageEditorState.test.d.ts +3 -0
  116. package/dist/__tests__/imageEditorState.test.d.ts.map +1 -0
  117. package/dist/__tests__/imageEditorState.test.js +148 -0
  118. package/dist/__tests__/imageEditorState.test.js.map +1 -0
  119. package/dist/__tests__/inlinePreviewGutter.test.d.ts +2 -0
  120. package/dist/__tests__/inlinePreviewGutter.test.d.ts.map +1 -0
  121. package/dist/__tests__/inlinePreviewGutter.test.js +51 -0
  122. package/dist/__tests__/inlinePreviewGutter.test.js.map +1 -0
  123. package/dist/__tests__/inlinePreviewGutterAllBlocks.test.d.ts +2 -0
  124. package/dist/__tests__/inlinePreviewGutterAllBlocks.test.d.ts.map +1 -0
  125. package/dist/__tests__/inlinePreviewGutterAllBlocks.test.js +63 -0
  126. package/dist/__tests__/inlinePreviewGutterAllBlocks.test.js.map +1 -0
  127. package/dist/__tests__/jsonEditor.test.d.ts +2 -0
  128. package/dist/__tests__/jsonEditor.test.d.ts.map +1 -0
  129. package/dist/__tests__/jsonEditor.test.js +134 -0
  130. package/dist/__tests__/jsonEditor.test.js.map +1 -0
  131. package/dist/__tests__/layersPanel.test.d.ts +2 -0
  132. package/dist/__tests__/layersPanel.test.d.ts.map +1 -0
  133. package/dist/__tests__/layersPanel.test.js +84 -0
  134. package/dist/__tests__/layersPanel.test.js.map +1 -0
  135. package/dist/__tests__/linkDialogDocPicker.test.d.ts +7 -0
  136. package/dist/__tests__/linkDialogDocPicker.test.d.ts.map +1 -0
  137. package/dist/__tests__/linkDialogDocPicker.test.js +75 -0
  138. package/dist/__tests__/linkDialogDocPicker.test.js.map +1 -0
  139. package/dist/__tests__/outlinePanel.test.d.ts +2 -0
  140. package/dist/__tests__/outlinePanel.test.d.ts.map +1 -0
  141. package/dist/__tests__/outlinePanel.test.js +68 -0
  142. package/dist/__tests__/outlinePanel.test.js.map +1 -0
  143. package/dist/__tests__/plainHtmlPreview.test.d.ts +2 -0
  144. package/dist/__tests__/plainHtmlPreview.test.d.ts.map +1 -0
  145. package/dist/__tests__/plainHtmlPreview.test.js +87 -0
  146. package/dist/__tests__/plainHtmlPreview.test.js.map +1 -0
  147. package/dist/__tests__/propertiesPanel.test.d.ts +2 -0
  148. package/dist/__tests__/propertiesPanel.test.d.ts.map +1 -0
  149. package/dist/__tests__/propertiesPanel.test.js +64 -0
  150. package/dist/__tests__/propertiesPanel.test.js.map +1 -0
  151. package/dist/__tests__/recorderFormats.test.d.ts +2 -0
  152. package/dist/__tests__/recorderFormats.test.d.ts.map +1 -0
  153. package/dist/__tests__/recorderFormats.test.js +121 -0
  154. package/dist/__tests__/recorderFormats.test.js.map +1 -0
  155. package/dist/__tests__/recorderTimingJson.test.d.ts +2 -0
  156. package/dist/__tests__/recorderTimingJson.test.d.ts.map +1 -0
  157. package/dist/__tests__/recorderTimingJson.test.js +37 -0
  158. package/dist/__tests__/recorderTimingJson.test.js.map +1 -0
  159. package/dist/__tests__/templateAnnotationRoundTrip.test.d.ts +2 -0
  160. package/dist/__tests__/templateAnnotationRoundTrip.test.d.ts.map +1 -0
  161. package/dist/__tests__/templateAnnotationRoundTrip.test.js +31 -0
  162. package/dist/__tests__/templateAnnotationRoundTrip.test.js.map +1 -0
  163. package/dist/__tests__/tiptapBridge.test.js +13 -0
  164. package/dist/__tests__/tiptapBridge.test.js.map +1 -1
  165. package/dist/__tests__/useImageEditor.test.d.ts +2 -0
  166. package/dist/__tests__/useImageEditor.test.d.ts.map +1 -0
  167. package/dist/__tests__/useImageEditor.test.js +131 -0
  168. package/dist/__tests__/useImageEditor.test.js.map +1 -0
  169. package/dist/__tests__/useMediaRecorder.test.d.ts +2 -0
  170. package/dist/__tests__/useMediaRecorder.test.d.ts.map +1 -0
  171. package/dist/__tests__/useMediaRecorder.test.js +153 -0
  172. package/dist/__tests__/useMediaRecorder.test.js.map +1 -0
  173. package/dist/__tests__/versionHistory.test.d.ts +2 -0
  174. package/dist/__tests__/versionHistory.test.d.ts.map +1 -0
  175. package/dist/__tests__/versionHistory.test.js +124 -0
  176. package/dist/__tests__/versionHistory.test.js.map +1 -0
  177. package/dist/blockSlice.d.ts +24 -0
  178. package/dist/blockSlice.d.ts.map +1 -0
  179. package/dist/blockSlice.js +63 -0
  180. package/dist/blockSlice.js.map +1 -0
  181. package/dist/buildPreviewDoc.d.ts.map +1 -1
  182. package/dist/buildPreviewDoc.js +52 -2
  183. package/dist/buildPreviewDoc.js.map +1 -1
  184. package/dist/emojiData.d.ts +81 -0
  185. package/dist/emojiData.d.ts.map +1 -0
  186. package/dist/emojiData.js +1283 -0
  187. package/dist/emojiData.js.map +1 -0
  188. package/dist/fileKind.d.ts +6 -2
  189. package/dist/fileKind.d.ts.map +1 -1
  190. package/dist/fileKind.js +25 -4
  191. package/dist/fileKind.js.map +1 -1
  192. package/dist/hooks/useFileDrop.d.ts.map +1 -1
  193. package/dist/hooks/useFileDrop.js +40 -4
  194. package/dist/hooks/useFileDrop.js.map +1 -1
  195. package/dist/imageEditor/CanvasSurface.d.ts +31 -0
  196. package/dist/imageEditor/CanvasSurface.d.ts.map +1 -0
  197. package/dist/imageEditor/CanvasSurface.js +264 -0
  198. package/dist/imageEditor/CanvasSurface.js.map +1 -0
  199. package/dist/imageEditor/ImageVersionHistoryDropdown.d.ts +39 -0
  200. package/dist/imageEditor/ImageVersionHistoryDropdown.d.ts.map +1 -0
  201. package/dist/imageEditor/ImageVersionHistoryDropdown.js +283 -0
  202. package/dist/imageEditor/ImageVersionHistoryDropdown.js.map +1 -0
  203. package/dist/imageEditor/LayersPanel.d.ts +14 -0
  204. package/dist/imageEditor/LayersPanel.d.ts.map +1 -0
  205. package/dist/imageEditor/LayersPanel.js +43 -0
  206. package/dist/imageEditor/LayersPanel.js.map +1 -0
  207. package/dist/imageEditor/PropertiesPanel.d.ts +14 -0
  208. package/dist/imageEditor/PropertiesPanel.d.ts.map +1 -0
  209. package/dist/imageEditor/PropertiesPanel.js +97 -0
  210. package/dist/imageEditor/PropertiesPanel.js.map +1 -0
  211. package/dist/imageEditor/Toolbar.d.ts +30 -0
  212. package/dist/imageEditor/Toolbar.d.ts.map +1 -0
  213. package/dist/imageEditor/Toolbar.js +108 -0
  214. package/dist/imageEditor/Toolbar.js.map +1 -0
  215. package/dist/imageEditor/icons.d.ts +24 -0
  216. package/dist/imageEditor/icons.d.ts.map +1 -0
  217. package/dist/imageEditor/icons.js +45 -0
  218. package/dist/imageEditor/icons.js.map +1 -0
  219. package/dist/imageEditor/layers/EditorImageLayer.d.ts +16 -0
  220. package/dist/imageEditor/layers/EditorImageLayer.d.ts.map +1 -0
  221. package/dist/imageEditor/layers/EditorImageLayer.js +37 -0
  222. package/dist/imageEditor/layers/EditorImageLayer.js.map +1 -0
  223. package/dist/imageEditor/layers/EditorShapeLayer.d.ts +15 -0
  224. package/dist/imageEditor/layers/EditorShapeLayer.d.ts.map +1 -0
  225. package/dist/imageEditor/layers/EditorShapeLayer.js +20 -0
  226. package/dist/imageEditor/layers/EditorShapeLayer.js.map +1 -0
  227. package/dist/imageEditor/layers/EditorTextLayer.d.ts +18 -0
  228. package/dist/imageEditor/layers/EditorTextLayer.d.ts.map +1 -0
  229. package/dist/imageEditor/layers/EditorTextLayer.js +13 -0
  230. package/dist/imageEditor/layers/EditorTextLayer.js.map +1 -0
  231. package/dist/imageEditor/layers/SelectionHandles.d.ts +17 -0
  232. package/dist/imageEditor/layers/SelectionHandles.d.ts.map +1 -0
  233. package/dist/imageEditor/layers/SelectionHandles.js +19 -0
  234. package/dist/imageEditor/layers/SelectionHandles.js.map +1 -0
  235. package/dist/imageEditor/state.d.ts +76 -0
  236. package/dist/imageEditor/state.d.ts.map +1 -0
  237. package/dist/imageEditor/state.js +87 -0
  238. package/dist/imageEditor/state.js.map +1 -0
  239. package/dist/imageEditor/useImageEditor.d.ts +53 -0
  240. package/dist/imageEditor/useImageEditor.d.ts.map +1 -0
  241. package/dist/imageEditor/useImageEditor.js +244 -0
  242. package/dist/imageEditor/useImageEditor.js.map +1 -0
  243. package/dist/imageEditor/useImageEditorTokens.d.ts +16 -0
  244. package/dist/imageEditor/useImageEditorTokens.d.ts.map +1 -0
  245. package/dist/imageEditor/useImageEditorTokens.js +45 -0
  246. package/dist/imageEditor/useImageEditorTokens.js.map +1 -0
  247. package/dist/index.d.ts +48 -1
  248. package/dist/index.d.ts.map +1 -1
  249. package/dist/index.js +36 -0
  250. package/dist/index.js.map +1 -1
  251. package/dist/jsonEditor/EmbeddedRichTextField.d.ts +15 -0
  252. package/dist/jsonEditor/EmbeddedRichTextField.d.ts.map +1 -0
  253. package/dist/jsonEditor/EmbeddedRichTextField.js +74 -0
  254. package/dist/jsonEditor/EmbeddedRichTextField.js.map +1 -0
  255. package/dist/jsonEditor/JsonEditor.d.ts +36 -0
  256. package/dist/jsonEditor/JsonEditor.d.ts.map +1 -0
  257. package/dist/jsonEditor/JsonEditor.js +15 -0
  258. package/dist/jsonEditor/JsonEditor.js.map +1 -0
  259. package/dist/jsonEditor/JsonEditorContext.d.ts +28 -0
  260. package/dist/jsonEditor/JsonEditorContext.d.ts.map +1 -0
  261. package/dist/jsonEditor/JsonEditorContext.js +41 -0
  262. package/dist/jsonEditor/JsonEditorContext.js.map +1 -0
  263. package/dist/jsonEditor/RenderNode.d.ts +16 -0
  264. package/dist/jsonEditor/RenderNode.d.ts.map +1 -0
  265. package/dist/jsonEditor/RenderNode.js +32 -0
  266. package/dist/jsonEditor/RenderNode.js.map +1 -0
  267. package/dist/jsonEditor/editors.d.ts +36 -0
  268. package/dist/jsonEditor/editors.d.ts.map +1 -0
  269. package/dist/jsonEditor/editors.js +347 -0
  270. package/dist/jsonEditor/editors.js.map +1 -0
  271. package/dist/jsonEditor/index.d.ts +3 -0
  272. package/dist/jsonEditor/index.d.ts.map +1 -0
  273. package/dist/jsonEditor/index.js +2 -0
  274. package/dist/jsonEditor/index.js.map +1 -0
  275. package/dist/jsonEditor/useJsonEditorTokens.d.ts +13 -0
  276. package/dist/jsonEditor/useJsonEditorTokens.d.ts.map +1 -0
  277. package/dist/jsonEditor/useJsonEditorTokens.js +38 -0
  278. package/dist/jsonEditor/useJsonEditorTokens.js.map +1 -0
  279. package/dist/recorder/RecorderButton.d.ts +31 -0
  280. package/dist/recorder/RecorderButton.d.ts.map +1 -0
  281. package/dist/recorder/RecorderButton.js +24 -0
  282. package/dist/recorder/RecorderButton.js.map +1 -0
  283. package/dist/recorder/RecorderModal.d.ts +59 -0
  284. package/dist/recorder/RecorderModal.d.ts.map +1 -0
  285. package/dist/recorder/RecorderModal.js +333 -0
  286. package/dist/recorder/RecorderModal.js.map +1 -0
  287. package/dist/recorder/RecorderPanel.d.ts +25 -0
  288. package/dist/recorder/RecorderPanel.d.ts.map +1 -0
  289. package/dist/recorder/RecorderPanel.js +30 -0
  290. package/dist/recorder/RecorderPanel.js.map +1 -0
  291. package/dist/recorder/formats.d.ts +51 -0
  292. package/dist/recorder/formats.d.ts.map +1 -0
  293. package/dist/recorder/formats.js +144 -0
  294. package/dist/recorder/formats.js.map +1 -0
  295. package/dist/recorder/hooks/useMediaRecorder.d.ts +90 -0
  296. package/dist/recorder/hooks/useMediaRecorder.d.ts.map +1 -0
  297. package/dist/recorder/hooks/useMediaRecorder.js +277 -0
  298. package/dist/recorder/hooks/useMediaRecorder.js.map +1 -0
  299. package/dist/recorder/hooks/useStreamPreview.d.ts +22 -0
  300. package/dist/recorder/hooks/useStreamPreview.d.ts.map +1 -0
  301. package/dist/recorder/hooks/useStreamPreview.js +44 -0
  302. package/dist/recorder/hooks/useStreamPreview.js.map +1 -0
  303. package/dist/recorder/sources/cameraStream.d.ts +22 -0
  304. package/dist/recorder/sources/cameraStream.d.ts.map +1 -0
  305. package/dist/recorder/sources/cameraStream.js +24 -0
  306. package/dist/recorder/sources/cameraStream.js.map +1 -0
  307. package/dist/recorder/sources/micStream.d.ts +15 -0
  308. package/dist/recorder/sources/micStream.d.ts.map +1 -0
  309. package/dist/recorder/sources/micStream.js +24 -0
  310. package/dist/recorder/sources/micStream.js.map +1 -0
  311. package/dist/recorder/sources/screenStream.d.ts +53 -0
  312. package/dist/recorder/sources/screenStream.d.ts.map +1 -0
  313. package/dist/recorder/sources/screenStream.js +114 -0
  314. package/dist/recorder/sources/screenStream.js.map +1 -0
  315. package/dist/recorder/timingJson.d.ts +51 -0
  316. package/dist/recorder/timingJson.d.ts.map +1 -0
  317. package/dist/recorder/timingJson.js +42 -0
  318. package/dist/recorder/timingJson.js.map +1 -0
  319. package/dist/tiptap/TiptapAudio.d.ts +26 -0
  320. package/dist/tiptap/TiptapAudio.d.ts.map +1 -0
  321. package/dist/tiptap/TiptapAudio.js +58 -0
  322. package/dist/tiptap/TiptapAudio.js.map +1 -0
  323. package/dist/tiptap/TiptapVideo.d.ts +30 -0
  324. package/dist/tiptap/TiptapVideo.d.ts.map +1 -0
  325. package/dist/tiptap/TiptapVideo.js +66 -0
  326. package/dist/tiptap/TiptapVideo.js.map +1 -0
  327. package/dist/tiptap/useResolvedMediaSrc.d.ts +2 -0
  328. package/dist/tiptap/useResolvedMediaSrc.d.ts.map +1 -0
  329. package/dist/tiptap/useResolvedMediaSrc.js +42 -0
  330. package/dist/tiptap/useResolvedMediaSrc.js.map +1 -0
  331. package/dist/tiptapBridge.d.ts.map +1 -1
  332. package/dist/tiptapBridge.js +171 -14
  333. package/dist/tiptapBridge.js.map +1 -1
  334. package/dist/useHeadingLayout.d.ts +54 -0
  335. package/dist/useHeadingLayout.d.ts.map +1 -0
  336. package/dist/useHeadingLayout.js +260 -0
  337. package/dist/useHeadingLayout.js.map +1 -0
  338. package/dist/utils/collectInlineFontAwesomeCss.d.ts +21 -0
  339. package/dist/utils/collectInlineFontAwesomeCss.d.ts.map +1 -0
  340. package/dist/utils/collectInlineFontAwesomeCss.js +68 -0
  341. package/dist/utils/collectInlineFontAwesomeCss.js.map +1 -0
  342. package/dist/utils/dropUtils.d.ts +21 -2
  343. package/dist/utils/dropUtils.d.ts.map +1 -1
  344. package/dist/utils/dropUtils.js +43 -4
  345. package/dist/utils/dropUtils.js.map +1 -1
  346. package/dist/utils/normalizeMalformedAssetUrl.d.ts +15 -0
  347. package/dist/utils/normalizeMalformedAssetUrl.d.ts.map +1 -0
  348. package/dist/utils/normalizeMalformedAssetUrl.js +27 -0
  349. package/dist/utils/normalizeMalformedAssetUrl.js.map +1 -0
  350. package/package.json +8 -5
  351. package/src/DocumentSettingsDialog.tsx +266 -0
  352. package/src/EditorContext.tsx +534 -10
  353. package/src/EditorShell.tsx +571 -55
  354. package/src/EmojiPicker.tsx +332 -0
  355. package/src/ImageEditor.tsx +327 -0
  356. package/src/ImageNodeView.tsx +222 -21
  357. package/src/ImageViewer.tsx +221 -0
  358. package/src/InlineIcon.ts +84 -0
  359. package/src/InlinePreviewGutter.tsx +582 -0
  360. package/src/LinkDialog.tsx +276 -0
  361. package/src/MentionExtension.tsx +10 -7
  362. package/src/OutlinePanel.tsx +295 -0
  363. package/src/PlainHtmlPreview.tsx +211 -0
  364. package/src/PreviewControls.tsx +130 -24
  365. package/src/PreviewPanel.tsx +38 -21
  366. package/src/RawEditor.tsx +215 -4
  367. package/src/RecorderEntry.tsx +164 -0
  368. package/src/TemplateAnnotation.ts +32 -6
  369. package/src/TemplatePicker.tsx +818 -0
  370. package/src/ThemeCustomizerPanel.tsx +595 -0
  371. package/src/ThemePicker.tsx +319 -0
  372. package/src/Toolbar.tsx +708 -111
  373. package/src/VersionHistoryPanel.tsx +329 -0
  374. package/src/ViewMenuPanel.tsx +188 -0
  375. package/src/WysiwygEditor.tsx +229 -9
  376. package/src/__tests__/detectMarkdown.test.ts +0 -15
  377. package/src/__tests__/documentSettingsDialog.test.tsx +147 -0
  378. package/src/__tests__/emojiPicker.test.tsx +133 -0
  379. package/src/__tests__/fileKind.test.ts +16 -0
  380. package/src/__tests__/imageEditAffordance.test.tsx +268 -0
  381. package/src/__tests__/imageEditorShell.test.tsx +57 -0
  382. package/src/__tests__/imageEditorState.test.ts +171 -0
  383. package/src/__tests__/inlinePreviewGutter.test.tsx +62 -0
  384. package/src/__tests__/inlinePreviewGutterAllBlocks.test.tsx +103 -0
  385. package/src/__tests__/jsonEditor.test.tsx +168 -0
  386. package/src/__tests__/layersPanel.test.tsx +97 -0
  387. package/src/__tests__/linkDialogDocPicker.test.tsx +137 -0
  388. package/src/__tests__/outlinePanel.test.tsx +79 -0
  389. package/src/__tests__/plainHtmlPreview.test.tsx +107 -0
  390. package/src/__tests__/propertiesPanel.test.tsx +69 -0
  391. package/src/__tests__/recorderFormats.test.ts +146 -0
  392. package/src/__tests__/recorderTimingJson.test.ts +41 -0
  393. package/src/__tests__/templateAnnotationRoundTrip.test.ts +34 -0
  394. package/src/__tests__/tiptapBridge.test.ts +15 -0
  395. package/src/__tests__/useImageEditor.test.tsx +159 -0
  396. package/src/__tests__/useMediaRecorder.test.ts +186 -0
  397. package/src/__tests__/versionHistory.test.tsx +197 -0
  398. package/src/blockSlice.ts +75 -0
  399. package/src/buildPreviewDoc.ts +61 -6
  400. package/src/emojiData.ts +1337 -0
  401. package/src/fileKind.ts +30 -6
  402. package/src/hooks/useFileDrop.ts +40 -4
  403. package/src/imageEditor/CanvasSurface.tsx +402 -0
  404. package/src/imageEditor/ImageVersionHistoryDropdown.tsx +396 -0
  405. package/src/imageEditor/LayersPanel.tsx +143 -0
  406. package/src/imageEditor/PropertiesPanel.tsx +428 -0
  407. package/src/imageEditor/Toolbar.tsx +242 -0
  408. package/src/imageEditor/icons.tsx +144 -0
  409. package/src/imageEditor/image-editor.css +450 -0
  410. package/src/imageEditor/layers/EditorImageLayer.tsx +45 -0
  411. package/src/imageEditor/layers/EditorShapeLayer.tsx +62 -0
  412. package/src/imageEditor/layers/EditorTextLayer.tsx +45 -0
  413. package/src/imageEditor/layers/SelectionHandles.tsx +86 -0
  414. package/src/imageEditor/state.ts +153 -0
  415. package/src/imageEditor/useImageEditor.ts +328 -0
  416. package/src/imageEditor/useImageEditorTokens.ts +70 -0
  417. package/src/index.ts +82 -0
  418. package/src/jsonEditor/EmbeddedRichTextField.tsx +81 -0
  419. package/src/jsonEditor/JsonEditor.tsx +81 -0
  420. package/src/jsonEditor/JsonEditorContext.tsx +75 -0
  421. package/src/jsonEditor/RenderNode.tsx +66 -0
  422. package/src/jsonEditor/editors.tsx +678 -0
  423. package/src/jsonEditor/index.ts +2 -0
  424. package/src/jsonEditor/json-editor.css +463 -0
  425. package/src/jsonEditor/useJsonEditorTokens.ts +63 -0
  426. package/src/recorder/RecorderButton.tsx +72 -0
  427. package/src/recorder/RecorderModal.tsx +596 -0
  428. package/src/recorder/RecorderPanel.tsx +93 -0
  429. package/src/recorder/formats.ts +159 -0
  430. package/src/recorder/hooks/useMediaRecorder.ts +378 -0
  431. package/src/recorder/hooks/useStreamPreview.ts +47 -0
  432. package/src/recorder/sources/cameraStream.ts +32 -0
  433. package/src/recorder/sources/micStream.ts +25 -0
  434. package/src/recorder/sources/screenStream.ts +162 -0
  435. package/src/recorder/timingJson.ts +66 -0
  436. package/src/styles/editor.css +2490 -51
  437. package/src/styles/image-edit-affordance.css +201 -0
  438. package/src/styles/index.css +10 -0
  439. package/src/tiptap/TiptapAudio.tsx +86 -0
  440. package/src/tiptap/TiptapVideo.tsx +119 -0
  441. package/src/tiptap/useResolvedMediaSrc.ts +47 -0
  442. package/src/tiptapBridge.ts +188 -20
  443. package/src/useHeadingLayout.ts +294 -0
  444. package/src/utils/collectInlineFontAwesomeCss.ts +69 -0
  445. package/src/utils/dropUtils.ts +54 -6
  446. package/src/utils/normalizeMalformedAssetUrl.ts +22 -0
@@ -0,0 +1,582 @@
1
+ /**
2
+ * InlinePreviewGutter
3
+ *
4
+ * Renders one small SVG preview card per template-annotated block in the
5
+ * current document, positioned to vertically align with its corresponding
6
+ * heading in whichever editor view is active. A diagonal connector line
7
+ * + vertical block-extent bars bridge each heading and its preview card.
8
+ *
9
+ * The gutter consumes the parsed `Doc` from `useEditorContext()` and reuses
10
+ * the same template-resolution path as `LinearDocView` (heading text →
11
+ * template defaults → `getLayers()` → `BlockRenderer`).
12
+ *
13
+ * Auto-hides via container queries when the surrounding container is too
14
+ * narrow (see `.squisq-editor-with-gutter` rules in `styles/editor.css`).
15
+ *
16
+ * Implementation notes:
17
+ * - **View-agnostic positioning** — heading positions, page edges, and
18
+ * click-to-scroll all come from `useHeadingLayout()`, which has
19
+ * per-view backends (DOM-query for WYSIWYG, Monaco API for Markdown).
20
+ * The component only handles card stacking (which needs rendered card
21
+ * heights) and the visual rendering layer.
22
+ * - **Card stacking** — cards are absolutely positioned with `top` set
23
+ * from each heading's hook-supplied position. If two headings sit too
24
+ * close together, the lower card slides down so cards never overlap;
25
+ * the connector line then slopes from the (still heading-anchored)
26
+ * bar across to the (now lower) card.
27
+ */
28
+
29
+ import { useLayoutEffect, useMemo, useRef, useState, type CSSProperties } from 'react';
30
+ import type { Block, DocBlock, ViewportConfig } from '@bendyline/squisq/schemas';
31
+ import { VIEWPORT_PRESETS } from '@bendyline/squisq/schemas';
32
+ import {
33
+ flattenBlocks,
34
+ getLayers,
35
+ hasTemplate,
36
+ DEFAULT_THEME,
37
+ type RenderContext,
38
+ } from '@bendyline/squisq/doc';
39
+ import { extractPlainText, getChildren } from '@bendyline/squisq/markdown';
40
+ import { usePreviewSettingsOptional } from './PreviewControls';
41
+ import type {
42
+ MarkdownBlockNode,
43
+ MarkdownList,
44
+ MarkdownNode,
45
+ MarkdownTable,
46
+ } from '@bendyline/squisq/markdown';
47
+ import { BlockRenderer, MediaContext } from '@bendyline/squisq-react';
48
+ import type { MediaProvider } from '@bendyline/squisq/schemas';
49
+ import { useEditorContext } from './EditorContext';
50
+ import { templateLabel } from './TemplatePicker';
51
+ import { useHeadingLayout } from './useHeadingLayout';
52
+
53
+ // ── Helpers (mirrored from LinearDocView; kept local to avoid cross-package
54
+ // churn — extract to a shared module if a fourth copy appears) ────────────
55
+
56
+ function isAnnotated(block: Block): boolean {
57
+ const annotation = block.sourceHeading?.templateAnnotation;
58
+ if (!annotation) return false;
59
+ return !!annotation.template && hasTemplate(annotation.template);
60
+ }
61
+
62
+ function extractBodyPlainText(contents?: MarkdownBlockNode[]): string {
63
+ if (!contents || contents.length === 0) return '';
64
+ return contents
65
+ .map((n) => extractPlainText(n))
66
+ .join('\n')
67
+ .trim();
68
+ }
69
+
70
+ function extractListItems(contents?: MarkdownBlockNode[]): string[] {
71
+ if (!contents) return [];
72
+ const items: string[] = [];
73
+ for (const node of contents) {
74
+ if (node.type === 'list') {
75
+ for (const item of (node as MarkdownList).children) {
76
+ const text = extractPlainText(item).trim();
77
+ if (text) items.push(text);
78
+ }
79
+ }
80
+ }
81
+ return items;
82
+ }
83
+
84
+ interface PreviewImage {
85
+ src: string;
86
+ alt: string;
87
+ width?: number;
88
+ height?: number;
89
+ }
90
+
91
+ function parseDim(raw: string | undefined): number | undefined {
92
+ if (raw === undefined) return undefined;
93
+ const n = parseFloat(raw);
94
+ return Number.isFinite(n) && n > 0 ? n : undefined;
95
+ }
96
+
97
+ function extractBlockImages(contents?: MarkdownBlockNode[]): PreviewImage[] {
98
+ if (!contents || contents.length === 0) return [];
99
+ const images: PreviewImage[] = [];
100
+
101
+ // Walk an HTML sub-DOM looking for <img src="..."> elements that the
102
+ // markdown parser produced for HTML-block / inline-HTML image tags.
103
+ // (When WYSIWYG persists an explicit width/height the image round-trips
104
+ // as <img …> rather than the ![](src) shorthand.)
105
+ function walkHtml(node: {
106
+ type: string;
107
+ tagName?: string;
108
+ attributes?: Record<string, string>;
109
+ children?: unknown[];
110
+ }): void {
111
+ if (node.type === 'htmlElement' && node.tagName === 'img') {
112
+ const src = node.attributes?.src;
113
+ if (src) {
114
+ images.push({
115
+ src,
116
+ alt: node.attributes?.alt ?? '',
117
+ width: parseDim(node.attributes?.width),
118
+ height: parseDim(node.attributes?.height),
119
+ });
120
+ }
121
+ }
122
+ if (Array.isArray(node.children)) {
123
+ for (const child of node.children) {
124
+ walkHtml(child as { type: string });
125
+ }
126
+ }
127
+ }
128
+
129
+ function walk(node: MarkdownNode): void {
130
+ if ('type' in node && node.type === 'image' && 'url' in node) {
131
+ const img = node as { url: string; alt?: string };
132
+ if (img.url) images.push({ src: img.url, alt: img.alt ?? '' });
133
+ }
134
+ if ('type' in node && node.type === 'htmlBlock') {
135
+ const block = node as unknown as { htmlChildren?: unknown[] };
136
+ for (const child of block.htmlChildren ?? []) {
137
+ walkHtml(child as { type: string });
138
+ }
139
+ }
140
+ for (const child of getChildren(node)) walk(child);
141
+ }
142
+ for (const node of contents) walk(node);
143
+ return images;
144
+ }
145
+
146
+ /** Like extractBlockImages, but also descends into the block's child blocks
147
+ * so an annotated parent block can pick up media defined in a sub-section. */
148
+ function collectImagesDeep(block: Block): PreviewImage[] {
149
+ const images: PreviewImage[] = [];
150
+ const seen = new Set<string>();
151
+ function visit(b: Block): void {
152
+ for (const img of extractBlockImages(b.contents)) {
153
+ if (seen.has(img.src)) continue;
154
+ seen.add(img.src);
155
+ images.push(img);
156
+ }
157
+ for (const child of b.children ?? []) visit(child);
158
+ }
159
+ visit(block);
160
+ return images;
161
+ }
162
+
163
+ function extractTableData(contents?: MarkdownBlockNode[]): {
164
+ headers: string[];
165
+ rows: string[][];
166
+ align?: (('left' | 'right' | 'center') | null)[];
167
+ } | null {
168
+ if (!contents) return null;
169
+ for (const node of contents) {
170
+ if (node.type === 'table') {
171
+ const table = node as MarkdownTable;
172
+ const [headerRow, ...bodyRows] = table.children;
173
+ if (!headerRow) return null;
174
+ const headers = headerRow.children.map((cell) => extractPlainText(cell).trim());
175
+ const rows = bodyRows.map((row) => row.children.map((cell) => extractPlainText(cell).trim()));
176
+ return { headers, rows, align: table.align };
177
+ }
178
+ }
179
+ return null;
180
+ }
181
+
182
+ function getTemplateDefaults(
183
+ templateName: string,
184
+ headingText: string,
185
+ bodyText: string,
186
+ contents?: MarkdownBlockNode[],
187
+ images: PreviewImage[] = [],
188
+ ): Record<string, unknown> {
189
+ switch (templateName) {
190
+ case 'statHighlight':
191
+ return { stat: headingText, description: bodyText || headingText };
192
+ case 'quote':
193
+ case 'fullBleedQuote':
194
+ case 'pullQuote':
195
+ return { quote: bodyText || headingText };
196
+ case 'factCard':
197
+ return { fact: headingText, explanation: bodyText || headingText };
198
+ case 'comparisonBar':
199
+ return { leftLabel: 'A', leftValue: 60, rightLabel: 'B', rightValue: 40 };
200
+ case 'list': {
201
+ const items = extractListItems(contents);
202
+ return { items: items.length > 0 ? items : ['Item 1', 'Item 2', 'Item 3'] };
203
+ }
204
+ case 'definitionCard':
205
+ return { term: headingText, definition: bodyText || headingText };
206
+ case 'dateEvent':
207
+ return { date: headingText, description: bodyText || headingText };
208
+ case 'dataTable': {
209
+ const tableData = extractTableData(contents);
210
+ return tableData ?? { headers: ['Column'], rows: [['Data']] };
211
+ }
212
+ case 'imageWithCaption': {
213
+ const first = images[0];
214
+ return {
215
+ imageSrc: first?.src ?? '',
216
+ imageAlt: first?.alt ?? headingText,
217
+ caption: bodyText || headingText,
218
+ captionPosition: 'bottom',
219
+ };
220
+ }
221
+ case 'leftFeature':
222
+ case 'rightFeature': {
223
+ // Mirror the LinearDocView / buildPreviewDoc defaults so the
224
+ // mini gutter card shows the same content as the main preview.
225
+ const first = images[0];
226
+ return {
227
+ imageSrc: first?.src ?? '',
228
+ imageAlt: first?.alt ?? headingText,
229
+ imageWidth: first?.width,
230
+ imageHeight: first?.height,
231
+ title: headingText,
232
+ body: bodyText,
233
+ };
234
+ }
235
+ case 'videoWithCaption': {
236
+ const first = images[0];
237
+ return {
238
+ videoSrc: first?.src ?? '',
239
+ caption: bodyText || headingText,
240
+ captionPosition: 'bottom',
241
+ };
242
+ }
243
+ default:
244
+ return {};
245
+ }
246
+ }
247
+
248
+ // ── Types ──────────────────────────────────────────────────────────
249
+
250
+ export interface InlinePreviewGutterProps {
251
+ /** Width of the gutter in pixels (default: 320). */
252
+ width?: number;
253
+ /** Base path for resolving media URLs in card thumbnails. */
254
+ basePath?: string;
255
+ /** Viewport used to render each card (default: landscape preset). */
256
+ viewport?: ViewportConfig;
257
+ /** Optional CSS class for the outer container. */
258
+ className?: string;
259
+ /**
260
+ * Width in pixels of the connector strip that bridges the editor and
261
+ * the gutter. Defaults to 24.
262
+ */
263
+ connectorWidth?: number;
264
+ /**
265
+ * Optional MediaProvider used to resolve relative image src values inside
266
+ * preview cards. When omitted, images with relative paths fall back to
267
+ * `basePath`-prefixed URLs (which usually 404 in container-backed editors).
268
+ */
269
+ mediaProvider?: MediaProvider | null;
270
+ }
271
+
272
+ interface PreviewItem {
273
+ id: string;
274
+ template: string;
275
+ headingText: string;
276
+ block: Block;
277
+ }
278
+
279
+ // ── Connector layout constants ─────────────────────────────────────
280
+
281
+ /** Distance from the strip's right edge to the bar's right edge (px). */
282
+ const BAR_RIGHT_OFFSET = 14;
283
+ /** Vertical offset within a card to the center of the caption row. Card
284
+ * padding (8) + half line-height of the 12px label (~8) ≈ 16. */
285
+ const CARD_LABEL_OFFSET = 16;
286
+ /** Vertical offset from heading top to the left connector dot. Matches
287
+ * CARD_LABEL_OFFSET so the line is horizontal when the card sits at its
288
+ * natural (heading-aligned) position. */
289
+ const EXTENT_TOP_PAD = CARD_LABEL_OFFSET;
290
+ /** Card's `left: 12px` inside the gutter — used to land the right circle on
291
+ * the card's left edge for the diagonal connector line. */
292
+ const CARD_LEFT_INSET = 12;
293
+
294
+ // ── Component ──────────────────────────────────────────────────────
295
+
296
+ export function InlinePreviewGutter({
297
+ width = 320,
298
+ basePath = '/',
299
+ viewport = VIEWPORT_PRESETS.landscape,
300
+ className,
301
+ connectorWidth = 24,
302
+ mediaProvider = null,
303
+ }: InlinePreviewGutterProps) {
304
+ const { doc } = useEditorContext();
305
+ const gutterRef = useRef<HTMLElement | null>(null);
306
+ const { entries: headingEntries, scrollToBlock } = useHeadingLayout(gutterRef);
307
+ // Follow the document's active theme so the mini cards reflect the
308
+ // same palette as the main preview surface. PreviewSettingsProvider
309
+ // resolves this from frontmatter (`squisq-theme` / legacy `theme`)
310
+ // and theme overrides, so we never duplicate that logic here. Falls
311
+ // back to DEFAULT_THEME when the gutter is mounted outside the
312
+ // provider (defensive — EditorShell always wraps with it).
313
+ const previewSettings = usePreviewSettingsOptional();
314
+ const activeTheme = previewSettings?.activeTheme ?? DEFAULT_THEME;
315
+
316
+ // Build the renderable PreviewItem list (just for annotated blocks).
317
+ const items = useMemo<PreviewItem[]>(() => {
318
+ if (!doc || !doc.blocks.length) return [];
319
+ const flat = flattenBlocks(doc.blocks);
320
+ const totalBlocks = flat.length;
321
+ const result: PreviewItem[] = [];
322
+
323
+ flat.forEach((block, index) => {
324
+ if (!isAnnotated(block)) return;
325
+ const annotation = block.sourceHeading!.templateAnnotation!;
326
+ const template = annotation.template!;
327
+ const headingText = block.sourceHeading ? extractPlainText(block.sourceHeading) : '';
328
+ const bodyText = extractBodyPlainText(block.contents);
329
+
330
+ const templateBlock: Record<string, unknown> = {
331
+ id: block.id,
332
+ template,
333
+ startTime: 0,
334
+ duration: 1,
335
+ audioSegment: 0,
336
+ title: headingText,
337
+ ...getTemplateDefaults(
338
+ template,
339
+ headingText,
340
+ bodyText,
341
+ block.contents,
342
+ collectImagesDeep(block),
343
+ ),
344
+ ...annotation.params,
345
+ ...block.templateOverrides,
346
+ };
347
+
348
+ const ctx: RenderContext = {
349
+ blockIndex: index,
350
+ totalBlocks,
351
+ theme: activeTheme,
352
+ viewport,
353
+ };
354
+
355
+ try {
356
+ const layers = getLayers(templateBlock as unknown as DocBlock, ctx);
357
+ const visualBlock = {
358
+ ...block,
359
+ layers,
360
+ template,
361
+ } as Block;
362
+ result.push({
363
+ id: block.id,
364
+ template,
365
+ headingText,
366
+ block: visualBlock,
367
+ });
368
+ } catch (err: unknown) {
369
+ const message = err instanceof Error ? err.message : String(err);
370
+ console.warn(`[InlinePreviewGutter] Skipped block "${block.id}" (${template}): ${message}`);
371
+ }
372
+ });
373
+
374
+ return result;
375
+ }, [doc, viewport, activeTheme]);
376
+
377
+ // Heading top per item id — derived from the layout hook. The connector
378
+ // dot tracks this even when stacking pushes the card below.
379
+ const connectorTops = useMemo(() => {
380
+ const m = new Map<string, number>();
381
+ headingEntries.forEach((e) => m.set(e.block.id, e.top));
382
+ return m;
383
+ }, [headingEntries]);
384
+
385
+ // Card stacking state — populated by a small post-render measurement
386
+ // pass that reads each card's height to push subsequent cards down.
387
+ const [positions, setPositions] = useState<Map<string, number>>(new Map());
388
+
389
+ useLayoutEffect(() => {
390
+ if (items.length === 0) {
391
+ setPositions((prev) => (prev.size === 0 ? prev : new Map()));
392
+ return;
393
+ }
394
+ const gutter = gutterRef.current;
395
+ if (!gutter) return;
396
+ const cardEls = gutter.querySelectorAll<HTMLElement>('.squisq-inline-preview-card');
397
+ const STACK_GAP = 12;
398
+ const FALLBACK_CARD_HEIGHT = 220;
399
+ let lastBottom = -Infinity;
400
+ const next = new Map<string, number>();
401
+ items.forEach((item, i) => {
402
+ const headingTop = connectorTops.get(item.id);
403
+ if (headingTop == null) return;
404
+ const top = Math.max(headingTop, lastBottom + STACK_GAP);
405
+ next.set(item.id, top);
406
+ const cardEl = cardEls[i];
407
+ const cardHeight = cardEl ? cardEl.getBoundingClientRect().height : FALLBACK_CARD_HEIGHT;
408
+ lastBottom = top + cardHeight;
409
+ });
410
+ setPositions((prev) => {
411
+ if (prev.size === next.size) {
412
+ let same = true;
413
+ for (const [k, v] of next) {
414
+ const p = prev.get(k);
415
+ if (p == null || Math.abs(p - v) > 0.5) {
416
+ same = false;
417
+ break;
418
+ }
419
+ }
420
+ if (same) return prev;
421
+ }
422
+ return next;
423
+ });
424
+ }, [items, connectorTops]);
425
+
426
+ const isEmpty = items.length === 0;
427
+ const totalWidth = width + connectorWidth;
428
+ const gutterStyle: CSSProperties = {
429
+ position: 'relative',
430
+ width: `${totalWidth}px`,
431
+ flex: `0 0 ${totalWidth}px`,
432
+ overflow: 'hidden',
433
+ };
434
+
435
+ return (
436
+ <aside
437
+ ref={gutterRef}
438
+ className={`squisq-inline-preview-gutter ${className ?? ''}`}
439
+ style={gutterStyle}
440
+ data-testid="inline-preview-gutter"
441
+ aria-label="Block previews"
442
+ >
443
+ {/* Connector strip — vertical bracket bars per heading, on the
444
+ left edge of the gutter so they sit immediately next to the
445
+ editor body. */}
446
+ {headingEntries.length > 0 && (
447
+ <div
448
+ className="squisq-inline-preview-connectors"
449
+ aria-hidden="true"
450
+ style={{
451
+ position: 'absolute',
452
+ top: 0,
453
+ left: 0,
454
+ width: connectorWidth,
455
+ height: '100%',
456
+ overflow: 'hidden',
457
+ pointerEvents: 'none',
458
+ }}
459
+ >
460
+ {headingEntries.map((ex, i) => {
461
+ const EXTENT_GAP = 6;
462
+ const height = Math.max(2, ex.bottom - ex.top - EXTENT_GAP);
463
+ return (
464
+ <div
465
+ key={`h-${i}-${ex.block.id}`}
466
+ className={`squisq-inline-preview-extent${
467
+ ex.annotated ? '' : ' squisq-inline-preview-extent--untagged'
468
+ }`}
469
+ style={{
470
+ position: 'absolute',
471
+ top: `${ex.top}px`,
472
+ height: `${height}px`,
473
+ right: `${BAR_RIGHT_OFFSET}px`,
474
+ }}
475
+ />
476
+ );
477
+ })}
478
+ </div>
479
+ )}
480
+ {/* Diagonal connector SVG — overlays the connector strip + the
481
+ start of the card area, drawing a line from each bar to its
482
+ card's caption row. */}
483
+ {items.length > 0 && (
484
+ <svg
485
+ className="squisq-inline-preview-connector-svg"
486
+ aria-hidden="true"
487
+ style={{
488
+ position: 'absolute',
489
+ top: 0,
490
+ left: 0,
491
+ width: connectorWidth + CARD_LEFT_INSET,
492
+ height: '100%',
493
+ pointerEvents: 'none',
494
+ overflow: 'visible',
495
+ }}
496
+ >
497
+ {items.map((item) => {
498
+ const headingTop = connectorTops.get(item.id);
499
+ const cardTop = positions.get(item.id);
500
+ if (headingTop == null || cardTop == null) return null;
501
+ const x1 = connectorWidth - BAR_RIGHT_OFFSET - 2;
502
+ const y1 = headingTop + EXTENT_TOP_PAD;
503
+ const x2 = connectorWidth + CARD_LEFT_INSET;
504
+ const y2 = cardTop + CARD_LABEL_OFFSET;
505
+ return (
506
+ <g key={item.id}>
507
+ <line
508
+ x1={x1}
509
+ y1={y1}
510
+ x2={x2}
511
+ y2={y2}
512
+ stroke="#6366f1"
513
+ strokeWidth="2"
514
+ strokeLinecap="round"
515
+ />
516
+ <circle cx={x1} cy={y1} r="4" fill="#6366f1" stroke="#ffffff" strokeWidth="2" />
517
+ <circle cx={x2} cy={y2} r="4" fill="#6366f1" stroke="#ffffff" strokeWidth="2" />
518
+ </g>
519
+ );
520
+ })}
521
+ </svg>
522
+ )}
523
+
524
+ {/* Card area — empty placeholder OR positioned cards. */}
525
+ {isEmpty ? (
526
+ <div
527
+ className="squisq-inline-preview-empty"
528
+ style={{ position: 'absolute', top: 12, left: connectorWidth + 12, right: 12 }}
529
+ >
530
+ <p>Tag a heading with a template to see a preview here.</p>
531
+ </div>
532
+ ) : (
533
+ <MediaContext.Provider value={mediaProvider ?? null}>
534
+ {items.map((item) => {
535
+ const top = positions.get(item.id);
536
+ const hidden = top == null;
537
+ return (
538
+ <div
539
+ key={item.id}
540
+ className="squisq-inline-preview-card"
541
+ data-template={item.template}
542
+ style={{
543
+ position: 'absolute',
544
+ top: `${top ?? 0}px`,
545
+ left: connectorWidth + CARD_LEFT_INSET,
546
+ right: 12,
547
+ visibility: hidden ? 'hidden' : 'visible',
548
+ }}
549
+ onClick={() => scrollToBlock(item.block)}
550
+ >
551
+ <div className="squisq-inline-preview-card-label">
552
+ <span className="squisq-inline-preview-card-template">
553
+ {templateLabel(item.template)}
554
+ </span>
555
+ {item.headingText && (
556
+ <>
557
+ <span className="squisq-inline-preview-card-sep">—</span>
558
+ <span className="squisq-inline-preview-card-title">{item.headingText}</span>
559
+ </>
560
+ )}
561
+ </div>
562
+ <div
563
+ className="squisq-inline-preview-card-svg"
564
+ style={{
565
+ aspectRatio: `${viewport.width} / ${viewport.height}`,
566
+ }}
567
+ >
568
+ <BlockRenderer
569
+ block={item.block}
570
+ blockTime={0}
571
+ basePath={basePath}
572
+ viewport={viewport}
573
+ />
574
+ </div>
575
+ </div>
576
+ );
577
+ })}
578
+ </MediaContext.Provider>
579
+ )}
580
+ </aside>
581
+ );
582
+ }