@eightyfourthousand/lib-editing 2026.3.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 (304) hide show
  1. package/.babelrc +12 -0
  2. package/.eslintrc.json +18 -0
  3. package/README.md +7 -0
  4. package/jest.config.ts +10 -0
  5. package/package.json +35 -0
  6. package/postcss.config.mjs +9 -0
  7. package/project.json +29 -0
  8. package/src/fixtures/basic/json.ts +396 -0
  9. package/src/fixtures/toh251/json.ts +4913 -0
  10. package/src/fixtures/toh251/passages.ts +2814 -0
  11. package/src/fixtures/types.ts +8 -0
  12. package/src/index.ts +4 -0
  13. package/src/lib/block.ts +226 -0
  14. package/src/lib/components/editor/BlockEditor.tsx +38 -0
  15. package/src/lib/components/editor/EditorBackMatterPage.tsx +95 -0
  16. package/src/lib/components/editor/EditorBodyPage.tsx +87 -0
  17. package/src/lib/components/editor/EditorHeader.tsx +22 -0
  18. package/src/lib/components/editor/EditorLayout.tsx +62 -0
  19. package/src/lib/components/editor/EditorLeftPanelPage.tsx +27 -0
  20. package/src/lib/components/editor/EditorProvider.tsx +399 -0
  21. package/src/lib/components/editor/PaginationProvider.tsx +472 -0
  22. package/src/lib/components/editor/TitlesBuilder.tsx +40 -0
  23. package/src/lib/components/editor/TranslationBuilder.tsx +171 -0
  24. package/src/lib/components/editor/TranslationEditor.tsx +32 -0
  25. package/src/lib/components/editor/extensions/Abbreviation/Abbreviation.ts +133 -0
  26. package/src/lib/components/editor/extensions/Audio/Audio.ts +69 -0
  27. package/src/lib/components/editor/extensions/Bold.ts +43 -0
  28. package/src/lib/components/editor/extensions/Document.ts +8 -0
  29. package/src/lib/components/editor/extensions/DragHandle/DragHandle.ts +429 -0
  30. package/src/lib/components/editor/extensions/EndNoteLink/EndNoteLink.tsx +39 -0
  31. package/src/lib/components/editor/extensions/EndNoteLink/EndNoteLinkHoverContent.tsx +139 -0
  32. package/src/lib/components/editor/extensions/EndNoteLink/EndNoteLinkMark.ts +236 -0
  33. package/src/lib/components/editor/extensions/EndNoteLink/endnote-utils.ts +412 -0
  34. package/src/lib/components/editor/extensions/GlobalConfig.ts +52 -0
  35. package/src/lib/components/editor/extensions/GlossaryInstance/GlossaryInput.tsx +54 -0
  36. package/src/lib/components/editor/extensions/GlossaryInstance/GlossaryInstance.tsx +129 -0
  37. package/src/lib/components/editor/extensions/GlossaryInstance/GlossaryInstanceNode.ts +148 -0
  38. package/src/lib/components/editor/extensions/Heading/Heading.ts +71 -0
  39. package/src/lib/components/editor/extensions/HoverInputField.tsx +54 -0
  40. package/src/lib/components/editor/extensions/Image.ts +18 -0
  41. package/src/lib/components/editor/extensions/Indent.ts +103 -0
  42. package/src/lib/components/editor/extensions/InternalLink/InternalLink.ts +173 -0
  43. package/src/lib/components/editor/extensions/InternalLink/InternalLinkHoverContent.tsx +137 -0
  44. package/src/lib/components/editor/extensions/InternalLink/InternalLinkInput.tsx +71 -0
  45. package/src/lib/components/editor/extensions/InternalLink/index.ts +1 -0
  46. package/src/lib/components/editor/extensions/Italic.ts +50 -0
  47. package/src/lib/components/editor/extensions/LeadingSpace.ts +106 -0
  48. package/src/lib/components/editor/extensions/Line/LineNode.ts +41 -0
  49. package/src/lib/components/editor/extensions/LineGroup/LineGroupNode.ts +124 -0
  50. package/src/lib/components/editor/extensions/Link/Link.ts +65 -0
  51. package/src/lib/components/editor/extensions/Link/LinkHoverContent.tsx +124 -0
  52. package/src/lib/components/editor/extensions/Link/index.ts +1 -0
  53. package/src/lib/components/editor/extensions/List.ts +74 -0
  54. package/src/lib/components/editor/extensions/Mantra/Mantra.ts +88 -0
  55. package/src/lib/components/editor/extensions/Mention/Mention.ts +184 -0
  56. package/src/lib/components/editor/extensions/Mention/MentionHoverContent.tsx +158 -0
  57. package/src/lib/components/editor/extensions/NodeWrapper.tsx +57 -0
  58. package/src/lib/components/editor/extensions/Paragraph/Paragraph.ts +25 -0
  59. package/src/lib/components/editor/extensions/ParagraphIndent.ts +87 -0
  60. package/src/lib/components/editor/extensions/Passage/EditLabel.tsx +57 -0
  61. package/src/lib/components/editor/extensions/Passage/EditorOptions.tsx +29 -0
  62. package/src/lib/components/editor/extensions/Passage/Passage.tsx +238 -0
  63. package/src/lib/components/editor/extensions/Passage/PassageNode.ts +223 -0
  64. package/src/lib/components/editor/extensions/Passage/ReaderOptions.tsx +55 -0
  65. package/src/lib/components/editor/extensions/Passage/ShowAnnotations.tsx +92 -0
  66. package/src/lib/components/editor/extensions/Passage/index.ts +1 -0
  67. package/src/lib/components/editor/extensions/Passage/label.spec.ts +118 -0
  68. package/src/lib/components/editor/extensions/Passage/label.ts +39 -0
  69. package/src/lib/components/editor/extensions/Placeholder.ts +9 -0
  70. package/src/lib/components/editor/extensions/SlashCommand/SlashCommand.ts +65 -0
  71. package/src/lib/components/editor/extensions/SlashCommand/SuggestionList.tsx +109 -0
  72. package/src/lib/components/editor/extensions/SlashCommand/Suggestions.ts +185 -0
  73. package/src/lib/components/editor/extensions/SmallCaps.ts +110 -0
  74. package/src/lib/components/editor/extensions/StarterKit.ts +36 -0
  75. package/src/lib/components/editor/extensions/Subscript.ts +43 -0
  76. package/src/lib/components/editor/extensions/Superscript.ts +43 -0
  77. package/src/lib/components/editor/extensions/Table.ts +32 -0
  78. package/src/lib/components/editor/extensions/TextAlign.ts +5 -0
  79. package/src/lib/components/editor/extensions/TitleMetadata.ts +40 -0
  80. package/src/lib/components/editor/extensions/TitleNode.ts +133 -0
  81. package/src/lib/components/editor/extensions/TitlesNode.ts +102 -0
  82. package/src/lib/components/editor/extensions/Trailer.ts +57 -0
  83. package/src/lib/components/editor/extensions/TranslationDocument.ts +7 -0
  84. package/src/lib/components/editor/extensions/TranslationMetadata.ts +58 -0
  85. package/src/lib/components/editor/extensions/Underline.ts +43 -0
  86. package/src/lib/components/editor/hooks/index.ts +4 -0
  87. package/src/lib/components/editor/hooks/useBlockEditor.ts +53 -0
  88. package/src/lib/components/editor/hooks/useDefaultExtensions.ts +39 -0
  89. package/src/lib/components/editor/hooks/useDirtyStore.ts +33 -0
  90. package/src/lib/components/editor/hooks/useTranslationExtensions.ts +148 -0
  91. package/src/lib/components/editor/index.ts +10 -0
  92. package/src/lib/components/editor/menus/EmptyBubbleMenu.tsx +42 -0
  93. package/src/lib/components/editor/menus/MainBubbleMenu.tsx +51 -0
  94. package/src/lib/components/editor/menus/TranslationBubbleMenu.tsx +57 -0
  95. package/src/lib/components/editor/menus/index.ts +3 -0
  96. package/src/lib/components/editor/menus/selectors/EndNoteSelector.tsx +388 -0
  97. package/src/lib/components/editor/menus/selectors/GlossarySelector.tsx +63 -0
  98. package/src/lib/components/editor/menus/selectors/LinkSelector.tsx +68 -0
  99. package/src/lib/components/editor/menus/selectors/MantraSelector.tsx +119 -0
  100. package/src/lib/components/editor/menus/selectors/NodeSelector.tsx +144 -0
  101. package/src/lib/components/editor/menus/selectors/ParagraphButtons.tsx +68 -0
  102. package/src/lib/components/editor/menus/selectors/SelectorInputField.tsx +68 -0
  103. package/src/lib/components/editor/menus/selectors/TextAlignSelector.tsx +89 -0
  104. package/src/lib/components/editor/menus/selectors/TextButtons.tsx +89 -0
  105. package/src/lib/components/editor/menus/selectors/TranslationNodeSelector.tsx +143 -0
  106. package/src/lib/components/editor/menus/selectors/TranslationTextButtons.tsx +125 -0
  107. package/src/lib/components/editor/menus/selectors/index.ts +5 -0
  108. package/src/lib/components/editor/save-filter.spec.ts +94 -0
  109. package/src/lib/components/editor/save-filter.ts +27 -0
  110. package/src/lib/components/editor/util.ts +304 -0
  111. package/src/lib/components/index.ts +3 -0
  112. package/src/lib/components/reader/ReaderBackMatterPage.tsx +62 -0
  113. package/src/lib/components/reader/ReaderBackMatterPanel.tsx +53 -0
  114. package/src/lib/components/reader/ReaderBodyPage.tsx +46 -0
  115. package/src/lib/components/reader/ReaderBodyPanel.tsx +68 -0
  116. package/src/lib/components/reader/ReaderLayout.tsx +39 -0
  117. package/src/lib/components/reader/ReaderLeftPanel.tsx +8 -0
  118. package/src/lib/components/reader/ReaderLeftPanelPage.tsx +31 -0
  119. package/src/lib/components/reader/TranslationReader.tsx +28 -0
  120. package/src/lib/components/reader/index.ts +2 -0
  121. package/src/lib/components/reader/ssr.ts +3 -0
  122. package/src/lib/components/shared/AiSummarizerPage.tsx +12 -0
  123. package/src/lib/components/shared/BackMatterPanel.tsx +143 -0
  124. package/src/lib/components/shared/BodyPanel.tsx +214 -0
  125. package/src/lib/components/shared/HoverCardProvider.tsx +407 -0
  126. package/src/lib/components/shared/Imprint.tsx +24 -0
  127. package/src/lib/components/shared/LabeledElement.tsx +133 -0
  128. package/src/lib/components/shared/LeftPanel.tsx +65 -0
  129. package/src/lib/components/shared/NavigationContext.ts +64 -0
  130. package/src/lib/components/shared/NavigationProvider.tsx +368 -0
  131. package/src/lib/components/shared/OpenGraphImage.tsx +75 -0
  132. package/src/lib/components/shared/PassageSkeleton.tsx +10 -0
  133. package/src/lib/components/shared/RestrictionWarning.tsx +177 -0
  134. package/src/lib/components/shared/SourceReader.tsx +83 -0
  135. package/src/lib/components/shared/SuggestRevisionForm.tsx +99 -0
  136. package/src/lib/components/shared/TableOfContents.tsx +280 -0
  137. package/src/lib/components/shared/ThreeColumnRenderer.tsx +54 -0
  138. package/src/lib/components/shared/TranslationHeader.tsx +86 -0
  139. package/src/lib/components/shared/TranslationHoverCard.tsx +84 -0
  140. package/src/lib/components/shared/TranslationSkeleton.tsx +16 -0
  141. package/src/lib/components/shared/TranslationTable.tsx +155 -0
  142. package/src/lib/components/shared/bibliography/BibliographyBody.tsx +28 -0
  143. package/src/lib/components/shared/bibliography/BibliographyList.tsx +63 -0
  144. package/src/lib/components/shared/bibliography/index.ts +1 -0
  145. package/src/lib/components/shared/generate-metadata.ts +44 -0
  146. package/src/lib/components/shared/glossary/GlossaryInstanceBody.tsx +144 -0
  147. package/src/lib/components/shared/glossary/GlossaryPaginationProvider.tsx +317 -0
  148. package/src/lib/components/shared/glossary/GlossarySkeleton.tsx +19 -0
  149. package/src/lib/components/shared/glossary/GlossaryTermList.tsx +58 -0
  150. package/src/lib/components/shared/glossary/index.ts +3 -0
  151. package/src/lib/components/shared/hooks/useGlossaryInstanceListener.tsx +42 -0
  152. package/src/lib/components/shared/hooks/useScrollInTab.tsx +43 -0
  153. package/src/lib/components/shared/hooks/useScrollPositionRestore.ts +274 -0
  154. package/src/lib/components/shared/hooks/useTohToggle.tsx +52 -0
  155. package/src/lib/components/shared/index.ts +11 -0
  156. package/src/lib/components/shared/ssr.ts +2 -0
  157. package/src/lib/components/shared/titles/FramedCard.tsx +132 -0
  158. package/src/lib/components/shared/titles/LongTitle.tsx +20 -0
  159. package/src/lib/components/shared/titles/LongTitles.tsx +28 -0
  160. package/src/lib/components/shared/titles/Title.tsx +54 -0
  161. package/src/lib/components/shared/titles/TitleDetails.tsx +47 -0
  162. package/src/lib/components/shared/titles/TitleForm.tsx +37 -0
  163. package/src/lib/components/shared/titles/Titles.tsx +114 -0
  164. package/src/lib/components/shared/titles/TitlesCard.tsx +113 -0
  165. package/src/lib/components/shared/titles/index.ts +8 -0
  166. package/src/lib/components/shared/types.ts +79 -0
  167. package/src/lib/components/ssr.ts +2 -0
  168. package/src/lib/exporters/abbreviation.spec.ts +31 -0
  169. package/src/lib/exporters/abbreviation.ts +22 -0
  170. package/src/lib/exporters/annotation.ts +193 -0
  171. package/src/lib/exporters/audio.spec.ts +77 -0
  172. package/src/lib/exporters/audio.ts +27 -0
  173. package/src/lib/exporters/blockquote.spec.ts +48 -0
  174. package/src/lib/exporters/blockquote.ts +24 -0
  175. package/src/lib/exporters/code.spec.ts +93 -0
  176. package/src/lib/exporters/code.ts +26 -0
  177. package/src/lib/exporters/end-note-link.spec.ts +104 -0
  178. package/src/lib/exporters/end-note-link.ts +35 -0
  179. package/src/lib/exporters/export.ts +12 -0
  180. package/src/lib/exporters/glossary-instance.spec.ts +85 -0
  181. package/src/lib/exporters/glossary-instance.ts +31 -0
  182. package/src/lib/exporters/has-abbreviation.spec.ts +31 -0
  183. package/src/lib/exporters/has-abbreviation.ts +21 -0
  184. package/src/lib/exporters/heading.spec.ts +80 -0
  185. package/src/lib/exporters/heading.ts +28 -0
  186. package/src/lib/exporters/image.spec.ts +48 -0
  187. package/src/lib/exporters/image.ts +25 -0
  188. package/src/lib/exporters/indent.spec.ts +58 -0
  189. package/src/lib/exporters/indent.ts +18 -0
  190. package/src/lib/exporters/index.ts +1 -0
  191. package/src/lib/exporters/internal-link.spec.ts +90 -0
  192. package/src/lib/exporters/internal-link.ts +35 -0
  193. package/src/lib/exporters/italic.spec.ts +84 -0
  194. package/src/lib/exporters/italic.ts +55 -0
  195. package/src/lib/exporters/leading-space.spec.ts +28 -0
  196. package/src/lib/exporters/leading-space.ts +16 -0
  197. package/src/lib/exporters/line-group.spec.ts +48 -0
  198. package/src/lib/exporters/line-group.ts +24 -0
  199. package/src/lib/exporters/line.spec.ts +48 -0
  200. package/src/lib/exporters/line.ts +24 -0
  201. package/src/lib/exporters/link.spec.ts +123 -0
  202. package/src/lib/exporters/link.ts +67 -0
  203. package/src/lib/exporters/list-item.spec.ts +48 -0
  204. package/src/lib/exporters/list-item.ts +24 -0
  205. package/src/lib/exporters/list.spec.ts +82 -0
  206. package/src/lib/exporters/list.ts +31 -0
  207. package/src/lib/exporters/mantra.spec.ts +51 -0
  208. package/src/lib/exporters/mantra.ts +25 -0
  209. package/src/lib/exporters/mention.ts +41 -0
  210. package/src/lib/exporters/paragraph.spec.ts +173 -0
  211. package/src/lib/exporters/paragraph.ts +32 -0
  212. package/src/lib/exporters/quote.spec.ts +56 -0
  213. package/src/lib/exporters/quote.ts +25 -0
  214. package/src/lib/exporters/span.spec.ts +118 -0
  215. package/src/lib/exporters/span.ts +44 -0
  216. package/src/lib/exporters/table-body-data.spec.ts +48 -0
  217. package/src/lib/exporters/table-body-data.ts +24 -0
  218. package/src/lib/exporters/table-body-header.spec.ts +48 -0
  219. package/src/lib/exporters/table-body-header.ts +24 -0
  220. package/src/lib/exporters/table-body-row.spec.ts +48 -0
  221. package/src/lib/exporters/table-body-row.ts +24 -0
  222. package/src/lib/exporters/table.spec.ts +48 -0
  223. package/src/lib/exporters/table.ts +24 -0
  224. package/src/lib/exporters/trailer.spec.ts +48 -0
  225. package/src/lib/exporters/trailer.ts +24 -0
  226. package/src/lib/exporters/util.ts +62 -0
  227. package/src/lib/passage.ts +182 -0
  228. package/src/lib/titles.ts +80 -0
  229. package/src/lib/transformers/abbreviation.spec.ts +87 -0
  230. package/src/lib/transformers/abbreviation.ts +30 -0
  231. package/src/lib/transformers/annotate.ts +146 -0
  232. package/src/lib/transformers/audio.spec.ts +55 -0
  233. package/src/lib/transformers/audio.ts +29 -0
  234. package/src/lib/transformers/blockquote.spec.ts +48 -0
  235. package/src/lib/transformers/blockquote.ts +41 -0
  236. package/src/lib/transformers/code.spec.ts +52 -0
  237. package/src/lib/transformers/code.ts +22 -0
  238. package/src/lib/transformers/deprecated.ts +7 -0
  239. package/src/lib/transformers/end-note-link.spec.ts +56 -0
  240. package/src/lib/transformers/end-note-link.ts +76 -0
  241. package/src/lib/transformers/glossary-instance.spec.ts +55 -0
  242. package/src/lib/transformers/glossary-instance.ts +40 -0
  243. package/src/lib/transformers/has-abbreviation.spec.ts +50 -0
  244. package/src/lib/transformers/has-abbreviation.ts +30 -0
  245. package/src/lib/transformers/heading.spec.ts +62 -0
  246. package/src/lib/transformers/heading.ts +30 -0
  247. package/src/lib/transformers/image.spec.ts +51 -0
  248. package/src/lib/transformers/image.ts +28 -0
  249. package/src/lib/transformers/indent.spec.ts +53 -0
  250. package/src/lib/transformers/indent.ts +17 -0
  251. package/src/lib/transformers/index.ts +33 -0
  252. package/src/lib/transformers/inline-title.spec.ts +59 -0
  253. package/src/lib/transformers/inline-title.ts +34 -0
  254. package/src/lib/transformers/internal-link.spec.ts +67 -0
  255. package/src/lib/transformers/internal-link.ts +65 -0
  256. package/src/lib/transformers/italic.ts +22 -0
  257. package/src/lib/transformers/leading-space.spec.ts +55 -0
  258. package/src/lib/transformers/leading-space.ts +17 -0
  259. package/src/lib/transformers/line-group.spec.ts +48 -0
  260. package/src/lib/transformers/line-group.ts +37 -0
  261. package/src/lib/transformers/line.spec.ts +54 -0
  262. package/src/lib/transformers/line.ts +27 -0
  263. package/src/lib/transformers/link.spec.ts +61 -0
  264. package/src/lib/transformers/link.ts +27 -0
  265. package/src/lib/transformers/list-item.spec.ts +48 -0
  266. package/src/lib/transformers/list-item.ts +37 -0
  267. package/src/lib/transformers/list.spec.ts +58 -0
  268. package/src/lib/transformers/list.ts +42 -0
  269. package/src/lib/transformers/mantra.spec.ts +58 -0
  270. package/src/lib/transformers/mantra.ts +28 -0
  271. package/src/lib/transformers/mention.ts +70 -0
  272. package/src/lib/transformers/paragraph.spec.ts +46 -0
  273. package/src/lib/transformers/paragraph.ts +26 -0
  274. package/src/lib/transformers/quote.spec.ts +42 -0
  275. package/src/lib/transformers/quote.ts +3 -0
  276. package/src/lib/transformers/quoted.ts +3 -0
  277. package/src/lib/transformers/recurse.ts +76 -0
  278. package/src/lib/transformers/reference.ts +3 -0
  279. package/src/lib/transformers/span.spec.ts +68 -0
  280. package/src/lib/transformers/span.ts +78 -0
  281. package/src/lib/transformers/split-at.ts +58 -0
  282. package/src/lib/transformers/split-block.ts +110 -0
  283. package/src/lib/transformers/split-content.ts +67 -0
  284. package/src/lib/transformers/split-insert.ts +76 -0
  285. package/src/lib/transformers/split-marks.ts +42 -0
  286. package/src/lib/transformers/split-node.ts +138 -0
  287. package/src/lib/transformers/table-body-data.spec.ts +44 -0
  288. package/src/lib/transformers/table-body-data.ts +29 -0
  289. package/src/lib/transformers/table-body-header.spec.ts +44 -0
  290. package/src/lib/transformers/table-body-header.ts +29 -0
  291. package/src/lib/transformers/table-body-row.spec.ts +44 -0
  292. package/src/lib/transformers/table-body-row.ts +29 -0
  293. package/src/lib/transformers/table.spec.ts +47 -0
  294. package/src/lib/transformers/table.ts +29 -0
  295. package/src/lib/transformers/trailer.spec.ts +43 -0
  296. package/src/lib/transformers/trailer.ts +26 -0
  297. package/src/lib/transformers/transformer.ts +25 -0
  298. package/src/lib/transformers/unknown.ts +8 -0
  299. package/src/lib/transformers/util.ts +20 -0
  300. package/src/lib/types.ts +10 -0
  301. package/src/ssr.ts +1 -0
  302. package/tsconfig.json +20 -0
  303. package/tsconfig.lib.json +29 -0
  304. package/tsconfig.spec.json +22 -0
@@ -0,0 +1,71 @@
1
+ import { Button, Input } from '@eightyfourthousand/design-system';
2
+ import { CheckIcon, Trash2Icon } from 'lucide-react';
3
+ import { useRef } from 'react';
4
+
5
+ export const InternalLinkInput = ({
6
+ type,
7
+ uuid,
8
+ onSubmit,
9
+ }: {
10
+ type: string;
11
+ uuid: string;
12
+ onSubmit: (type?: string, entity?: string) => void;
13
+ }) => {
14
+ const typeRef = useRef<HTMLInputElement>(null);
15
+ const uuidRef = useRef<HTMLInputElement>(null);
16
+
17
+ const submit = () => {
18
+ onSubmit(typeRef.current?.value, uuidRef.current?.value);
19
+ };
20
+
21
+ return (
22
+ <div className="flex justify-between gap-2 p-0 w-fit max-w-80">
23
+ <Input
24
+ ref={typeRef}
25
+ placeholder="Entity type..."
26
+ value={type}
27
+ onKeyDown={(e) => {
28
+ if (e.key === 'Enter') {
29
+ e.preventDefault();
30
+ submit();
31
+ }
32
+ }}
33
+ />
34
+ <Input
35
+ ref={uuidRef}
36
+ placeholder="Entity uuid..."
37
+ value={uuid}
38
+ onKeyDown={(e) => {
39
+ if (e.key === 'Enter') {
40
+ e.preventDefault();
41
+ submit();
42
+ }
43
+ }}
44
+ />
45
+ <Button
46
+ size="icon"
47
+ variant="ghost"
48
+ disabled={!!uuidRef.current && !!typeRef.current}
49
+ onClick={submit}
50
+ >
51
+ <CheckIcon />
52
+ </Button>
53
+ <Button
54
+ variant="ghost"
55
+ size="icon"
56
+ type="button"
57
+ onClick={() => {
58
+ if (uuidRef.current) {
59
+ uuidRef.current.value = '';
60
+ }
61
+ if (typeRef.current) {
62
+ typeRef.current.value = '';
63
+ }
64
+ onSubmit();
65
+ }}
66
+ >
67
+ <Trash2Icon className="text-destructive" />
68
+ </Button>
69
+ </div>
70
+ );
71
+ };
@@ -0,0 +1 @@
1
+ export * from './InternalLink';
@@ -0,0 +1,50 @@
1
+ import { Italic as TiptapItalic } from '@tiptap/extension-italic';
2
+ import { mergeAttributes } from '@tiptap/react';
3
+ import { v4 as uuidv4 } from 'uuid';
4
+
5
+ export const Italic = TiptapItalic.extend({
6
+ addAttributes() {
7
+ return {
8
+ ...this.parent?.(),
9
+ type: {
10
+ default: undefined,
11
+ parseHTML: (element) => element.getAttribute('data-type'),
12
+ renderHTML(attributes) {
13
+ return mergeAttributes(attributes, { 'data-type': attributes.type });
14
+ },
15
+ },
16
+ lang: {
17
+ default: undefined,
18
+ parseHTML: (element) => element.getAttribute('data-lang'),
19
+ renderHTML(attributes) {
20
+ return mergeAttributes(attributes, { 'data-lang': attributes.lang });
21
+ },
22
+ },
23
+ textStyle: {
24
+ default: undefined,
25
+ parseHTML: (element) => element.getAttribute('data-text-style'),
26
+ renderHTML(attributes) {
27
+ return mergeAttributes(attributes, {
28
+ 'data-text-style': attributes.textStyle,
29
+ });
30
+ },
31
+ },
32
+ };
33
+ },
34
+ addCommands() {
35
+ const name = this.name;
36
+ return {
37
+ ...this.parent?.(),
38
+ setItalic() {
39
+ return ({ commands }) => {
40
+ return commands.setMark(name, { uuid: uuidv4() });
41
+ };
42
+ },
43
+ toggleItalic() {
44
+ return ({ commands }) => {
45
+ return commands.toggleMark(name, { uuid: uuidv4() });
46
+ };
47
+ },
48
+ };
49
+ },
50
+ });
@@ -0,0 +1,106 @@
1
+ import { Extension, mergeAttributes } from '@tiptap/core';
2
+ import { v4 as uuidv4 } from 'uuid';
3
+
4
+ export interface LeadingSpaceOptions {
5
+ types: string[];
6
+ defaultHasLeadingSpace: boolean;
7
+ }
8
+
9
+ declare module '@tiptap/core' {
10
+ interface Commands<ReturnType> {
11
+ leadingSpace: {
12
+ /**
13
+ * Add a leading space
14
+ */
15
+ setLeadingSpace: () => ReturnType;
16
+ /**
17
+ * Remove a leading space
18
+ */
19
+ unsetLeadingSpace: () => ReturnType;
20
+
21
+ /**
22
+ * Toggle a leading space
23
+ */
24
+ toggleLeadingSpace: () => ReturnType;
25
+ };
26
+ }
27
+ }
28
+
29
+ export const LeadingSpace = Extension.create<LeadingSpaceOptions>({
30
+ name: 'leadingSpace',
31
+ addOptions() {
32
+ return {
33
+ types: ['paragraph', 'heading', 'lineGroup'],
34
+ defaultHasLeadingSpace: false,
35
+ };
36
+ },
37
+ addGlobalAttributes() {
38
+ return [
39
+ {
40
+ types: this.options.types,
41
+ attributes: {
42
+ hasLeadingSpace: {
43
+ default: this.options.defaultHasLeadingSpace,
44
+ parseHTML: (element) => element.className.includes('mt-6'),
45
+ renderHTML: (attributes) => {
46
+ if (attributes.hasLeadingSpace) {
47
+ return mergeAttributes(attributes, { class: 'mt-6 no-indent' });
48
+ }
49
+ return {};
50
+ },
51
+ },
52
+ leadingSpaceUuid: {
53
+ default: undefined,
54
+ parseHTML: (element) =>
55
+ element.getAttribute('data-leading-space-uuid') || undefined,
56
+ renderHTML: (attributes) => {
57
+ if (!attributes.leadingSpaceUuid) {
58
+ return {};
59
+ }
60
+ return {
61
+ 'data-leading-space-uuid': attributes.leadingSpaceUuid,
62
+ };
63
+ },
64
+ },
65
+ },
66
+ },
67
+ ];
68
+ },
69
+
70
+ addCommands() {
71
+ return {
72
+ setLeadingSpace:
73
+ () =>
74
+ ({ commands }) => {
75
+ return this.options.types
76
+ .map((type) =>
77
+ commands.updateAttributes(type, {
78
+ hasLeadingSpace: true,
79
+ leadingSpaceUuid: uuidv4(),
80
+ }),
81
+ )
82
+ .every((response) => response);
83
+ },
84
+ unsetLeadingSpace:
85
+ () =>
86
+ ({ commands }) => {
87
+ return this.options.types
88
+ .map((type) =>
89
+ commands.resetAttributes(type, [
90
+ 'hasLeadingSpace',
91
+ 'leadingSpaceUuid',
92
+ ]),
93
+ )
94
+ .every((response) => response);
95
+ },
96
+ toggleLeadingSpace:
97
+ () =>
98
+ ({ editor, commands }) => {
99
+ if (editor.isActive({ hasLeadingSpace: true })) {
100
+ return commands.unsetLeadingSpace();
101
+ }
102
+ return commands.setLeadingSpace();
103
+ },
104
+ };
105
+ },
106
+ });
@@ -0,0 +1,41 @@
1
+ import { Node } from '@tiptap/core';
2
+ import { mergeAttributes } from '@tiptap/core';
3
+
4
+ export const LineNode = Node.create({
5
+ name: 'line',
6
+
7
+ addOptions() {
8
+ return {
9
+ HTMLAttributes: {},
10
+ };
11
+ },
12
+
13
+ content: 'text*',
14
+
15
+ defining: true,
16
+
17
+ parseHTML() {
18
+ return [
19
+ {
20
+ tag: 'li[type="line"]',
21
+ },
22
+ ];
23
+ },
24
+
25
+ renderHTML({ HTMLAttributes }) {
26
+ return [
27
+ 'li',
28
+ mergeAttributes(
29
+ { type: 'line', class: '-indent-8 pl-8' },
30
+ HTMLAttributes,
31
+ ),
32
+ 0,
33
+ ];
34
+ },
35
+
36
+ addKeyboardShortcuts() {
37
+ return {
38
+ Enter: () => this.editor.commands.splitListItem(this.name),
39
+ };
40
+ },
41
+ });
@@ -0,0 +1,124 @@
1
+ import { Node } from '@tiptap/core';
2
+ import { mergeAttributes, wrappingInputRule } from '@tiptap/react';
3
+ import { v4 as uuidv4 } from 'uuid';
4
+
5
+ export interface LineGroupOptions {
6
+ iitemTypeName: string;
7
+ HTMLAttributes: Record<string, unknown>;
8
+ keepMarks: boolean;
9
+ keepAttributes: boolean;
10
+ }
11
+
12
+ declare module '@tiptap/core' {
13
+ interface Commands<ReturnType> {
14
+ lineGroup: {
15
+ toggleLineGroup: () => ReturnType;
16
+ };
17
+ }
18
+ }
19
+
20
+ /**
21
+ * Matches a bullet list to a tilde.
22
+ */
23
+ export const inputRegex = /^\s*([~])\s$/;
24
+
25
+ export const LineGroupNode = Node.create({
26
+ name: 'lineGroup',
27
+
28
+ addOptions() {
29
+ return {
30
+ itemTypeName: 'line',
31
+ HTMLAttributes: {},
32
+ keepMarks: false,
33
+ keepAttributes: false,
34
+ };
35
+ },
36
+
37
+ group: 'block list',
38
+
39
+ content() {
40
+ return `${this.options.itemTypeName}+`;
41
+ },
42
+
43
+ parseHTML() {
44
+ return [
45
+ {
46
+ tag: 'ul[type="line-group"]',
47
+ },
48
+ ];
49
+ },
50
+
51
+ renderHTML({ HTMLAttributes }) {
52
+ return [
53
+ 'ul',
54
+ mergeAttributes({ type: 'line-group', class: 'my-4' }, HTMLAttributes),
55
+ 0,
56
+ ];
57
+ },
58
+
59
+ addCommands() {
60
+ return {
61
+ toggleLineGroup:
62
+ () =>
63
+ ({ commands, chain }) => {
64
+ if (this.options.keepAttributes) {
65
+ return chain()
66
+ .toggleList(
67
+ this.name,
68
+ this.options.itemTypeName,
69
+ this.options.keepMarks,
70
+ {
71
+ uuid: uuidv4(),
72
+ },
73
+ )
74
+ .updateAttributes('line', this.editor.getAttributes('textStyle'))
75
+ .run();
76
+ }
77
+ return commands.toggleList(
78
+ this.name,
79
+ this.options.itemTypeName,
80
+ this.options.keepMarks,
81
+ {
82
+ uuid: uuidv4(),
83
+ },
84
+ );
85
+ },
86
+ };
87
+ },
88
+
89
+ addAttributes() {
90
+ return {
91
+ type: {
92
+ default: 'line-group',
93
+ parseHTML: (element) => element.getAttribute('type'),
94
+ },
95
+ };
96
+ },
97
+
98
+ addKeyboardShortcuts() {
99
+ return {
100
+ 'Mod-Shift-9': () => this.editor.commands.toggleLineGroup(),
101
+ };
102
+ },
103
+
104
+ addInputRules() {
105
+ let inputRule = wrappingInputRule({
106
+ find: inputRegex,
107
+ type: this.type,
108
+ });
109
+
110
+ if (this.options.keepMarks || this.options.keepAttributes) {
111
+ inputRule = wrappingInputRule({
112
+ find: inputRegex,
113
+ type: this.type,
114
+ keepMarks: this.options.keepMarks,
115
+ keepAttributes: this.options.keepAttributes,
116
+ getAttributes: () => {
117
+ return this.editor.getAttributes('textStyle');
118
+ },
119
+ editor: this.editor,
120
+ });
121
+ }
122
+ return [inputRule];
123
+ },
124
+ });
@@ -0,0 +1,65 @@
1
+ import { Link as TipTapLink } from '@tiptap/extension-link';
2
+ import { v4 as uuidv4 } from 'uuid';
3
+ import { createMarkViewDom, registerEditorElement } from '../../util';
4
+
5
+ export const Link = TipTapLink.extend({
6
+ addAttributes() {
7
+ return {
8
+ ...this.parent?.(),
9
+ uuid: {
10
+ default: undefined,
11
+ parseHTML: (element) => element.getAttribute('uuid'),
12
+ },
13
+ };
14
+ },
15
+ addCommands() {
16
+ const name = this.name;
17
+ return {
18
+ ...this.parent?.(),
19
+ setLink(attributes) {
20
+ return ({ commands }) => {
21
+ return commands.setMark(name, {
22
+ ...attributes,
23
+ uuid: uuidv4(),
24
+ });
25
+ };
26
+ },
27
+ toggleLink() {
28
+ return ({ commands }) => {
29
+ return commands.toggleMark(name, { uuid: uuidv4() });
30
+ };
31
+ },
32
+ };
33
+ },
34
+ addMarkView() {
35
+ return (props) => {
36
+ const isEditable = props.editor.isEditable;
37
+ const { dom } = createMarkViewDom({
38
+ ...props,
39
+ element: 'a',
40
+ });
41
+
42
+ dom.setAttribute('href', props.mark.attrs.href);
43
+ dom.setAttribute('target', '_blank');
44
+ dom.setAttribute('rel', 'noreferrer');
45
+
46
+ // Set uuid attribute for HoverCardProvider identification
47
+ if (props.mark.attrs.uuid) {
48
+ dom.setAttribute('uuid', props.mark.attrs.uuid);
49
+ }
50
+
51
+ // Only add type attribute in edit mode for hover card detection
52
+ if (isEditable) {
53
+ dom.setAttribute('type', 'link');
54
+ registerEditorElement(dom, props.editor);
55
+ }
56
+
57
+ return {
58
+ dom,
59
+ contentDOM: dom,
60
+ };
61
+ };
62
+ },
63
+ }).configure({
64
+ openOnClick: true,
65
+ });
@@ -0,0 +1,124 @@
1
+ import { Button } from '@eightyfourthousand/design-system';
2
+ import { Editor } from '@tiptap/core';
3
+ import { GlobeIcon, PencilIcon, Trash2Icon } from 'lucide-react';
4
+ import { useCallback, useState } from 'react';
5
+ import { HoverInputField } from '../HoverInputField';
6
+ import { findMarkByUuid } from '../../util';
7
+ import { useHoverCard } from '../../../shared/HoverCardProvider';
8
+
9
+ const EDITOR_UPDATE_DELAY_MS = 100;
10
+
11
+ export const LinkHoverContent = ({
12
+ uuid,
13
+ href,
14
+ editor,
15
+ anchor,
16
+ }: {
17
+ uuid: string;
18
+ href: string;
19
+ editor: Editor;
20
+ anchor: HTMLElement;
21
+ }) => {
22
+ const [isEditing, setIsEditingLocal] = useState(false);
23
+ const { close, setIsEditing: setIsEditingContext } = useHoverCard();
24
+
25
+ const setIsEditing = useCallback(
26
+ (editing: boolean) => {
27
+ setIsEditingLocal(editing);
28
+ setIsEditingContext(editing);
29
+ },
30
+ [setIsEditingContext],
31
+ );
32
+
33
+ const deleteLink = useCallback(() => {
34
+ setIsEditing(false);
35
+ close();
36
+
37
+ setTimeout(() => {
38
+ const range = findMarkByUuid({ editor, uuid, markType: 'link' });
39
+ if (!range) {
40
+ console.warn('Link mark not found in the document.');
41
+ return;
42
+ }
43
+
44
+ const { from, to, mark } = range;
45
+ const { tr } = editor.state;
46
+ tr.removeMark(from, to, mark.type);
47
+ editor.view.dispatch(tr);
48
+ }, EDITOR_UPDATE_DELAY_MS);
49
+ }, [editor, uuid, close, setIsEditing]);
50
+
51
+ const updateLink = useCallback(
52
+ (newHref: string) => {
53
+ setIsEditing(false);
54
+ close();
55
+
56
+ setTimeout(() => {
57
+ const range = findMarkByUuid({ editor, uuid, markType: 'link' });
58
+ if (!range) {
59
+ console.warn('Link mark not found in the document.');
60
+ return;
61
+ }
62
+
63
+ const { from, to, mark } = range;
64
+ const { tr } = editor.state;
65
+ tr.removeMark(from, to, mark.type);
66
+ tr.addMark(
67
+ from,
68
+ to,
69
+ mark.type.create({ ...mark.attrs, href: newHref }),
70
+ );
71
+ editor.view.dispatch(tr);
72
+
73
+ // Update the DOM attribute directly for immediate feedback
74
+ anchor.setAttribute('href', newHref);
75
+ }, EDITOR_UPDATE_DELAY_MS);
76
+ },
77
+ [editor, uuid, anchor, close, setIsEditing],
78
+ );
79
+
80
+ return (
81
+ <div className="flex justify-between gap-2 p-2 w-fit max-w-72">
82
+ {isEditing ? (
83
+ <HoverInputField
84
+ type="link"
85
+ attr="href"
86
+ valueRef={href}
87
+ placeholder="Add link..."
88
+ onSubmit={(value) => {
89
+ if (value) {
90
+ updateLink(value);
91
+ } else {
92
+ deleteLink();
93
+ }
94
+ setIsEditing(false);
95
+ }}
96
+ />
97
+ ) : (
98
+ <>
99
+ <GlobeIcon className="text-primary my-auto size-6 [&_svg]:size-4" />
100
+ <span className="truncate text-muted-foreground text-sm my-auto">
101
+ {href}
102
+ </span>
103
+ <span className="flex-grow" />
104
+ <Button
105
+ variant="ghost"
106
+ size="icon"
107
+ className="size-6 [&_svg]:size-4"
108
+ onClick={() => setIsEditing(true)}
109
+ >
110
+ <PencilIcon className="text-primary my-auto" />
111
+ </Button>
112
+ <Button
113
+ variant="ghost"
114
+ size="icon"
115
+ className="size-6 [&_svg]:size-4"
116
+ onClick={deleteLink}
117
+ >
118
+ <Trash2Icon className="text-destructive my-auto" />
119
+ </Button>
120
+ </>
121
+ )}
122
+ </div>
123
+ );
124
+ };
@@ -0,0 +1 @@
1
+ export * from './Link';
@@ -0,0 +1,74 @@
1
+ import { BulletList, ListItem as TipTapListItem } from '@tiptap/extension-list';
2
+ import { mergeAttributes } from '@tiptap/react';
3
+
4
+ const ITEM_STYLE_TO_CLASS: { [key: string]: string } = {
5
+ none: 'list-none',
6
+ dots: 'list-[disc]',
7
+ circles: 'list-[circle]',
8
+ numbers: 'list-decimal',
9
+ letters: 'list-[lower-latin]',
10
+ };
11
+
12
+ const LIST_SPACING_TO_CLASS: { [key: string]: string } = {
13
+ horizontal: 'ml-4',
14
+ vertical: 'mt-4',
15
+ };
16
+
17
+ export const ListItem = TipTapListItem.extend({
18
+ content: '(paragraph | heading) block*',
19
+ });
20
+
21
+ export const List = BulletList.extend({
22
+ addOptions() {
23
+ return {
24
+ itemTypeName: 'listItem',
25
+ keepMarks: false,
26
+ keepAttributes: false,
27
+ ...this.parent?.(),
28
+ HTMLAttributes: {},
29
+ };
30
+ },
31
+ addAttributes() {
32
+ return {
33
+ ...this.parent?.(),
34
+ spacing: {
35
+ default: 'horizontal',
36
+ parseHTML: (element) => element.getAttribute('data-spacing'),
37
+ renderHTML(attributes) {
38
+ return mergeAttributes(attributes, {
39
+ 'data-spacing': attributes.spacing,
40
+ class:
41
+ LIST_SPACING_TO_CLASS[attributes.spacing || 'horizontal'] ||
42
+ 'ml-4',
43
+ });
44
+ },
45
+ },
46
+ nesting: {
47
+ default: undefined,
48
+ parseHTML: (element) => {
49
+ const nesting = element.getAttribute('data-nesting');
50
+ return nesting ? parseInt(nesting, 10) : 0;
51
+ },
52
+ renderHTML(attributes) {
53
+ return mergeAttributes(attributes, {
54
+ 'data-nesting': `${attributes.nesting}`,
55
+ });
56
+ },
57
+ },
58
+ itemStyle: {
59
+ default: 'none',
60
+ parseHTML: (element) => element.getAttribute('data-item-style'),
61
+ renderHTML(attributes) {
62
+ return mergeAttributes(attributes, {
63
+ 'data-item-style': attributes.itemStyle,
64
+ class:
65
+ ITEM_STYLE_TO_CLASS[attributes.itemStyle || 'none'] ||
66
+ 'list-none',
67
+ });
68
+ },
69
+ },
70
+ };
71
+ },
72
+ });
73
+
74
+ export default List;