reactjs-tiptap-editor-pro 0.2.29 → 0.2.30

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 (333) hide show
  1. package/package.json +4 -60
  2. package/src/components/ActionButton.tsx +103 -0
  3. package/src/components/ActionMenuButton.tsx +76 -0
  4. package/src/components/BubbleMenu.tsx +93 -0
  5. package/src/components/CharactorCount.tsx +50 -0
  6. package/src/components/ColorPicker.tsx +272 -0
  7. package/src/components/RichTextEditor.tsx +212 -0
  8. package/src/components/SizeSetter/SizeSetter.tsx +102 -0
  9. package/src/components/Toolbar.tsx +108 -0
  10. package/src/components/icons/Activity.tsx +19 -0
  11. package/src/components/icons/Animas.tsx +24 -0
  12. package/src/components/icons/AspectRatio.tsx +13 -0
  13. package/src/components/icons/Blockquote.tsx +17 -0
  14. package/src/components/icons/ColumnAddLeft.tsx +7 -0
  15. package/src/components/icons/ColumnAddRight.tsx +7 -0
  16. package/src/components/icons/DeleteColumn.tsx +26 -0
  17. package/src/components/icons/DeleteRow.tsx +26 -0
  18. package/src/components/icons/Direction.tsx +7 -0
  19. package/src/components/icons/Excalidraw.tsx +7 -0
  20. package/src/components/icons/ExportPdf.tsx +8 -0
  21. package/src/components/icons/ExportWord.tsx +24 -0
  22. package/src/components/icons/FileWordOutline.tsx +13 -0
  23. package/src/components/icons/Flag.tsx +19 -0
  24. package/src/components/icons/Food.tsx +20 -0
  25. package/src/components/icons/GIfIcon.tsx +10 -0
  26. package/src/components/icons/Icon.tsx +30 -0
  27. package/src/components/icons/ImportWord.tsx +23 -0
  28. package/src/components/icons/LeftToRight.tsx +7 -0
  29. package/src/components/icons/LineHeight.tsx +13 -0
  30. package/src/components/icons/MenuDown.tsx +24 -0
  31. package/src/components/icons/Mermaid.tsx +13 -0
  32. package/src/components/icons/NoFill.tsx +7 -0
  33. package/src/components/icons/Object.tsx +24 -0
  34. package/src/components/icons/RightToLeft.tsx +7 -0
  35. package/src/components/icons/SizeL.tsx +9 -0
  36. package/src/components/icons/SizeM.tsx +13 -0
  37. package/src/components/icons/SizeS.tsx +13 -0
  38. package/src/components/icons/Symbol.tsx +19 -0
  39. package/src/components/icons/Travel.tsx +24 -0
  40. package/src/components/icons/Twitter.tsx +7 -0
  41. package/src/components/icons/icons.ts +212 -0
  42. package/src/components/icons/index.ts +12 -0
  43. package/src/components/index.ts +9 -0
  44. package/src/components/menus/bubble.ts +395 -0
  45. package/src/components/menus/components/BubbleMenuContent.tsx +15 -0
  46. package/src/components/menus/components/BubbleMenuDrawer.tsx +128 -0
  47. package/src/components/menus/components/BubbleMenuExcalidraw.tsx +91 -0
  48. package/src/components/menus/components/BubbleMenuIframe.tsx +143 -0
  49. package/src/components/menus/components/BubbleMenuKatex.tsx +136 -0
  50. package/src/components/menus/components/BubbleMenuLink.tsx +99 -0
  51. package/src/components/menus/components/BubbleMenuMedia.tsx +235 -0
  52. package/src/components/menus/components/BubbleMenuMermaid.tsx +128 -0
  53. package/src/components/menus/components/BubbleMenuText.tsx +102 -0
  54. package/src/components/menus/components/BubbleMenuTwitter.tsx +91 -0
  55. package/src/components/menus/components/ColumnsBubbleMenu.tsx +59 -0
  56. package/src/components/menus/components/ContentMenu.tsx +396 -0
  57. package/src/components/menus/components/TableBubbleMenu.tsx +362 -0
  58. package/src/components/menus/index.ts +7 -0
  59. package/src/components/ui/button.tsx +56 -0
  60. package/src/components/ui/checkbox.tsx +30 -0
  61. package/src/components/ui/dialog.tsx +128 -0
  62. package/src/components/ui/dropdown-menu.tsx +203 -0
  63. package/src/components/ui/emoji-picker.tsx +166 -0
  64. package/src/components/ui/index.ts +14 -0
  65. package/src/components/ui/input.tsx +25 -0
  66. package/src/components/ui/label.tsx +26 -0
  67. package/src/components/ui/popover.tsx +31 -0
  68. package/src/components/ui/select.tsx +162 -0
  69. package/src/components/ui/separator.tsx +31 -0
  70. package/src/components/ui/switch.tsx +29 -0
  71. package/src/components/ui/tabs.tsx +55 -0
  72. package/src/components/ui/textarea.tsx +24 -0
  73. package/src/components/ui/toast.tsx +129 -0
  74. package/src/components/ui/toaster.tsx +44 -0
  75. package/src/components/ui/toggle.tsx +45 -0
  76. package/src/components/ui/tooltip.tsx +30 -0
  77. package/src/components/ui/use-toast.ts +197 -0
  78. package/src/constants/index.ts +223 -0
  79. package/src/constants/resetCSS.ts +139 -0
  80. package/src/extension-bundle.ts +2 -0
  81. package/src/extensions/Attachment/Attachment.ts +144 -0
  82. package/src/extensions/Attachment/components/NodeViewAttachment/FileIcon.tsx +69 -0
  83. package/src/extensions/Attachment/components/NodeViewAttachment/FileIconString.ts +28 -0
  84. package/src/extensions/Attachment/components/NodeViewAttachment/NodeViewAttachment.tsx +155 -0
  85. package/src/extensions/Attachment/components/NodeViewAttachment/index.module.scss +23 -0
  86. package/src/extensions/Attachment/index.ts +1 -0
  87. package/src/extensions/BaseKit.ts +253 -0
  88. package/src/extensions/Blockquote/Blockquote.ts +31 -0
  89. package/src/extensions/Blockquote/index.ts +1 -0
  90. package/src/extensions/Bold/Bold.ts +26 -0
  91. package/src/extensions/Bold/index.ts +1 -0
  92. package/src/extensions/BulletList/BulletList.ts +28 -0
  93. package/src/extensions/BulletList/index.ts +1 -0
  94. package/src/extensions/Clear/Clear.ts +24 -0
  95. package/src/extensions/Clear/index.ts +1 -0
  96. package/src/extensions/Code/Code.ts +26 -0
  97. package/src/extensions/Code/index.ts +1 -0
  98. package/src/extensions/CodeBlock/CodeBlock.ts +54 -0
  99. package/src/extensions/CodeBlock/components/CodeBlockActiveButton.tsx +66 -0
  100. package/src/extensions/CodeBlock/components/NodeViewCodeBlock/NodeViewCodeBlock.tsx +89 -0
  101. package/src/extensions/CodeBlock/components/NodeViewCodeBlock/index.module.scss +81 -0
  102. package/src/extensions/CodeBlock/highlighter.ts +132 -0
  103. package/src/extensions/CodeBlock/index.ts +1 -0
  104. package/src/extensions/CodeBlock/shiki-plugin.ts +213 -0
  105. package/src/extensions/Color/Color.ts +52 -0
  106. package/src/extensions/Color/components/ColorActionButton.tsx +104 -0
  107. package/src/extensions/Color/index.ts +1 -0
  108. package/src/extensions/Document/Document.ts +8 -0
  109. package/src/extensions/Document/index.ts +1 -0
  110. package/src/extensions/Drawer/Drawer.ts +177 -0
  111. package/src/extensions/Drawer/components/ControlDrawer/ControlDrawer.module.scss +85 -0
  112. package/src/extensions/Drawer/components/ControlDrawer/ControlDrawer.tsx +598 -0
  113. package/src/extensions/Drawer/components/ControlDrawer/icon.tsx +500 -0
  114. package/src/extensions/Drawer/components/DrawerActiveButton.tsx +239 -0
  115. package/src/extensions/Drawer/components/EditDrawerBlock.tsx +238 -0
  116. package/src/extensions/Drawer/components/NodeViewDrawer/NodeViewDrawer.tsx +260 -0
  117. package/src/extensions/Drawer/index.ts +1 -0
  118. package/src/extensions/Emoji/Emoji.ts +146 -0
  119. package/src/extensions/Emoji/components/EmojiList/EmojiList.tsx +103 -0
  120. package/src/extensions/Emoji/components/EmojiList/emojis.ts +1858 -0
  121. package/src/extensions/Emoji/components/EmojiPicker/EmojiPicker.tsx +61 -0
  122. package/src/extensions/Emoji/index.ts +1 -0
  123. package/src/extensions/Excalidraw/Excalidraw.ts +123 -0
  124. package/src/extensions/Excalidraw/components/ExcalidrawActiveButton.tsx +138 -0
  125. package/src/extensions/Excalidraw/components/NodeViewExcalidraw/NodeViewExcalidraw.tsx +178 -0
  126. package/src/extensions/Excalidraw/components/NodeViewExcalidraw/index.module.scss +43 -0
  127. package/src/extensions/Excalidraw/index.ts +1 -0
  128. package/src/extensions/ExportPdf/ExportPdf.ts +25 -0
  129. package/src/extensions/ExportPdf/index.ts +1 -0
  130. package/src/extensions/ExportWord/ExportWord.ts +87 -0
  131. package/src/extensions/ExportWord/index.ts +1 -0
  132. package/src/extensions/FontFamily/FontFamily.ts +64 -0
  133. package/src/extensions/FontFamily/components/FontFamilyButton.tsx +97 -0
  134. package/src/extensions/FontFamily/index.ts +1 -0
  135. package/src/extensions/FontSize/FontSize.ts +119 -0
  136. package/src/extensions/FontSize/components/FontSizeMenuButton.tsx +84 -0
  137. package/src/extensions/FontSize/index.ts +1 -0
  138. package/src/extensions/FormatPainter/FormatPainter.ts +121 -0
  139. package/src/extensions/FormatPainter/index.ts +1 -0
  140. package/src/extensions/Heading/Heading.ts +57 -0
  141. package/src/extensions/Heading/components/HeadingButton.tsx +96 -0
  142. package/src/extensions/Heading/index.ts +1 -0
  143. package/src/extensions/Highlight/Highlight.ts +36 -0
  144. package/src/extensions/Highlight/components/HighlightActionButton.tsx +108 -0
  145. package/src/extensions/Highlight/index.ts +1 -0
  146. package/src/extensions/History/History.ts +39 -0
  147. package/src/extensions/History/components/HistoryActionButton.tsx +74 -0
  148. package/src/extensions/History/index.ts +1 -0
  149. package/src/extensions/HorizontalRule/HorizontalRule.ts +42 -0
  150. package/src/extensions/HorizontalRule/index.ts +1 -0
  151. package/src/extensions/Iframe/Iframe.ts +140 -0
  152. package/src/extensions/Iframe/components/IframeNodeView.tsx +92 -0
  153. package/src/extensions/Iframe/components/index.module.scss +40 -0
  154. package/src/extensions/Iframe/embed.ts +487 -0
  155. package/src/extensions/Iframe/index.ts +1 -0
  156. package/src/extensions/Image/Image.ts +303 -0
  157. package/src/extensions/Image/components/ActionImageButton.tsx +186 -0
  158. package/src/extensions/Image/components/ImageCropper.tsx +198 -0
  159. package/src/extensions/Image/components/ImageView.tsx +271 -0
  160. package/src/extensions/Image/index.ts +1 -0
  161. package/src/extensions/Image/store.ts +15 -0
  162. package/src/extensions/ImageGif/ImageGif.ts +176 -0
  163. package/src/extensions/ImageGif/components/ImageGifActionButton.tsx +138 -0
  164. package/src/extensions/ImageGif/components/ImageGifView.tsx +260 -0
  165. package/src/extensions/ImageGif/index.ts +1 -0
  166. package/src/extensions/ImportWord/ImportWord.ts +52 -0
  167. package/src/extensions/ImportWord/components/ImportWordButton.tsx +151 -0
  168. package/src/extensions/ImportWord/index.ts +1 -0
  169. package/src/extensions/Indent/Indent.ts +110 -0
  170. package/src/extensions/Indent/index.ts +1 -0
  171. package/src/extensions/Italic/Italic.ts +29 -0
  172. package/src/extensions/Italic/index.ts +1 -0
  173. package/src/extensions/Katex/Katex.ts +109 -0
  174. package/src/extensions/Katex/components/KatexActiveButton.tsx +117 -0
  175. package/src/extensions/Katex/components/KatexWrapper.tsx +53 -0
  176. package/src/extensions/Katex/index.ts +1 -0
  177. package/src/extensions/LineHeight/LineHeight.ts +76 -0
  178. package/src/extensions/LineHeight/components/LineHeightDropdown.tsx +93 -0
  179. package/src/extensions/LineHeight/index.ts +1 -0
  180. package/src/extensions/Link/Link.ts +92 -0
  181. package/src/extensions/Link/components/LinkEditBlock.tsx +110 -0
  182. package/src/extensions/Link/components/LinkEditPopover.tsx +46 -0
  183. package/src/extensions/Link/components/LinkViewBlock.tsx +54 -0
  184. package/src/extensions/Link/index.ts +1 -0
  185. package/src/extensions/ListItem/ListItem.ts +1 -0
  186. package/src/extensions/ListItem/index.ts +1 -0
  187. package/src/extensions/Mention/Mention.ts +100 -0
  188. package/src/extensions/Mention/components/NodeViewMentionList/NodeViewMentionList.tsx +94 -0
  189. package/src/extensions/Mention/components/NodeViewMentionList/index.module.scss +38 -0
  190. package/src/extensions/Mention/index.ts +1 -0
  191. package/src/extensions/Mermaid/Mermaid.ts +177 -0
  192. package/src/extensions/Mermaid/components/EditMermaidBlock.tsx +155 -0
  193. package/src/extensions/Mermaid/components/MermaidActiveButton.tsx +151 -0
  194. package/src/extensions/Mermaid/components/NodeViewMermaid/NodeViewMermaid.tsx +260 -0
  195. package/src/extensions/Mermaid/index.ts +1 -0
  196. package/src/extensions/MoreMark/MoreMark.ts +102 -0
  197. package/src/extensions/MoreMark/components/ActionMoreButton.tsx +97 -0
  198. package/src/extensions/MoreMark/index.ts +1 -0
  199. package/src/extensions/MultiColumn/Column.ts +36 -0
  200. package/src/extensions/MultiColumn/MultiColumn.ts +111 -0
  201. package/src/extensions/MultiColumn/components/ColumnActionButton.ts +22 -0
  202. package/src/extensions/MultiColumn/index.ts +3 -0
  203. package/src/extensions/OrderedList/OrderedList.ts +28 -0
  204. package/src/extensions/OrderedList/index.ts +1 -0
  205. package/src/extensions/SearchAndReplace/SearchAndReplace.ts +395 -0
  206. package/src/extensions/SearchAndReplace/components/SearchAndReplaceButton.tsx +190 -0
  207. package/src/extensions/SearchAndReplace/index.ts +1 -0
  208. package/src/extensions/Selection/Selection.ts +32 -0
  209. package/src/extensions/Selection/index.ts +1 -0
  210. package/src/extensions/SlashCommand/SlashCommand.ts +255 -0
  211. package/src/extensions/SlashCommand/components/CommandsList.tsx +180 -0
  212. package/src/extensions/SlashCommand/groups.ts +183 -0
  213. package/src/extensions/SlashCommand/index.ts +1 -0
  214. package/src/extensions/SlashCommand/types.ts +24 -0
  215. package/src/extensions/Strike/Strike.ts +26 -0
  216. package/src/extensions/Strike/index.ts +1 -0
  217. package/src/extensions/Subscript/Subscript.ts +88 -0
  218. package/src/extensions/Subscript/index.ts +1 -0
  219. package/src/extensions/Table/Cell.ts +131 -0
  220. package/src/extensions/Table/Header.ts +93 -0
  221. package/src/extensions/Table/Row.ts +8 -0
  222. package/src/extensions/Table/Table.ts +60 -0
  223. package/src/extensions/Table/cell-background.ts +112 -0
  224. package/src/extensions/Table/components/CreateTablePopover.tsx +132 -0
  225. package/src/extensions/Table/components/TableActionButton.tsx +42 -0
  226. package/src/extensions/Table/index.ts +6 -0
  227. package/src/extensions/Table/utils.ts +352 -0
  228. package/src/extensions/TableOfContent/TableOfContent.ts +124 -0
  229. package/src/extensions/TableOfContent/components/NodeViewTableOfContent.tsx +116 -0
  230. package/src/extensions/TableOfContent/components/TableOfContentActionButton.tsx +27 -0
  231. package/src/extensions/TableOfContent/components/index.module.scss +40 -0
  232. package/src/extensions/TableOfContent/index.ts +1 -0
  233. package/src/extensions/TaskList/TaskList.ts +46 -0
  234. package/src/extensions/TaskList/index.ts +1 -0
  235. package/src/extensions/TextAlign/TextAlign.ts +68 -0
  236. package/src/extensions/TextAlign/components/TextAlignMenuButton.tsx +103 -0
  237. package/src/extensions/TextAlign/index.ts +1 -0
  238. package/src/extensions/TextBubble/TextBubble.ts +22 -0
  239. package/src/extensions/TextBubble/components/TextDropdown.tsx +146 -0
  240. package/src/extensions/TextBubble/index.ts +1 -0
  241. package/src/extensions/TextDirection/TextDirection.ts +97 -0
  242. package/src/extensions/TextDirection/components/TextDirectionButton.tsx +103 -0
  243. package/src/extensions/TextDirection/index.ts +1 -0
  244. package/src/extensions/TrailingNode/TrailingNode.ts +71 -0
  245. package/src/extensions/TrailingNode/index.ts +1 -0
  246. package/src/extensions/Twitter/Twitter.ts +161 -0
  247. package/src/extensions/Twitter/components/FormEditLinkTwitter.tsx +68 -0
  248. package/src/extensions/Twitter/components/NodeViewTweet.tsx +30 -0
  249. package/src/extensions/Twitter/components/TwitterActiveButton.tsx +41 -0
  250. package/src/extensions/Twitter/index.ts +1 -0
  251. package/src/extensions/UnderLine/Underline.ts +30 -0
  252. package/src/extensions/UnderLine/index.ts +1 -0
  253. package/src/extensions/Video/Video.ts +204 -0
  254. package/src/extensions/Video/components/ActiveVideoButton.tsx +191 -0
  255. package/src/extensions/Video/index.ts +1 -0
  256. package/src/extensions/Video/store.ts +15 -0
  257. package/src/extensions/index.ts +122 -0
  258. package/src/hooks/useActive.tsx +24 -0
  259. package/src/hooks/useAttributes.tsx +45 -0
  260. package/src/hooks/useCopy.tsx +20 -0
  261. package/src/hooks/useEditorState.tsx +23 -0
  262. package/src/hooks/useExtension.tsx +29 -0
  263. package/src/index.ts +8 -0
  264. package/src/lib/utils.ts +7 -0
  265. package/src/locale-bundle.ts +3 -0
  266. package/src/locales/en.ts +173 -0
  267. package/src/locales/hu.ts +173 -0
  268. package/src/locales/index.tsx +163 -0
  269. package/src/locales/pt-br.ts +173 -0
  270. package/src/locales/vi.ts +173 -0
  271. package/src/locales/zh-cn.ts +173 -0
  272. package/src/plugins/DragHandle/index.ts +375 -0
  273. package/src/plugins/DragHandle/range.ts +114 -0
  274. package/src/plugins/DragHandle/utils.ts +80 -0
  275. package/src/plugins/image-upload.ts +160 -0
  276. package/src/store/ProviderRichText.tsx +53 -0
  277. package/src/store/editableEditor.ts +15 -0
  278. package/src/store/fast-context.tsx +70 -0
  279. package/src/store/store.ts +35 -0
  280. package/src/styles/ProseMirror.scss +176 -0
  281. package/src/styles/columns.scss +23 -0
  282. package/src/styles/editor.scss +411 -0
  283. package/src/styles/global.scss +87 -0
  284. package/src/styles/index.scss +5 -0
  285. package/src/styles/mention.scss +6 -0
  286. package/src/theme/theme.ts +15 -0
  287. package/src/types.ts +271 -0
  288. package/src/utils/_event.ts +55 -0
  289. package/src/utils/color.ts +67 -0
  290. package/src/utils/columns.ts +142 -0
  291. package/src/utils/customEvents/customEvents.ts +18 -0
  292. package/src/utils/customEvents/events.constant.ts +11 -0
  293. package/src/utils/delete-node.ts +46 -0
  294. package/src/utils/dom-dataset.ts +121 -0
  295. package/src/utils/download.ts +17 -0
  296. package/src/utils/dynamicCSS.ts +192 -0
  297. package/src/utils/editor-container-size.ts +28 -0
  298. package/src/utils/file.ts +112 -0
  299. package/src/utils/getRenderContainer.ts +41 -0
  300. package/src/utils/indent.ts +99 -0
  301. package/src/utils/is-mobile.ts +57 -0
  302. package/src/utils/json.ts +18 -0
  303. package/src/utils/line-height.ts +109 -0
  304. package/src/utils/lru-cache.ts +145 -0
  305. package/src/utils/mitt.ts +114 -0
  306. package/src/utils/node.ts +92 -0
  307. package/src/utils/pdf.ts +72 -0
  308. package/src/utils/plateform.ts +49 -0
  309. package/src/utils/shortId.ts +5 -0
  310. package/src/utils/storage.ts +18 -0
  311. package/src/utils/utils.ts +71 -0
  312. package/src/vite-env.d.ts +3 -0
  313. package/lib/RichTextEditor-BwbqJLnA.cjs +0 -141
  314. package/lib/RichTextEditor-iGJ6-rbq.js +0 -8833
  315. package/lib/extension-bundle.cjs +0 -33
  316. package/lib/extension-bundle.d.cts +0 -947
  317. package/lib/extension-bundle.d.ts +0 -947
  318. package/lib/extension-bundle.js +0 -5755
  319. package/lib/index-DV-nXpU1.cjs +0 -1
  320. package/lib/index-M6H3FoBi.js +0 -1147
  321. package/lib/index.cjs +0 -1
  322. package/lib/index.d.cts +0 -513
  323. package/lib/index.d.ts +0 -513
  324. package/lib/index.js +0 -16
  325. package/lib/locale-bundle.cjs +0 -1
  326. package/lib/locale-bundle.d.cts +0 -1140
  327. package/lib/locale-bundle.d.ts +0 -1140
  328. package/lib/locale-bundle.js +0 -9
  329. package/lib/style.css +0 -1
  330. package/lib/tiptap-DkWHMWDt.js +0 -6061
  331. package/lib/tiptap-gBG-1T-V.cjs +0 -116
  332. package/lib/vendor-BJ0Yf78E.cjs +0 -8114
  333. package/lib/vendor-Cpa6z-M0.js +0 -67575
@@ -0,0 +1,121 @@
1
+ /* eslint-disable @typescript-eslint/ban-ts-comment */
2
+
3
+ import { safeJSONParse } from '@/utils/json';
4
+
5
+ /**
6
+ * @param json
7
+ */
8
+ export function jsonToStr(json: Record<string, unknown>) {
9
+ try {
10
+ return JSON.stringify(json);
11
+ } catch {
12
+ return JSON.stringify({});
13
+ }
14
+ }
15
+
16
+ /**
17
+ * @param str
18
+ */
19
+ export function strToJSON(str: string) {
20
+ return safeJSONParse(str);
21
+ }
22
+
23
+ /**
24
+ * @param element
25
+ * @param json
26
+ */
27
+ export function jsonToDOMDataset(json: Record<string, unknown>) {
28
+ return Object.keys(json).map((key) => {
29
+ let value = json[key];
30
+
31
+ if (typeof value === 'object') {
32
+ value = JSON.stringify(value);
33
+ }
34
+
35
+ return {
36
+ key: `data-${key}`,
37
+ value: encodeURIComponent(value as string),
38
+ };
39
+ });
40
+ }
41
+
42
+ /**
43
+ * @param element
44
+ * @param attribute
45
+ * @param transformToJSON
46
+ */
47
+ export function getDatasetAttribute(attribute: string, transformToJSON = false) {
48
+ return (element: HTMLElement) => {
49
+ const dataKey = attribute.startsWith('data-') ? attribute : `data-${attribute}`;
50
+ // @ts-ignore
51
+ let value = decodeURIComponent(element.getAttribute(dataKey));
52
+
53
+ if (value == null || (typeof value === 'string' && value === 'null')) {
54
+ try {
55
+ const html = element.outerHTML;
56
+
57
+ const texts = html.match(/([\S\s])+?="([\S\s])+?"/g);
58
+ if (texts && texts.length > 0) {
59
+ const params = texts
60
+ .map(str => str.trim())
61
+ .reduce((accu, item) => {
62
+ const i = item.indexOf('=');
63
+ const arr = [item.slice(0, i), item.slice(i + 1).slice(1, -1)];
64
+ // @ts-expect-error
65
+ accu[arr[0]] = arr[1];
66
+ return accu;
67
+ }, {});
68
+
69
+ // @ts-expect-error
70
+ value = (params[attribute.toLowerCase()] || '').replaceAll('&quot;', '"');
71
+ }
72
+ } catch (e: any) {
73
+ console.error('Error getDatasetAttribute ', e.message, element);
74
+ }
75
+ }
76
+
77
+ if (transformToJSON) {
78
+ try {
79
+ return JSON.parse(value);
80
+ } catch {
81
+ return {};
82
+ }
83
+ }
84
+
85
+ if (value.includes('%') || value.includes('auto')) {
86
+ return value;
87
+ }
88
+
89
+ const toNumber = Number.parseInt(value);
90
+ return toNumber !== toNumber ? value : toNumber;
91
+ };
92
+ }
93
+
94
+ /**
95
+ * 将节点属性转换为 dataset
96
+ * @param node
97
+ * @returns
98
+ */
99
+ export function nodeAttrsToDataset(node: Node) {
100
+ const { attrs } = node as any;
101
+
102
+ return Object.keys(attrs).reduce((accu, key) => {
103
+ const value = attrs[key];
104
+
105
+ if (value == null) {
106
+ return accu;
107
+ }
108
+
109
+ let encodeValue = '';
110
+
111
+ if (typeof value === 'object') {
112
+ encodeValue = jsonToStr(value);
113
+ } else {
114
+ encodeValue = value;
115
+ }
116
+
117
+ accu[key] = encodeValue;
118
+
119
+ return accu;
120
+ }, Object.create(null));
121
+ }
@@ -0,0 +1,17 @@
1
+ const isBrowser = typeof window !== 'undefined';
2
+
3
+ export function downloadFromBlob(blob: Blob, filename: string) {
4
+ if (isBrowser) {
5
+ const url = window.URL.createObjectURL(blob);
6
+ const a = document.createElement('a');
7
+ a.href = url;
8
+ a.download = filename;
9
+ a.click();
10
+ window.URL.revokeObjectURL(url);
11
+ return Promise.resolve();
12
+ }
13
+
14
+ console.error('Download is not supported in Node.js');
15
+
16
+ return Promise.resolve();
17
+ }
@@ -0,0 +1,192 @@
1
+ /* eslint-disable @typescript-eslint/ban-ts-comment */
2
+ const APPEND_ORDER = 'data-rc-order';
3
+ const APPEND_PRIORITY = 'data-rc-priority';
4
+ const MARK_KEY = 'rc-util-key';
5
+
6
+ const containerCache = new Map<ContainerType, Node & ParentNode>();
7
+
8
+ export type ContainerType = Element | ShadowRoot;
9
+ export type Prepend = boolean | 'queue';
10
+ export type AppendType = 'prependQueue' | 'append' | 'prepend';
11
+
12
+ export default function contains(root: Node | null | undefined, n?: Node) {
13
+ if (!root) {
14
+ return false;
15
+ }
16
+
17
+ // Use native if support
18
+ if (root.contains) {
19
+ return root.contains(n as any);
20
+ }
21
+
22
+ // `document.contains` not support with IE11
23
+ let node = n;
24
+ while (node) {
25
+ if (node === root) {
26
+ return true;
27
+ }
28
+ // @ts-ignore
29
+ node = node.parentNode;
30
+ }
31
+
32
+ return false;
33
+ }
34
+
35
+ interface Options {
36
+ attachTo?: ContainerType
37
+ csp?: { nonce?: string }
38
+ prepend?: Prepend
39
+ /**
40
+ * Config the `priority` of `prependQueue`. Default is `0`.
41
+ * It's useful if you need to insert style before other style.
42
+ */
43
+ priority?: number
44
+ mark?: string
45
+ }
46
+
47
+ function getMark({ mark }: Options = {}) {
48
+ if (mark) {
49
+ return mark.startsWith('data-') ? mark : `data-${mark}`;
50
+ }
51
+ return MARK_KEY;
52
+ }
53
+
54
+ function getContainer(option: Options) {
55
+ if (option.attachTo) {
56
+ return option.attachTo;
57
+ }
58
+
59
+ const head = document.querySelector('head');
60
+ return head || document.body;
61
+ }
62
+
63
+ function getOrder(prepend?: Prepend): AppendType {
64
+ if (prepend === 'queue') {
65
+ return 'prependQueue';
66
+ }
67
+
68
+ return prepend ? 'prepend' : 'append';
69
+ }
70
+
71
+ /**
72
+ * Find style which inject by rc-util
73
+ */
74
+ function findStyles(container: ContainerType) {
75
+ return [...(containerCache.get(container) || container).children].filter(
76
+ node => node.tagName === 'STYLE',
77
+ ) as HTMLStyleElement[];
78
+ }
79
+
80
+ export function injectCSS(css: string, option: Options = {}) {
81
+ const { csp, prepend, priority = 0 } = option;
82
+ const mergedOrder = getOrder(prepend);
83
+ const isPrependQueue = mergedOrder === 'prependQueue';
84
+
85
+ const styleNode = document.createElement('style');
86
+ styleNode.setAttribute(APPEND_ORDER, mergedOrder);
87
+
88
+ if (isPrependQueue && priority) {
89
+ styleNode.setAttribute(APPEND_PRIORITY, `${priority}`);
90
+ }
91
+
92
+ if (csp?.nonce) {
93
+ styleNode.nonce = csp?.nonce;
94
+ }
95
+ styleNode.innerHTML = css;
96
+
97
+ const container = getContainer(option);
98
+ const { firstChild } = container as any;
99
+
100
+ if (prepend) {
101
+ // If is queue `prepend`, it will prepend first style and then append rest style
102
+ if (isPrependQueue) {
103
+ const existStyle = findStyles(container).filter((node) => {
104
+ // Ignore style which not injected by rc-util with prepend
105
+ // @ts-ignore
106
+ if (!['prepend', 'prependQueue'].includes(node.getAttribute(APPEND_ORDER))) {
107
+ return false;
108
+ }
109
+
110
+ // Ignore style which priority less then new style
111
+ const nodePriority = Number(node.getAttribute(APPEND_PRIORITY) || 0);
112
+ return priority >= nodePriority;
113
+ });
114
+
115
+ if (existStyle.length > 0) {
116
+ // @ts-ignore
117
+ container.insertBefore(styleNode, existStyle.at(-1).nextSibling);
118
+
119
+ return styleNode;
120
+ }
121
+ }
122
+
123
+ // Use `insertBefore` as `prepend`
124
+ firstChild.before(styleNode);
125
+ } else {
126
+ container.append(styleNode);
127
+ }
128
+
129
+ return styleNode;
130
+ }
131
+
132
+ function findExistNode(key: string, option: Options = {}) {
133
+ const container = getContainer(option);
134
+
135
+ return findStyles(container).find(node => node.getAttribute(getMark(option)) === key);
136
+ }
137
+
138
+ export function removeCSS(key: string, option: Options = {}) {
139
+ const existNode = findExistNode(key, option);
140
+ if (existNode) {
141
+ // const container = getContainer(option);
142
+ existNode.remove();
143
+ }
144
+ }
145
+
146
+ /**
147
+ * qiankun will inject `appendChild` to insert into other
148
+ */
149
+ function syncRealContainer(container: ContainerType, option: Options) {
150
+ const cachedRealContainer = containerCache.get(container);
151
+
152
+ // Find real container when not cached or cached container removed
153
+ if (!cachedRealContainer || !contains(document, cachedRealContainer)) {
154
+ const placeholderStyle = injectCSS('', option);
155
+ const { parentNode } = placeholderStyle;
156
+ // @ts-ignore
157
+ containerCache.set(container, parentNode);
158
+ placeholderStyle.remove();
159
+ }
160
+ }
161
+
162
+ /**
163
+ * manually clear container cache to avoid global cache in unit testes
164
+ */
165
+ export function clearContainerCache() {
166
+ containerCache.clear();
167
+ }
168
+
169
+ export function updateCSS(css: string, key: string, option: Options = {}) {
170
+ const container = getContainer(option);
171
+
172
+ // Sync real parent
173
+ syncRealContainer(container, option);
174
+
175
+ const existNode = findExistNode(key, option);
176
+
177
+ if (existNode) {
178
+ if (option.csp?.nonce && existNode.nonce !== option.csp?.nonce) {
179
+ existNode.nonce = option.csp?.nonce;
180
+ }
181
+
182
+ if (existNode.innerHTML !== css) {
183
+ existNode.innerHTML = css;
184
+ }
185
+
186
+ return existNode;
187
+ }
188
+
189
+ const newNode = injectCSS(css, option);
190
+ newNode.setAttribute(getMark(option), key);
191
+ return newNode;
192
+ }
@@ -0,0 +1,28 @@
1
+ import type { Editor } from '@tiptap/core';
2
+
3
+ const cache = new Map();
4
+
5
+ export function getEditorContainerDOMSize(editor: Editor): { width: number } {
6
+ const targetNode = editor.options.element as HTMLElement;
7
+
8
+ if (!cache.has('width')) {
9
+ cache.set('width', targetNode.clientWidth);
10
+ }
11
+
12
+ if (cache.has('width') && cache.get('width') <= 0) {
13
+ cache.set('width', targetNode.clientWidth);
14
+ }
15
+
16
+ const config = { attributes: true, childList: true, subtree: true };
17
+ const callback = function () {
18
+ cache.set('width', targetNode.clientWidth);
19
+ };
20
+ const observer = new MutationObserver(callback);
21
+ observer.observe(targetNode, config);
22
+
23
+ editor.on('destroy', () => {
24
+ observer.disconnect();
25
+ });
26
+
27
+ return { width: cache.get('width') };
28
+ }
@@ -0,0 +1,112 @@
1
+ /* eslint-disable unicorn/prefer-add-event-listener */
2
+ /* eslint-disable @typescript-eslint/ban-ts-comment */
3
+ /**
4
+ * 获取文件名
5
+ *
6
+ * @example
7
+ * > extractFilename('https://gitlab.com/images/logo-full.png')
8
+ * < 'logo-full'
9
+ *
10
+ * @param {string} src The URL to extract filename from
11
+ * @returns {string}
12
+ */
13
+ export function extractFilename(src: any) {
14
+ return src.replace(/^.*\/|\..+$/g, '');
15
+ }
16
+
17
+ /**
18
+ * extractFileExtension
19
+ * @param {string} fileName The file name to extract extension from
20
+ * @returns {string}
21
+ */
22
+ export function extractFileExtension(fileName: any) {
23
+ return fileName.split('.').pop();
24
+ }
25
+
26
+ export function normalizeFileSize(size: any) {
27
+ if (size < 1024) {
28
+ return `${size} Byte`;
29
+ }
30
+ if (size < 1024 * 1024) {
31
+ return `${(size / 1024).toFixed(2)} KB`;
32
+ }
33
+ return `${(size / 1024 / 1024).toFixed(2)} MB`;
34
+ }
35
+
36
+ export type FileType = 'image' | 'audio' | 'video' | 'pdf' | 'word' | 'excel' | 'ppt' | 'file';
37
+
38
+ export function normalizeFileType(fileType: any): FileType {
39
+ if (!fileType)
40
+ return 'file';
41
+
42
+ if (fileType === 'application/pdf')
43
+ return 'pdf';
44
+
45
+ if (fileType.startsWith('application/') && ['.document', 'word'].some(type => fileType.includes(type)))
46
+ return 'word';
47
+
48
+ if (fileType.startsWith('application/') && ['presentation'].some(type => fileType.includes(type)))
49
+ return 'excel';
50
+
51
+ if (fileType.startsWith('application/') && ['sheet'].some(type => fileType.includes(type)))
52
+ return 'ppt';
53
+
54
+ if (fileType.startsWith('image')) {
55
+ return 'image';
56
+ }
57
+
58
+ if (fileType.startsWith('audio')) {
59
+ return 'audio';
60
+ }
61
+
62
+ if (fileType.startsWith('video')) {
63
+ return 'video';
64
+ }
65
+
66
+ return 'file';
67
+ }
68
+
69
+ export function readImageAsBase64(file: File): Promise<{ alt: string, src: string }> {
70
+ return new Promise((resolve) => {
71
+ const reader = new FileReader();
72
+ reader.addEventListener(
73
+ 'load',
74
+ () => {
75
+ resolve({
76
+ alt: file.name,
77
+ src: reader.result as string,
78
+ });
79
+ },
80
+ false,
81
+ );
82
+ reader.readAsDataURL(file);
83
+ });
84
+ }
85
+
86
+ export function getImageWidthHeight(url: string): Promise<{ width: number | string, height: number | string }> {
87
+ return new Promise((resolve) => {
88
+ const img = document.createElement('img');
89
+ img.addEventListener('load', () => {
90
+ resolve({ width: img.width, height: img.height });
91
+ });
92
+ img.onerror = () => {
93
+ resolve({ width: 'auto', height: 'auto' });
94
+ };
95
+ img.src = url;
96
+ });
97
+ }
98
+
99
+ export const FILE_CHUNK_SIZE = 2 * 1024 * 1024;
100
+
101
+ export function dataURLtoFile(dataurl: string, filename: string) {
102
+ const arr = dataurl.split(',');
103
+ // @ts-expect-error
104
+ const mime = arr[0].match(/:(.*?);/)[1];
105
+ const bstr = atob(arr[arr.length - 1]);
106
+ let n = bstr.length;
107
+ const u8arr = new Uint8Array(n);
108
+ while (n--) {
109
+ u8arr[n] = bstr.charCodeAt(n);
110
+ }
111
+ return new File([u8arr], filename, { type: mime });
112
+ }
@@ -0,0 +1,41 @@
1
+ import type { Editor } from '@tiptap/react';
2
+
3
+ export function getRenderContainer(editor: Editor, nodeType: string) {
4
+ const {
5
+ view,
6
+ state: {
7
+ selection: { from },
8
+ },
9
+ } = editor;
10
+
11
+ const elements = document.querySelectorAll('.has-focus');
12
+ const elementCount = elements.length;
13
+ const innermostNode = elements[elementCount - 1];
14
+ const element = innermostNode as any;
15
+
16
+ if (
17
+ (element && element.dataset.type && element.dataset.type === nodeType)
18
+ || (element && element.classList && element.classList.contains(nodeType))
19
+ ) {
20
+ return element;
21
+ }
22
+
23
+ const node = view.domAtPos(from).node as HTMLElement;
24
+ let container: any = node;
25
+
26
+ if (!container.tagName) {
27
+ container = node.parentElement;
28
+ }
29
+
30
+ while (
31
+ container
32
+ && !(container.dataset.type && container.dataset.type === nodeType)
33
+ && !container.classList.contains(nodeType)
34
+ ) {
35
+ container = container.parentElement;
36
+ }
37
+
38
+ return container;
39
+ }
40
+
41
+ export default getRenderContainer;
@@ -0,0 +1,99 @@
1
+ /* eslint-disable no-restricted-syntax */
2
+ import type { Command, Editor } from '@tiptap/core';
3
+ import { isList } from '@tiptap/core';
4
+ import type { Transaction } from '@tiptap/pm/state';
5
+ import { AllSelection, TextSelection } from '@tiptap/pm/state';
6
+
7
+ export const enum IndentProps {
8
+ max = 7,
9
+ min = 0,
10
+
11
+ more = 1,
12
+ less = -1,
13
+ }
14
+
15
+ export function clamp(val: number, min: number, max: number): number {
16
+ if (val < min) {
17
+ return min;
18
+ }
19
+ if (val > max) {
20
+ return max;
21
+ }
22
+ return val;
23
+ }
24
+ function updateIndentLevel(
25
+ tr: Transaction,
26
+ delta: number,
27
+ types: string[],
28
+ editor: Editor,
29
+ ): Transaction {
30
+ const { doc, selection } = tr;
31
+
32
+ if (!doc || !selection) {
33
+ return tr;
34
+ }
35
+
36
+ if (!(selection instanceof TextSelection || selection instanceof AllSelection)) {
37
+ return tr;
38
+ }
39
+
40
+ const { from, to } = selection;
41
+
42
+ doc.nodesBetween(from, to, (node, pos) => {
43
+ const nodeType = node.type;
44
+
45
+ if (types.includes(nodeType.name)) {
46
+ tr = setNodeIndentMarkup(tr, pos, delta);
47
+ return false;
48
+ } else if (isList(node.type.name, editor.extensionManager.extensions)) {
49
+ return false;
50
+ }
51
+ return true;
52
+ });
53
+
54
+ return tr;
55
+ }
56
+
57
+ export function setNodeIndentMarkup(tr: Transaction, pos: number, delta: number): Transaction {
58
+ if (!tr.doc) {
59
+ return tr;
60
+ }
61
+
62
+ const node = tr.doc.nodeAt(pos);
63
+ if (!node) {
64
+ return tr;
65
+ }
66
+
67
+ const minIndent = IndentProps.min;
68
+ const maxIndent = IndentProps.max;
69
+
70
+ const indent = clamp((node.attrs.indent || 0) + delta, minIndent, maxIndent);
71
+
72
+ if (indent === node.attrs.indent) {
73
+ return tr;
74
+ }
75
+
76
+ const nodeAttrs = {
77
+ ...node.attrs,
78
+ indent,
79
+ };
80
+
81
+ return tr.setNodeMarkup(pos, node.type, nodeAttrs, node.marks);
82
+ }
83
+
84
+ export function createIndentCommand({ delta, types }: { delta: number, types: string[] }): Command {
85
+ return ({ state, dispatch, editor }) => {
86
+ const { selection } = state;
87
+ let { tr } = state;
88
+ tr = tr.setSelection(selection);
89
+ tr = updateIndentLevel(tr, delta, types, editor);
90
+
91
+ if (tr.docChanged) {
92
+ if (dispatch)
93
+ dispatch(tr);
94
+ return true;
95
+ }
96
+
97
+ return false;
98
+ };
99
+ }
@@ -0,0 +1,57 @@
1
+ interface HttpRequestHeadersInterfaceMock {
2
+ [id: string]: string | string[] | undefined
3
+ }
4
+
5
+ interface HttpRequestInterfaceMock {
6
+ headers: HttpRequestHeadersInterfaceMock
7
+ [id: string]: any
8
+ }
9
+
10
+ export interface IsMobileOptions {
11
+ ua?: string | HttpRequestInterfaceMock
12
+ tablet?: boolean
13
+ featureDetect?: boolean
14
+ }
15
+
16
+ const mobileRE
17
+ = /(android|bb\d+|meego).+mobile|armv7l|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|hiptop|iemobile|ip(hone|od)|iris|kindle|lge |maemo|midp|mmp|mobile.+firefox|netfront|opera m(ob|in)i|palm( os)?|phone|p(ixi|re)\/|plucker|pocket|psp|series[46]0|samsungbrowser.*mobile|symbian|treo|up\.(browser|link)|vodafone|wap|windows (ce|phone)|xda|xiino/i;
18
+ const notMobileRE = /CrOS/;
19
+ const tabletRE = /android|ipad|playbook|silk/i;
20
+
21
+ /**
22
+ * Determines if the current device is a mobile or tablet device.
23
+ * @param opts - Options for the detection.
24
+ * @returns `true` if the device is mobile or tablet, `false` otherwise.
25
+ */
26
+ export function isMobile(opts: IsMobileOptions = {}): boolean {
27
+ let ua = opts.ua || (typeof navigator !== 'undefined' && navigator.userAgent);
28
+
29
+ if (ua && typeof ua === 'object' && ua.headers && typeof ua.headers['user-agent'] === 'string') {
30
+ ua = ua.headers['user-agent'];
31
+ }
32
+
33
+ if (typeof ua !== 'string') {
34
+ return false;
35
+ }
36
+
37
+ if (mobileRE.test(ua) && !notMobileRE.test(ua)) {
38
+ return true;
39
+ }
40
+
41
+ if (opts.tablet && tabletRE.test(ua)) {
42
+ return true;
43
+ }
44
+
45
+ if (
46
+ opts.tablet
47
+ && opts.featureDetect
48
+ && navigator
49
+ && navigator.maxTouchPoints > 1
50
+ && ua.includes('Macintosh')
51
+ && ua.includes('Safari')
52
+ ) {
53
+ return true;
54
+ }
55
+
56
+ return false;
57
+ }
@@ -0,0 +1,18 @@
1
+ export function safeJSONParse(str: any, defaultValue = {}) {
2
+ if (typeof str === 'object')
3
+ return str;
4
+
5
+ try {
6
+ return JSON.parse(str);
7
+ } catch {
8
+ return defaultValue;
9
+ }
10
+ }
11
+
12
+ export function safeJSONStringify(obj: any, defaultValue = '{}') {
13
+ try {
14
+ return JSON.stringify(obj);
15
+ } catch {
16
+ return defaultValue;
17
+ }
18
+ }