@seafile/seafile-editor 0.4.7 → 1.0.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 (443) hide show
  1. package/.babelrc +5 -0
  2. package/.eslintignore +19 -0
  3. package/.eslintrc.json +52 -0
  4. package/.github/workflows/nodejs.yml +25 -0
  5. package/.vscode/settings.json +5 -0
  6. package/config/config.common.js +157 -0
  7. package/config/config.dev.js +53 -0
  8. package/config/jest/cssTransform.js +16 -0
  9. package/config/jest/fileTransform.js +40 -0
  10. package/config/utils/theme.css.js +103 -0
  11. package/config.js +61 -0
  12. package/dev-server.js +145 -0
  13. package/jest.config.js +16 -0
  14. package/package.json +81 -132
  15. package/public/locales/en/seafile-editor.json +145 -148
  16. package/public/locales/zh-CN/seafile-editor.json +123 -135
  17. package/public/media/seafile-ui.css +6 -11166
  18. package/scripts/ejs.js +16 -0
  19. package/site/_i18n/index.js +35 -0
  20. package/site/api/index.js +212 -0
  21. package/site/app.js +21 -0
  22. package/site/assets/css/app.css +47 -0
  23. package/site/assets/css/seafile-editor.css +33 -0
  24. package/site/commons/loading/index.js +14 -0
  25. package/site/commons/loading/style.css +49 -0
  26. package/site/commons/switch/index.css +14 -0
  27. package/site/commons/switch/index.js +37 -0
  28. package/site/commons/toast/alert.js +106 -0
  29. package/site/commons/toast/index.js +5 -0
  30. package/site/commons/toast/toast.js +195 -0
  31. package/site/commons/toast/toastManager.js +138 -0
  32. package/site/commons/toast/toaster.js +64 -0
  33. package/site/index.html +23 -0
  34. package/site/index.js +21 -0
  35. package/site/pages/home.js +21 -0
  36. package/site/pages/plain-markdown-view.js +6 -0
  37. package/site/pages/rich-seafile-editor.js +72 -0
  38. package/site/pages/seafile-editor.js +39 -0
  39. package/site/pages/seafile-viewer.js +45 -0
  40. package/site/pages/simple-editor.js +44 -0
  41. package/site/pages/wiki-viewer.js +7 -0
  42. package/site/setting.js +36 -0
  43. package/site/setting.local.dist.js +23 -0
  44. package/src/assets/css/slate-editor.css +57 -0
  45. package/src/constants/event-types.js +11 -0
  46. package/src/constants/hot-keys.js +47 -0
  47. package/src/containers/article-info/index.js +49 -0
  48. package/src/containers/article-info/style.css +52 -0
  49. package/src/containers/custom/get-event-transfer.js +28 -0
  50. package/{dist → src}/containers/custom/set-event-transfer.js +13 -11
  51. package/src/containers/hotkeys-helper/classify-hotkeys.js +30 -0
  52. package/src/containers/hotkeys-helper/index.js +43 -0
  53. package/{dist/assets/css/user-help.css → src/containers/hotkeys-helper/style.css} +37 -29
  54. package/src/containers/loading/index.js +14 -0
  55. package/src/containers/loading/style.css +49 -0
  56. package/src/containers/outline/index.js +75 -0
  57. package/src/containers/outline/outline-item.js +28 -0
  58. package/src/containers/outline/style.css +45 -0
  59. package/src/editors/plain-markdown-editor/code-mirror.css +8 -0
  60. package/src/editors/plain-markdown-editor/code-mirror.js +73 -0
  61. package/src/editors/plain-markdown-editor/index.js +136 -0
  62. package/{dist/assets/editor/plain-editor.css → src/editors/plain-markdown-editor/style.css} +12 -4
  63. package/src/editors/simple-slate-editor /index.js +71 -0
  64. package/src/editors/simple-slate-editor /with-props-editor.js +15 -0
  65. package/src/editors/slate-editor/editor-help/index.js +46 -0
  66. package/src/editors/slate-editor/editor-help/style.css +15 -0
  67. package/src/editors/slate-editor/index.js +77 -0
  68. package/src/editors/slate-editor/with-props-editor.js +15 -0
  69. package/src/editors/slate-viewer/index.js +35 -0
  70. package/src/editors/slate-viewer/style.css +40 -0
  71. package/src/extension/commons/element-popover/index.js +52 -0
  72. package/src/extension/commons/index.js +10 -0
  73. package/src/extension/commons/menu/index.js +9 -0
  74. package/src/extension/commons/menu/menu-drop-down.js +110 -0
  75. package/src/extension/commons/menu/menu-group.js +20 -0
  76. package/src/extension/commons/menu/menu-item.js +60 -0
  77. package/src/extension/commons/menu/menu.css +120 -0
  78. package/src/extension/commons/tooltip/index.css +21 -0
  79. package/src/extension/commons/tooltip/index.js +26 -0
  80. package/src/extension/constants/element-types.js +23 -0
  81. package/src/extension/constants/index.js +37 -0
  82. package/src/extension/constants/keyboard.js +29 -0
  83. package/src/extension/constants/menus-config.js +149 -0
  84. package/src/extension/core/index.js +3 -0
  85. package/src/extension/core/queries/index.js +410 -0
  86. package/src/extension/core/transforms/focus-editor.js +12 -0
  87. package/src/extension/core/transforms/index.js +4 -0
  88. package/src/extension/core/transforms/move-children.js +23 -0
  89. package/src/extension/core/transforms/remove-node-children.js +8 -0
  90. package/src/extension/core/transforms/replace-node-children.js +14 -0
  91. package/src/extension/core/utils/index.js +76 -0
  92. package/src/extension/editor.js +14 -0
  93. package/src/extension/index.js +15 -0
  94. package/src/extension/plugins/blockquote/helpers.js +65 -0
  95. package/src/extension/plugins/blockquote/index.js +13 -0
  96. package/src/extension/plugins/blockquote/menu/index.js +31 -0
  97. package/src/extension/plugins/blockquote/plugin.js +64 -0
  98. package/src/extension/plugins/blockquote/render-elem.js +9 -0
  99. package/src/extension/plugins/check-list/helper.js +34 -0
  100. package/src/extension/plugins/check-list/index.js +14 -0
  101. package/src/extension/plugins/check-list/menu/index.js +44 -0
  102. package/src/extension/plugins/check-list/plugin.js +46 -0
  103. package/src/extension/plugins/check-list/render-elem.js +23 -0
  104. package/src/extension/plugins/clear-format/helpers.js +48 -0
  105. package/src/extension/plugins/clear-format/menu/index.js +39 -0
  106. package/src/extension/plugins/code-block/helpers.js +107 -0
  107. package/src/extension/plugins/code-block/index.js +14 -0
  108. package/src/extension/plugins/code-block/menu/index.js +40 -0
  109. package/src/extension/plugins/code-block/plugin.js +210 -0
  110. package/src/extension/plugins/code-block/render-elem/constant.js +17 -0
  111. package/src/extension/plugins/code-block/render-elem/index.js +55 -0
  112. package/src/extension/plugins/code-block/render-elem/language-selector.js +38 -0
  113. package/src/extension/plugins/code-block/render-elem/style.css +27 -0
  114. package/src/extension/plugins/formula/formula.css +22 -0
  115. package/src/extension/plugins/formula/helper.js +68 -0
  116. package/src/extension/plugins/formula/index.js +14 -0
  117. package/src/extension/plugins/formula/menu/formula-modal.js +73 -0
  118. package/src/extension/plugins/formula/menu/index.js +63 -0
  119. package/src/extension/plugins/formula/plugin.js +18 -0
  120. package/src/extension/plugins/formula/render-elem.js +40 -0
  121. package/src/extension/plugins/header/helper.js +47 -0
  122. package/src/extension/plugins/header/index.js +14 -0
  123. package/src/extension/plugins/header/menu/index.js +114 -0
  124. package/src/extension/plugins/header/menu/style.css +115 -0
  125. package/src/extension/plugins/header/plugin.js +134 -0
  126. package/src/extension/plugins/header/render-elem.js +31 -0
  127. package/src/extension/plugins/image/helper.js +50 -0
  128. package/src/extension/plugins/image/index.js +14 -0
  129. package/src/extension/plugins/image/menu/image-menu-dialog.js +68 -0
  130. package/src/extension/plugins/image/menu/image-menu-popover.js +76 -0
  131. package/src/extension/plugins/image/menu/index.js +61 -0
  132. package/src/extension/plugins/image/menu/style.css +36 -0
  133. package/src/extension/plugins/image/plugin.js +23 -0
  134. package/src/extension/plugins/image/render-element/image-previewer.js +71 -0
  135. package/src/extension/plugins/image/render-element/index.js +94 -0
  136. package/src/extension/plugins/image/render-element/style.css +60 -0
  137. package/src/extension/plugins/index.js +47 -0
  138. package/src/extension/plugins/link/helper.js +161 -0
  139. package/src/extension/plugins/link/index.js +14 -0
  140. package/src/extension/plugins/link/menu/index.js +82 -0
  141. package/src/extension/plugins/link/menu/link-modal.js +135 -0
  142. package/src/extension/plugins/link/plugin.js +85 -0
  143. package/src/extension/plugins/link/render-elem/index.js +69 -0
  144. package/src/extension/plugins/link/render-elem/link-popover.js +79 -0
  145. package/src/extension/plugins/link/render-elem/style.css +60 -0
  146. package/src/extension/plugins/list/constant.js +3 -0
  147. package/src/extension/plugins/list/helpers.js +53 -0
  148. package/src/extension/plugins/list/index.js +14 -0
  149. package/src/extension/plugins/list/menu/index.js +48 -0
  150. package/src/extension/plugins/list/plugin/index.js +48 -0
  151. package/src/extension/plugins/list/plugin/insert-break-list.js +25 -0
  152. package/src/extension/plugins/list/plugin/insert-fragment-list.js +162 -0
  153. package/src/extension/plugins/list/plugin/normalize-list.js +64 -0
  154. package/src/extension/plugins/list/plugin/on-tab-handle.js +34 -0
  155. package/src/extension/plugins/list/queries/index.js +39 -0
  156. package/src/extension/plugins/list/render-elem/index.js +26 -0
  157. package/src/extension/plugins/list/render-elem/style.css +6 -0
  158. package/src/extension/plugins/list/transforms/index.js +23 -0
  159. package/src/extension/plugins/list/transforms/insert-list-item.js +65 -0
  160. package/src/extension/plugins/list/transforms/move-list-item-down.js +40 -0
  161. package/src/extension/plugins/list/transforms/move-list-item-up.js +105 -0
  162. package/src/extension/plugins/list/transforms/move-list-items-to-list.js +58 -0
  163. package/src/extension/plugins/list/transforms/move-list-items.js +76 -0
  164. package/src/extension/plugins/list/transforms/normalize-list-item.js +92 -0
  165. package/src/extension/plugins/list/transforms/normalize-nested-list.js +31 -0
  166. package/src/extension/plugins/list/transforms/remove-first-list-item.js +11 -0
  167. package/src/extension/plugins/list/transforms/transforms-to-list.js +91 -0
  168. package/src/extension/plugins/list/transforms/unwrap-list.js +44 -0
  169. package/src/extension/plugins/node-id/constants.js +18 -0
  170. package/src/extension/plugins/node-id/helpers.js +61 -0
  171. package/src/extension/plugins/node-id/index.js +7 -0
  172. package/src/extension/plugins/node-id/with-node-id.js +37 -0
  173. package/src/extension/plugins/paragraph/helper.js +9 -0
  174. package/src/extension/plugins/paragraph/index.js +12 -0
  175. package/src/extension/plugins/paragraph/plugin.js +40 -0
  176. package/src/extension/plugins/paragraph/render-elem.js +11 -0
  177. package/src/extension/plugins/table/constant.js +3 -0
  178. package/src/extension/plugins/table/helper.js +175 -0
  179. package/src/extension/plugins/table/index.js +15 -0
  180. package/src/extension/plugins/table/menu/index.js +69 -0
  181. package/src/extension/plugins/table/menu/style.css +32 -0
  182. package/src/extension/plugins/table/menu/table-operator.js +101 -0
  183. package/src/extension/plugins/table/menu/table-size-selector.js +71 -0
  184. package/src/extension/plugins/table/model.js +69 -0
  185. package/src/extension/plugins/table/plugin.js +159 -0
  186. package/src/extension/plugins/table/render-elem/context-menu.js +78 -0
  187. package/src/extension/plugins/table/render-elem/index.js +157 -0
  188. package/src/extension/plugins/table/render-elem/style.css +94 -0
  189. package/src/extension/plugins/table/table-operations.js +162 -0
  190. package/src/extension/plugins/text-style/helpers.js +38 -0
  191. package/src/extension/plugins/text-style/index.js +12 -0
  192. package/src/extension/plugins/text-style/menu/index.js +41 -0
  193. package/src/extension/plugins/text-style/plugin.js +37 -0
  194. package/src/extension/plugins/text-style/render-elem.js +45 -0
  195. package/src/extension/render/render-element.js +82 -0
  196. package/src/extension/render/render-leaf.js +16 -0
  197. package/src/extension/toolbar/header-toolbar/index.js +129 -0
  198. package/src/extension/toolbar/header-toolbar/style.css +41 -0
  199. package/src/extension/toolbar/index.js +5 -0
  200. package/src/extension/toolbar/user-help/shortcut-dialog.js +144 -0
  201. package/src/hooks/use-mathjax.js +44 -0
  202. package/src/hooks/use-scroll-context.js +14 -0
  203. package/src/hooks/use-selection-update.js +14 -0
  204. package/src/index.js +22 -0
  205. package/src/pages/markdown-editor.js +53 -0
  206. package/src/pages/markdown-view.js +36 -0
  207. package/src/pages/rich-markdown-editor.js +95 -0
  208. package/src/pages/simple-editor.js +52 -0
  209. package/src/slate-convert/html-to-slate/constants.js +54 -0
  210. package/src/slate-convert/html-to-slate/helper.js +22 -0
  211. package/src/slate-convert/html-to-slate/index.js +144 -0
  212. package/src/slate-convert/html-to-slate/rules/blockquote.js +16 -0
  213. package/src/slate-convert/html-to-slate/rules/check-list.js +22 -0
  214. package/src/slate-convert/html-to-slate/rules/code-block.js +91 -0
  215. package/src/slate-convert/html-to-slate/rules/header.js +18 -0
  216. package/src/slate-convert/html-to-slate/rules/image.js +18 -0
  217. package/src/slate-convert/html-to-slate/rules/index.js +23 -0
  218. package/src/slate-convert/html-to-slate/rules/link.js +24 -0
  219. package/src/slate-convert/html-to-slate/rules/list.js +52 -0
  220. package/src/slate-convert/html-to-slate/rules/paragraph.js +16 -0
  221. package/src/slate-convert/html-to-slate/rules/table.js +37 -0
  222. package/src/slate-convert/html-to-slate/rules/text.js +63 -0
  223. package/src/slate-convert/index.js +11 -0
  224. package/src/slate-convert/md-to-html/index.js +44 -0
  225. package/src/slate-convert/md-to-html/sanitize-schema.js +41 -0
  226. package/src/slate-convert/md-to-slate/index.js +42 -0
  227. package/src/slate-convert/md-to-slate/transform.js +336 -0
  228. package/src/slate-convert/slate-to-md/index.js +44 -0
  229. package/src/slate-convert/slate-to-md/transform.js +248 -0
  230. package/src/utils/common.js +30 -0
  231. package/src/utils/deserialize-html.js +174 -0
  232. package/src/utils/dom-utils.js +45 -0
  233. package/src/utils/event-bus.js +37 -0
  234. package/src/utils/event-handler.js +36 -0
  235. package/src/utils/object-utils.js +51 -0
  236. package/tests/core/constants/index.js +54 -0
  237. package/tests/core/index.js +11 -0
  238. package/tests/core/jsx/index.js +9 -0
  239. package/tests/core/stub-editor/index.js +26 -0
  240. package/tests/core/utils.js +58 -0
  241. package/tests/extension/plugins/list/insert-block.test.js +40 -0
  242. package/tests/extension/plugins/list/insert-fragment-list.test.js +392 -0
  243. package/tests/extension/plugins/list/toggle-list.test.js +160 -0
  244. package/tests/extension/plugins/text/text.test.js +29 -0
  245. package/tests/extension/plugins/text-style/toggle-text-style.test.js +148 -0
  246. package/tests/slate-convert/html-to-slate/blockquote.test.js +107 -0
  247. package/tests/slate-convert/html-to-slate/code-block.test.js +32 -0
  248. package/tests/slate-convert/html-to-slate/header.test.js +56 -0
  249. package/tests/slate-convert/html-to-slate/link.test.js +77 -0
  250. package/tests/slate-convert/html-to-slate/list.test.js +190 -0
  251. package/tests/slate-convert/html-to-slate/paragraph.test.js +40 -0
  252. package/tests/slate-convert/html-to-slate/text.test.js +131 -0
  253. package/tests/slate-convert/md-to-html/blockquote.test.js +110 -0
  254. package/tests/slate-convert/md-to-html/check-list.test.js +19 -0
  255. package/tests/slate-convert/md-to-html/code-block.test.js +19 -0
  256. package/tests/slate-convert/md-to-html/header.test.js +20 -0
  257. package/tests/slate-convert/md-to-html/image.test.js +17 -0
  258. package/tests/slate-convert/md-to-html/image_reference.test.js +17 -0
  259. package/tests/slate-convert/md-to-html/link.test.js +15 -0
  260. package/tests/slate-convert/md-to-html/link_reference.test.js +15 -0
  261. package/tests/slate-convert/md-to-html/list-ordered.test.js +22 -0
  262. package/tests/slate-convert/md-to-html/list-unordered.test.js +22 -0
  263. package/tests/slate-convert/md-to-html/paragraph.test.js +15 -0
  264. package/tests/slate-convert/md-to-html/table.test.js +29 -0
  265. package/tests/slate-convert/md-to-html/text_base.test.js +14 -0
  266. package/tests/slate-convert/md-to-html/text_bold.test.js +14 -0
  267. package/tests/slate-convert/md-to-html/text_bold_italic.test.js +15 -0
  268. package/tests/slate-convert/md-to-html/text_code.test.js +14 -0
  269. package/tests/slate-convert/md-to-html/text_code_bold.test.js +14 -0
  270. package/tests/slate-convert/md-to-html/text_code_bold_italic.test.js +14 -0
  271. package/tests/slate-convert/md-to-html/text_code_italic.test.js +14 -0
  272. package/tests/slate-convert/md-to-html/text_italic.test.js +14 -0
  273. package/tests/slate-convert/md-to-slate/blockquote.test.js +200 -0
  274. package/tests/slate-convert/md-to-slate/check-list.test.js +34 -0
  275. package/tests/slate-convert/md-to-slate/code-block.test.js +32 -0
  276. package/tests/slate-convert/md-to-slate/header.test.js +49 -0
  277. package/tests/slate-convert/md-to-slate/image.test.js +31 -0
  278. package/tests/slate-convert/md-to-slate/link.test.js +30 -0
  279. package/tests/slate-convert/md-to-slate/list-ordered.test.js +42 -0
  280. package/tests/slate-convert/md-to-slate/list-unordered.test.js +42 -0
  281. package/tests/slate-convert/md-to-slate/paragraph.test.js +19 -0
  282. package/tests/slate-convert/md-to-slate/table.test.js +34 -0
  283. package/tests/slate-convert/md-to-slate/text_base.test.js +19 -0
  284. package/tests/slate-convert/md-to-slate/text_bold.test.js +22 -0
  285. package/tests/slate-convert/md-to-slate/text_bold_italic.test.js +23 -0
  286. package/tests/slate-convert/md-to-slate/text_code.test.js +22 -0
  287. package/tests/slate-convert/md-to-slate/text_code_bold.test.js +23 -0
  288. package/tests/slate-convert/md-to-slate/text_code_bold_italic.test.js +24 -0
  289. package/tests/slate-convert/md-to-slate/text_code_italic.test.js +23 -0
  290. package/tests/slate-convert/md-to-slate/text_italic.test.js +22 -0
  291. package/tests/slate-convert/slate-to-md/blockquote.test.js +416 -0
  292. package/tests/slate-convert/slate-to-md/check-list.test.js +96 -0
  293. package/tests/slate-convert/slate-to-md/code-block.test.js +35 -0
  294. package/tests/slate-convert/slate-to-md/formula.test.js +26 -0
  295. package/tests/slate-convert/slate-to-md/header.test.js +109 -0
  296. package/tests/slate-convert/slate-to-md/image.test.js +98 -0
  297. package/tests/slate-convert/slate-to-md/link.test.js +52 -0
  298. package/tests/slate-convert/slate-to-md/list-ordered.test.js +87 -0
  299. package/tests/slate-convert/slate-to-md/list-unordered.test.js +81 -0
  300. package/tests/slate-convert/slate-to-md/paragraph.test.js +28 -0
  301. package/tests/slate-convert/slate-to-md/table.test.js +56 -0
  302. package/tests/slate-convert/slate-to-md/text_base.test.js +28 -0
  303. package/tests/slate-convert/slate-to-md/text_bold.test.js +36 -0
  304. package/tests/slate-convert/slate-to-md/text_bold_italic.test.js +42 -0
  305. package/tests/slate-convert/slate-to-md/text_code.test.js +31 -0
  306. package/tests/slate-convert/slate-to-md/text_code_bold.test.js +37 -0
  307. package/tests/slate-convert/slate-to-md/text_code_bold_italic.test.js +43 -0
  308. package/tests/slate-convert/slate-to-md/text_code_italic.test.js +37 -0
  309. package/tests/slate-convert/slate-to-md/text_italic.test.js +36 -0
  310. package/README.md +0 -52
  311. package/TODO.md +0 -55
  312. package/dist/assets/css/comment-dialog.css +0 -50
  313. package/dist/assets/css/diff-viewer.css +0 -105
  314. package/dist/assets/css/formula.css +0 -19
  315. package/dist/assets/css/history-viewer.css +0 -104
  316. package/dist/assets/css/image.css +0 -134
  317. package/dist/assets/css/issue-card.css +0 -43
  318. package/dist/assets/css/link.css +0 -7
  319. package/dist/assets/css/markdown-editor.css +0 -12
  320. package/dist/assets/css/markdown-viewer.css +0 -69
  321. package/dist/assets/css/navbar-imgbutton.css +0 -83
  322. package/dist/assets/css/outline.css +0 -29
  323. package/dist/assets/css/table.css +0 -57
  324. package/dist/assets/css/textlink-hovermenu.css +0 -47
  325. package/dist/assets/css/topbar.css +0 -350
  326. package/dist/assets/css/tree-view.css +0 -67
  327. package/dist/assets/editor/seatable-editor.css +0 -77
  328. package/dist/assets/editor/simple-editor.css +0 -77
  329. package/dist/components/click-outside.js +0 -46
  330. package/dist/components/context-menu.js +0 -97
  331. package/dist/components/dialogs/add-formula-dialog.js +0 -116
  332. package/dist/components/dialogs/add-image-dialog.js +0 -69
  333. package/dist/components/dialogs/add-link-dialog.js +0 -133
  334. package/dist/components/dialogs/comment-dialog.js +0 -97
  335. package/dist/components/dialogs/shortcut-dialog.js +0 -131
  336. package/dist/components/error-boundary.js +0 -28
  337. package/dist/components/load-script/index.js +0 -69
  338. package/dist/components/loading.js +0 -25
  339. package/dist/components/markdown-lint.js +0 -72
  340. package/dist/components/menu/index.js +0 -4
  341. package/dist/components/menu/item.js +0 -49
  342. package/dist/components/menu/menu.js +0 -38
  343. package/dist/components/menu/style.css +0 -42
  344. package/dist/components/modal-portal.js +0 -38
  345. package/dist/components/outline/index.js +0 -92
  346. package/dist/components/outline/outline-item.js +0 -58
  347. package/dist/components/select/_option.js +0 -44
  348. package/dist/components/select/field-setting.js +0 -106
  349. package/dist/components/select/index.js +0 -149
  350. package/dist/components/select/style.css +0 -144
  351. package/dist/components/svg-icons/check-mark-icon.js +0 -14
  352. package/dist/components/svg-icons/column-icon.js +0 -17
  353. package/dist/components/svg-icons/text-icon.js +0 -34
  354. package/dist/components/text-link-hover-menu/index.js +0 -123
  355. package/dist/components/toast/alert.js +0 -138
  356. package/dist/components/toast/index.js +0 -3
  357. package/dist/components/toast/toast.js +0 -159
  358. package/dist/components/toast/toastManager.js +0 -139
  359. package/dist/components/toast/toaster.js +0 -65
  360. package/dist/components/toolbar/header-list.js +0 -114
  361. package/dist/components/toolbar/help-group.js +0 -33
  362. package/dist/components/toolbar/index.js +0 -4
  363. package/dist/components/toolbar/insert-image.js +0 -106
  364. package/dist/components/toolbar/insert-table.js +0 -136
  365. package/dist/components/toolbar/table-group.js +0 -74
  366. package/dist/components/toolbar/toolbar.js +0 -317
  367. package/dist/components/toolbar/widgets/button-group.js +0 -24
  368. package/dist/components/toolbar/widgets/button-item.js +0 -129
  369. package/dist/components/toolbar/widgets/drop-list.js +0 -88
  370. package/dist/components/user-help/index.js +0 -179
  371. package/dist/constants/cell-types.js +0 -29
  372. package/dist/constants/column.js +0 -4
  373. package/dist/containers/code-highlight-package.js +0 -14
  374. package/dist/containers/controller/block-element-controller.js +0 -375
  375. package/dist/containers/controller/index.js +0 -5
  376. package/dist/containers/controller/inline-element-controller.js +0 -134
  377. package/dist/containers/controller/normalize-controller.js +0 -95
  378. package/dist/containers/controller/shortcut-controller.js +0 -385
  379. package/dist/containers/controller/void-element-controller.js +0 -9
  380. package/dist/containers/custom/custom.js +0 -18
  381. package/dist/containers/custom/get-event-transfer.js +0 -33
  382. package/dist/containers/custom/getNodesByTypeAtRange.js +0 -57
  383. package/dist/containers/custom/insertNodes.js +0 -120
  384. package/dist/containers/custom/is-empty-paragraph.js +0 -9
  385. package/dist/containers/custom/split-nodes-at-point.js +0 -136
  386. package/dist/containers/custom/unwrap-node-by-type-at-range.js +0 -70
  387. package/dist/containers/editor-context.js +0 -85
  388. package/dist/containers/editor-utils/block-element-utils/blockquote-utils.js +0 -80
  389. package/dist/containers/editor-utils/block-element-utils/code-utils.js +0 -145
  390. package/dist/containers/editor-utils/block-element-utils/formula-utils.js +0 -51
  391. package/dist/containers/editor-utils/block-element-utils/index.js +0 -31
  392. package/dist/containers/editor-utils/block-element-utils/list-utils.js +0 -395
  393. package/dist/containers/editor-utils/block-element-utils/table-utils.js +0 -412
  394. package/dist/containers/editor-utils/clear-format-utils.js +0 -84
  395. package/dist/containers/editor-utils/common-editor-utils.js +0 -492
  396. package/dist/containers/editor-utils/inline-element-utils/index.js +0 -95
  397. package/dist/containers/editor-utils/mark-utils.js +0 -20
  398. package/dist/containers/editor-utils/range-utils.js +0 -7
  399. package/dist/containers/editor-utils/selection-utils.js +0 -30
  400. package/dist/containers/editor-utils/text-utils.js +0 -117
  401. package/dist/containers/editor-widgets/check-list-item.js +0 -53
  402. package/dist/containers/editor-widgets/code-block.js +0 -128
  403. package/dist/containers/editor-widgets/column.js +0 -100
  404. package/dist/containers/editor-widgets/formula.js +0 -67
  405. package/dist/containers/editor-widgets/image.js +0 -237
  406. package/dist/containers/editor-widgets/link.js +0 -9
  407. package/dist/containers/editor-widgets/table.js +0 -144
  408. package/dist/containers/element-model/blockquote.js +0 -13
  409. package/dist/containers/element-model/column.js +0 -19
  410. package/dist/containers/element-model/image.js +0 -16
  411. package/dist/containers/element-model/link.js +0 -16
  412. package/dist/containers/element-model/table.js +0 -57
  413. package/dist/containers/element-model/text.js +0 -10
  414. package/dist/containers/render-utils/common-utils.js +0 -80
  415. package/dist/containers/render-utils/editor-utils.js +0 -133
  416. package/dist/containers/render-utils/viewer-utils.js +0 -198
  417. package/dist/containers/viewer-widgets/viewer-formula/index.js +0 -54
  418. package/dist/containers/viewer-widgets/viewer-image/index.js +0 -70
  419. package/dist/containers/viewer-widgets/viewer-image/viewer-image.css +0 -3
  420. package/dist/editors/index.js +0 -78
  421. package/dist/editors/markdown-editor.js +0 -280
  422. package/dist/editors/plain-markdown-editor.js +0 -285
  423. package/dist/editors/seatable-editor.js +0 -210
  424. package/dist/editors/simple-editor.js +0 -200
  425. package/dist/index.js +0 -11
  426. package/dist/utils/deserialize-html.js +0 -260
  427. package/dist/utils/diff/compare-strings.js +0 -35
  428. package/dist/utils/diff/diff.js +0 -769
  429. package/dist/utils/diff/index.js +0 -2
  430. package/dist/utils/seafile-markdown2html.js +0 -52
  431. package/dist/utils/slate2markdown/deserialize.js +0 -588
  432. package/dist/utils/slate2markdown/index.js +0 -3
  433. package/dist/utils/slate2markdown/serialize.js +0 -366
  434. package/dist/utils/utils.js +0 -68
  435. package/dist/viewer/diff-viewer.js +0 -85
  436. package/dist/viewer/index.js +0 -4
  437. package/dist/viewer/markdown-viewer.js +0 -121
  438. package/dist/viewer/seatable-viewer.js +0 -63
  439. package/dist/viewer/slate-viewer.js +0 -71
  440. package/public/index.html +0 -45
  441. package/public/manifest.json +0 -15
  442. package/public/media/seafile-logo.png +0 -0
  443. /package/{dist/assets/css/keyboard-shortcuts.css → src/extension/toolbar/user-help/style.css} +0 -0
@@ -0,0 +1,47 @@
1
+ import { Editor, Transforms } from 'slate';
2
+ import { getNodeType, getParentNode, isTextNode } from '../../core/queries';
3
+ import { ELementTypes } from '../../constants';
4
+ import { focusEditor } from '../../core';
5
+
6
+ export const isMenuDisabled = (editor, readonly = false) => {
7
+ if (readonly) return true;
8
+ if (!editor.selection) return true;
9
+ const [match] = Editor.nodes(
10
+ editor,
11
+ {
12
+ match(node, path) {
13
+ let type = getNodeType(node);
14
+ if (!type && isTextNode(node) && node.id) {
15
+ const parentNode = getParentNode(node, node.id);
16
+ type = getNodeType(parentNode);
17
+ }
18
+ if (type === ELementTypes.PARAGRAPH) return true;
19
+ if (type.startsWith(ELementTypes.HEADER)) return true;
20
+ return false;
21
+ },
22
+ universal: true,
23
+ mode: 'highest',
24
+ },
25
+ );
26
+ return !match;
27
+ };
28
+
29
+ export const getHeaderType = (editor) => {
30
+ const [match] = Editor.nodes(editor, {
31
+ match: node => {
32
+ const nodeType = getNodeType(node);
33
+ if (nodeType.includes(ELementTypes.HEADER)) return true;
34
+ return false;
35
+ },
36
+ universal: true,
37
+ });
38
+ if (!match) return;
39
+ const [matchedNode] = match;
40
+ return getNodeType(matchedNode);
41
+ };
42
+
43
+ export const setHeaderType = (editor, type) => {
44
+ if (!type) return;
45
+ Transforms.setNodes(editor, { type });
46
+ focusEditor(editor);
47
+ };
@@ -0,0 +1,14 @@
1
+ import { ELementTypes } from '../../constants';
2
+ import { renderHeader } from './render-elem';
3
+ import withHeader from './plugin';
4
+ import HeaderMenu from './menu';
5
+
6
+ const HeaderPlugin = {
7
+ type: ELementTypes.HEADER,
8
+ nodeType: 'element',
9
+ editorMenus: [HeaderMenu],
10
+ editorPlugin: withHeader,
11
+ renderElements: [renderHeader],
12
+ };
13
+
14
+ export default HeaderPlugin;
@@ -0,0 +1,114 @@
1
+ import React, { useState, useRef, Fragment, useMemo, useCallback } from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import classnames from 'classnames';
4
+ import { useTranslation } from 'react-i18next';
5
+ import { getHeaderType, isMenuDisabled, setHeaderType } from '../helper';
6
+ import Tooltip from '../../../commons/tooltip';
7
+ import { MAC_HOTKEYS_TIP_HEADER, WIN_HOTKEYS_EVENT_HEADER } from '../../../constants/keyboard';
8
+ import { ELementTypes, HEADERS, HEADER_TITLE_MAP } from '../../../constants';
9
+
10
+ import './style.css';
11
+
12
+ const propTypes = {
13
+ editor: PropTypes.object.isRequired,
14
+ readonly: PropTypes.bool.isRequired,
15
+ className: PropTypes.string,
16
+ };
17
+
18
+ const headerPopoverShowerList = [ELementTypes.PARAGRAPH, ...HEADERS];
19
+
20
+ const HeaderMenu = ({ editor, readonly, isRichEditor }) => {
21
+ const [isShowHeaderPopover, setIsShowHeaderPopover] = useState(false);
22
+ const headerPopoverRef = useRef();
23
+ const { t } = useTranslation();
24
+
25
+ const currentHeaderType = getHeaderType(editor);
26
+ // eslint-disable-next-line react-hooks/exhaustive-deps
27
+ const isDisabled = useMemo(() => isMenuDisabled(editor, readonly), [editor.selection, readonly]);
28
+
29
+ const onHideHeaderMenu = useCallback((e) => {
30
+ const menu = headerPopoverRef.current;
31
+ const clickIsInMenu = menu && menu.contains(e.target) && menu !== e.target;
32
+
33
+ if (clickIsInMenu) return;
34
+ setIsShowHeaderPopover(false);
35
+ unregisterEventHandler();
36
+ // eslint-disable-next-line react-hooks/exhaustive-deps
37
+ }, []);
38
+
39
+ const registerEventHandler = useCallback(() => {
40
+ document.addEventListener('click', onHideHeaderMenu, true);
41
+ // eslint-disable-next-line react-hooks/exhaustive-deps
42
+ }, []);
43
+
44
+ const unregisterEventHandler = useCallback(() => {
45
+ document.removeEventListener('click', onHideHeaderMenu, true);
46
+ // eslint-disable-next-line react-hooks/exhaustive-deps
47
+ }, []);
48
+
49
+
50
+ const onToggleClick = useCallback((event) => {
51
+ event.stopPropagation();
52
+ event.nativeEvent.stopImmediatePropagation();
53
+ setIsShowHeaderPopover((isShowHeaderPopover) => {
54
+ const newIsShowHeaderPopover = !isShowHeaderPopover;
55
+ newIsShowHeaderPopover ? registerEventHandler() : unregisterEventHandler();
56
+ return newIsShowHeaderPopover;
57
+ });
58
+ }, [registerEventHandler, unregisterEventHandler]);
59
+
60
+ const onMouseDown = useCallback((type) => {
61
+ setHeaderType(editor, type);
62
+ setIsShowHeaderPopover(false);
63
+ unregisterEventHandler();
64
+ }, [editor, unregisterEventHandler]);
65
+
66
+ const getToolTip = (type) => {
67
+ // chrome in Mac: "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/116.0.0.0 Safari/537.36"
68
+ const isMac = window.navigator.userAgent.indexOf('Macintosh') !== -1;
69
+ return isMac ? MAC_HOTKEYS_TIP_HEADER[type] : WIN_HOTKEYS_EVENT_HEADER[type];
70
+ };
71
+ return (
72
+ <div className={classnames('header-menu', { 'header-popover-showed': isShowHeaderPopover, 'header-toggle-disabled': isDisabled })}>
73
+ <div
74
+ className={classnames('header-toggle', { 'header-toggle-disabled': isDisabled, 'header-popover-showed': isShowHeaderPopover })}
75
+ onClick={isDisabled ? void 0 : onToggleClick}
76
+ >
77
+ <span className='active'>{t(HEADER_TITLE_MAP[currentHeaderType ?? ELementTypes.PARAGRAPH])}</span>
78
+ {!isDisabled && (<span className={`sf-font sdoc-${isShowHeaderPopover ? 'caret-up' : 'drop-down'}`}></span>)}
79
+ </div>
80
+ {
81
+ isShowHeaderPopover && (
82
+ <div ref={headerPopoverRef} className='header-popover '>
83
+ {headerPopoverShowerList.map((item, index) => {
84
+ const id = `${item}-${index}`;
85
+ const isSelected = currentHeaderType === item;
86
+ return (
87
+ <Fragment key={index}>
88
+ <div
89
+ id={id}
90
+ className={classnames('sf-dropdown-menu-item', { 'position-relative': isSelected })}
91
+ onClick={() => onMouseDown(item)}
92
+ >
93
+ {isSelected && (<i className="sf-font sf-check-mark"></i>)}
94
+ <span>{t(HEADER_TITLE_MAP[item])}</span>
95
+ </div>
96
+ <Tooltip
97
+ target={id}
98
+ placement="right"
99
+ >
100
+ {getToolTip(item)}
101
+ </Tooltip>
102
+ </Fragment>
103
+ );
104
+ })}
105
+ </div>
106
+ )
107
+ }
108
+ </div >
109
+ );
110
+ };
111
+
112
+ HeaderMenu.propTypes = propTypes;
113
+
114
+ export default HeaderMenu;
@@ -0,0 +1,115 @@
1
+ .header-menu {
2
+ display: flex;
3
+ height: 100%;
4
+ width: 140px;
5
+ padding: 6px 8px;
6
+ border-right: 1px solid #e5e6e8;
7
+ position: relative;
8
+ align-items: center;
9
+ }
10
+
11
+ .header-menu::after {
12
+ content: '';
13
+ position: absolute;
14
+ top: 12px;
15
+ right: 12px;
16
+ border-top: 6px solid #222428;
17
+ border-bottom: 6px solid transparent;
18
+ border-left: 6px solid transparent;
19
+ border-right: 6px solid transparent;
20
+ }
21
+
22
+ .header-menu.header-toggle-disabled::after {
23
+ content: '';
24
+ position: absolute;
25
+ top: 12px;
26
+ right: 12px;
27
+ border-top: 6px solid #e5e6e8;
28
+ }
29
+
30
+ .header-menu.header-popover-showed::after {
31
+ content: '';
32
+ position: absolute;
33
+ top: 7px;
34
+ right: 12px;
35
+ border-top-color: transparent;
36
+ border-bottom-color: #222428;
37
+ }
38
+
39
+ .header-menu .header-toggle {
40
+ flex: 1;
41
+ display: flex;
42
+ justify-content: space-between;
43
+ align-items: center;
44
+ padding-left: 5px;
45
+ height: 24px;
46
+ cursor: pointer;
47
+ }
48
+
49
+
50
+ .header-menu .header-toggle .active {
51
+ font-size: 14px;
52
+ }
53
+
54
+ .header-menu .header-toggle-disabled {
55
+ color: #BDBDBD;
56
+ cursor: unset;
57
+ }
58
+
59
+
60
+ .header-menu .header-toggle.header-popover-showed {
61
+ background-color: #e5e6e8;
62
+ border-radius: 2px;
63
+ }
64
+
65
+ .header-menu .header-toggle:not(.header-toggle-disabled):hover {
66
+ background-color: #e5e6e8;
67
+ border-radius: 2px;
68
+ }
69
+
70
+ .header-menu .header-toggle .sf-font {
71
+ color: #999;
72
+ width: 12px;
73
+ height: 24px;
74
+ margin-left: 8px;
75
+ display: inline-flex;
76
+ justify-content: center;
77
+ align-items: center;
78
+ font-size: 12px;
79
+ cursor: pointer;
80
+ }
81
+
82
+ .header-menu .header-popover {
83
+ position: absolute;
84
+ top: 42px;
85
+ left: 8px;
86
+ padding: 8px 0;
87
+ width: 100%;
88
+ background-color: #fff;
89
+ border: 1px solid #e5e6e8;
90
+ border-radius: 2px;
91
+ box-shadow: 0 0 10px #ccc;
92
+ display: flex;
93
+ flex-direction: column;
94
+ align-items: flex-start;
95
+ z-index: 101;
96
+ }
97
+
98
+ .header-menu .header-popover .sf-dropdown-menu-item {
99
+ font-size: 14px;
100
+ height: auto;
101
+ padding: 9px 16px 9px 28px;
102
+ width: 100%;
103
+ line-height: 1;
104
+ }
105
+
106
+ .header-menu .header-popover .sf-dropdown-menu-item:hover {
107
+ background-color: #F2F2F2;
108
+ }
109
+
110
+ .header-menu .header-popover .sf-dropdown-menu-item .sf-check-mark {
111
+ position: absolute;
112
+ left: 8px;
113
+ transform: scale(.8);
114
+ color: #798d99;
115
+ }
@@ -0,0 +1,134 @@
1
+ import { Editor, Element, Transforms, Node } from 'slate';
2
+ import isHotkey from 'is-hotkey';
3
+ import { generateEmptyElement, getSelectedNodeByTypes } from '../../core';
4
+ import { getHeaderType, isMenuDisabled, setHeaderType } from './helper';
5
+ import { MAC_HOTKEYS_EVENT_HEADER, WIN_HOTKEYS_EVENT_HEADER } from '../../constants/keyboard';
6
+ import { isMac } from '../../../utils/common';
7
+ import { HEADERS, LIST_TYPE_ARRAY, ELementTypes } from '../../constants';
8
+
9
+ const isSelectionAtLineEnd = (editor, path) => {
10
+ const { selection } = editor;
11
+
12
+ if (!selection) return false;
13
+ const isAtLineEnd = Editor.isEnd(editor, selection.anchor, path) || Editor.isEnd(editor, selection.focus, path);
14
+ return isAtLineEnd;
15
+ };
16
+
17
+ const isSelectionAtLineStart = (editor, path) => {
18
+ const { selection } = editor;
19
+
20
+ if (!selection) return false;
21
+ const isAtLineEnd = Editor.isStart(editor, selection.anchor, path) || Editor.isStart(editor, selection.focus, path);
22
+ return isAtLineEnd;
23
+ };
24
+
25
+ const withHeader = (editor) => {
26
+ const { insertBreak, insertFragment, insertText, deleteBackward, onHotKeyDown } = editor;
27
+ const newEditor = editor;
28
+
29
+ // Rewrite insertBreak - insert paragraph when carriage return at the end of header
30
+ newEditor.insertBreak = () => {
31
+ const [match] = Editor.nodes(newEditor, {
32
+ match: n => {
33
+ if (!Element.isElement(n)) return false;
34
+ if (n.type.startsWith(ELementTypes.HEADER)) return true;
35
+ return false;
36
+ }, // Matches nodes whose node.type starts with header
37
+ universal: true,
38
+ });
39
+
40
+ if (!match) {
41
+ insertBreak();
42
+ return;
43
+ }
44
+
45
+ const isAtLineEnd = isSelectionAtLineEnd(editor, match[1]);
46
+
47
+ const nextNode = Editor.next(editor, { at: match[1] });
48
+ if (isAtLineEnd && nextNode && editor.children.length === 2) {
49
+ const [node, path] = nextNode;
50
+ if (node && node.children[0].text === '') {
51
+ Transforms.select(editor, path);
52
+ return;
53
+ }
54
+ }
55
+
56
+ // If an empty p is inserted at the end of the line, otherwise wrap normally
57
+ if (isAtLineEnd) {
58
+ const p = generateEmptyElement(ELementTypes.PARAGRAPH);
59
+ Transforms.insertNodes(newEditor, p, { mode: 'highest' });
60
+ } else {
61
+ insertBreak();
62
+ }
63
+ };
64
+
65
+ newEditor.deleteBackward = (data) => {
66
+ const [match] = Editor.nodes(newEditor, {
67
+ match: n => {
68
+ if (!Element.isElement(n)) return false;
69
+ if (n.type.startsWith(ELementTypes.HEADER)) return true;
70
+ return false;
71
+ }, // Matches nodes whose node.type starts with header
72
+ universal: true,
73
+ });
74
+
75
+ if (!match) {
76
+ deleteBackward(data);
77
+ return false;
78
+ }
79
+
80
+ const isAtLineStart = isSelectionAtLineStart(editor, match[1]);
81
+ if (isAtLineStart) {
82
+ setHeaderType(editor, ELementTypes.PARAGRAPH);
83
+ return true;
84
+ }
85
+
86
+ return deleteBackward(data);
87
+ };
88
+
89
+ newEditor.insertFragment = (data) => {
90
+ const headerNode = getSelectedNodeByTypes(editor, HEADERS);
91
+ const headerText = Node.string(headerNode || { children: [] });
92
+ const isSingleListItem = data.length === 1 && data[0]?.children?.length === 1 && LIST_TYPE_ARRAY.includes(data[0].type);
93
+ // Insert a list item when the header is empty, insert only the text
94
+ if ((headerNode && headerText.length === 0) && isSingleListItem) {
95
+ const text = Node.string(data[0]);
96
+ insertText(text);
97
+ return;
98
+ }
99
+ return insertFragment(data);
100
+ };
101
+
102
+ newEditor.onHotKeyDown = (event) => {
103
+ const HOT_KEYS = isMac() ? MAC_HOTKEYS_EVENT_HEADER : WIN_HOTKEYS_EVENT_HEADER;
104
+ const hotEntries = Object.entries(HOT_KEYS);
105
+ let isHeaderEvent = false;
106
+ let headerType = '';
107
+
108
+ for (const element of hotEntries) {
109
+ const [key, value] = element;
110
+ isHeaderEvent = isHotkey(value, event);
111
+ if (isHeaderEvent) {
112
+ headerType = key;
113
+ break;
114
+ }
115
+ }
116
+ if (!isHeaderEvent) {
117
+ return onHotKeyDown && onHotKeyDown(event);
118
+ }
119
+
120
+ event.preventDefault();
121
+ if (isMenuDisabled(newEditor)) return true;
122
+ const currentHeaderType = getHeaderType(editor);
123
+ if (currentHeaderType === headerType) {
124
+ setHeaderType(newEditor, ELementTypes.PARAGRAPH);
125
+ } else {
126
+ setHeaderType(newEditor, headerType);
127
+ }
128
+ return true;
129
+ };
130
+
131
+ return newEditor;
132
+ };
133
+
134
+ export default withHeader;
@@ -0,0 +1,31 @@
1
+ import React from 'react';
2
+ import { getNodeType } from '../../core';
3
+ import { ELementTypes } from '../../constants';
4
+
5
+ export const renderHeader = (props, editor) => {
6
+ const { element, attributes, children } = props;
7
+ const HeaderTagName = `h${getHeaderTagName(element)}`;
8
+ const style = {
9
+ textAlign: element.align,
10
+ };
11
+
12
+ function getHeaderTagName(element) {
13
+ const currentNodeType = getNodeType(element);
14
+ if (currentNodeType.indexOf(ELementTypes.HEADER) > -1) {
15
+ return currentNodeType.split(ELementTypes.HEADER)[1];
16
+ }
17
+ return 'p';
18
+ }
19
+
20
+ return (
21
+ <HeaderTagName
22
+ data-id={element.id}
23
+ id={element.id} // used for click left outline item, page scroll this element
24
+ data-root='true'
25
+ {...attributes}
26
+ style={style}
27
+ >
28
+ {children}
29
+ </HeaderTagName>
30
+ );
31
+ };
@@ -0,0 +1,50 @@
1
+ import { Transforms } from 'slate';
2
+ import slugid from 'slugid';
3
+ import { getHeaderType } from '../header/helper';
4
+ import { IMAGE } from '../../constants/element-types';
5
+ import { generateDefaultText, getNodeType } from '../../core';
6
+ import { isInCodeBlock } from '../code-block/helpers';
7
+
8
+ export const isMenuDisabled = (editor, readonly) => {
9
+ if (readonly) return true;
10
+ const isHeader = getHeaderType(editor);
11
+ if (isHeader) return true;
12
+ if (isInCodeBlock(editor)) return true;
13
+ return false;
14
+ };
15
+
16
+ export const insertImage = (editor, url) => {
17
+ const imageNode = {
18
+ type: IMAGE,
19
+ id: slugid.nice(),
20
+ data: {
21
+ src: url,
22
+ },
23
+ children: [generateDefaultText()]
24
+ };
25
+ Transforms.insertNodes(editor, imageNode, { at: editor.selection, select: true });
26
+ };
27
+
28
+ export const updateImage = (editor, data) => {
29
+ Transforms.setNodes(editor, { data }, {
30
+ match: (n) => getNodeType(n) === IMAGE,
31
+ at: editor.selection,
32
+ voids: true
33
+ });
34
+ };
35
+
36
+ export const getImagesUrlList = (nodes) => {
37
+ let nodeIndex = 0;
38
+ const list = [];
39
+ while (nodes && nodeIndex <= nodes.length - 1) {
40
+ const currentNode = nodes[nodeIndex];
41
+ if (currentNode.type === IMAGE) {
42
+ const { data = {} } = currentNode;
43
+ data.src && list.push(data.src);
44
+ } else {
45
+ list.push(...getImagesUrlList(currentNode.children));
46
+ }
47
+ nodeIndex++;
48
+ }
49
+ return list;
50
+ };
@@ -0,0 +1,14 @@
1
+ import withImages from './plugin';
2
+ import { IMAGE } from '../../constants/element-types';
3
+ import ImageMenu from './menu';
4
+ import renderImage from './render-element';
5
+
6
+ const ImagePlugin = {
7
+ type: IMAGE,
8
+ nodeType: 'element',
9
+ editorMenus: [ImageMenu],
10
+ editorPlugin: withImages,
11
+ renderElements: [renderImage],
12
+ };
13
+
14
+ export default ImagePlugin;
@@ -0,0 +1,68 @@
1
+ import React, { useState, useRef, useMemo } from 'react';
2
+ import { Button, Modal, ModalHeader, ModalBody, ModalFooter, Form, FormGroup, Label, Input, FormFeedback } from 'reactstrap';
3
+ import PropTypes from 'prop-types';
4
+ import { useTranslation } from 'react-i18next';
5
+ import isUrl from 'is-url';
6
+ import { insertImage } from '../helper';
7
+
8
+ const ImageMenuInsertInternetDialog = ({ editor, onToggleImageDialog, className }) => {
9
+ const [imageUrl, setSetImageUrl] = useState('');
10
+ const imgUrlInputRef = useRef(null);
11
+ const { t } = useTranslation();
12
+
13
+ const isCommitBtnDisabled = useMemo(() => {
14
+ if (imageUrl.length === 0) return true;
15
+ if (!isUrl(imageUrl)) return true;
16
+ return false;
17
+ }, [imageUrl]);
18
+
19
+ const onModalOpened = () => {
20
+ imgUrlInputRef?.current?.focus();
21
+ };
22
+
23
+ const handleSubmit = (e) => {
24
+ if (!isUrl(imageUrl)) return false;
25
+ if (imageUrl.length === 0) return;
26
+ insertImage(editor, imageUrl);
27
+ onToggleImageDialog();
28
+ };
29
+
30
+ const handleInputKeyDown = (e) => {
31
+ if (e.key === 'Enter') {
32
+ handleSubmit(e);
33
+ }
34
+ };
35
+
36
+ return (
37
+ <Modal isOpen={true} toggle={onToggleImageDialog} className={className} zIndex={1071} onOpened={onModalOpened} onClosed={onToggleImageDialog}>
38
+ <ModalHeader toggle={onToggleImageDialog}>{t('Insert_image')}</ModalHeader>
39
+ <ModalBody>
40
+ <Form>
41
+ <FormGroup>
42
+ <Label for="imageUrlInput" >{t('Image_address')}</Label>
43
+ <Input invalid={imageUrl && isCommitBtnDisabled} type='url' onKeyDown={handleInputKeyDown} id="imageUrlInput" innerRef={imgUrlInputRef} value={imageUrl} onChange={e => setSetImageUrl(e.target.value.trim())} />
44
+ <FormFeedback>
45
+ {t('Image_address_invalid')}
46
+ </FormFeedback>
47
+ </FormGroup>
48
+ </Form>
49
+ </ModalBody>
50
+ <ModalFooter>
51
+ <Button color="secondary" onClick={onToggleImageDialog}>{t('Cancel')}</Button>
52
+ <Button color="primary" disabled={isCommitBtnDisabled} onClick={handleSubmit}>{t('Submit')}</Button>
53
+ </ModalFooter>
54
+ </Modal>
55
+ );
56
+ };
57
+
58
+ ImageMenuInsertInternetDialog.defaultProps = {
59
+ classname: '',
60
+ };
61
+
62
+ ImageMenuInsertInternetDialog.propTypes = {
63
+ editor: PropTypes.object.isRequired,
64
+ onToggleImageDialog: PropTypes.func.isRequired,
65
+ className: PropTypes.string
66
+ };
67
+
68
+ export default ImageMenuInsertInternetDialog;
@@ -0,0 +1,76 @@
1
+ import React, { useCallback, useState } from 'react';
2
+ import propTypes from 'prop-types';
3
+ import { useTranslation } from 'react-i18next';
4
+ import { Label } from 'reactstrap';
5
+ import ImageMenuInsertInternetDialog from './image-menu-dialog';
6
+ import { insertImage } from '../helper';
7
+
8
+ import './style.css';
9
+
10
+ const ImageMenuPopover = ({ editor, handelClosePopover }) => {
11
+ const [isShowInternetImageModal, setIsShowInternetImageModal] = useState(false);
12
+ const { t } = useTranslation();
13
+
14
+ const handleInsertNextworkImage = (e) => {
15
+ e.nativeEvent.stopImmediatePropagation();
16
+ e.stopPropagation();
17
+ setIsShowInternetImageModal(true);
18
+ };
19
+
20
+ const handleClickFileInput = useCallback((e) => {
21
+ e.stopPropagation();
22
+ e.target.value = null;
23
+ e.nativeEvent.stopImmediatePropagation();
24
+ }, []);
25
+
26
+ const handleUploadLocalImage = useCallback(async (e) => {
27
+ if (editor.api.uploadLocalImage) {
28
+ const file = e.target.files[0];
29
+ try {
30
+ const imgUrl = await editor.api.uploadLocalImage(file);
31
+ insertImage(editor, imgUrl);
32
+ } catch (error) {
33
+ console.log('error', error);
34
+ }
35
+ }
36
+ handelClosePopover();
37
+ }, [editor, handelClosePopover]);
38
+
39
+ const onToggleImageDialog = useCallback(() => {
40
+ setIsShowInternetImageModal(false);
41
+ handelClosePopover();
42
+ }, [handelClosePopover]);
43
+
44
+ return (
45
+ <div className='image-popover'>
46
+ <div className='image-popover-item' onClick={handleInsertNextworkImage}>{t('Insert_network_image')}</div>
47
+ <Label className='image-popover-item' for='image-uploader' onClick={handleClickFileInput} >
48
+ {t('Upload_local_image')}
49
+ </Label>
50
+ <input
51
+ onClick={handleClickFileInput}
52
+ onChange={handleUploadLocalImage}
53
+ type="file"
54
+ accept='image/*'
55
+ className='image-uploader'
56
+ id='image-uploader'
57
+ />
58
+ {isShowInternetImageModal && (
59
+ <ImageMenuInsertInternetDialog
60
+ editor={editor}
61
+ onToggleImageDialog={onToggleImageDialog}
62
+ />)
63
+ }
64
+ </div>
65
+ );
66
+ };
67
+
68
+ ImageMenuPopover.defaultProps = {
69
+ };
70
+
71
+ ImageMenuPopover.propTypes = {
72
+ editor: propTypes.object.isRequired,
73
+ handelClosePopover: propTypes.func.isRequired,
74
+ };
75
+
76
+ export default ImageMenuPopover;