@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,210 @@
1
+ import isHotkey from 'is-hotkey';
2
+ import { Transforms, Node, Range, Editor } from 'slate';
3
+ import { getNodeType, isLastNode, getSelectedNodeByType, generateEmptyElement, generateElement } from '../../core';
4
+ import { getCodeBlockNodeEntry } from './helpers';
5
+ import { CODE_BLOCK, CODE_LINE, PARAGRAPH } from '../../constants/element-types';
6
+
7
+ const withCodeBlock = (editor) => {
8
+ const { normalizeNode, insertFragment, insertText, insertBreak, insertData, insertNode, onHotKeyDown } = editor;
9
+ const newEditor = editor;
10
+
11
+ newEditor.insertData = (data) => {
12
+ if (data.types.includes('text/code-block') && !getSelectedNodeByType(editor, CODE_BLOCK)) {
13
+ const codeBlockNode = JSON.parse(data.getData('text/code-block'));
14
+ return insertNode(codeBlockNode);
15
+ }
16
+ insertData(data);
17
+ };
18
+
19
+ newEditor.insertFragment = (data) => {
20
+ // only selected code block content
21
+ if (data.length === 1 && data[0].type === CODE_BLOCK && !getSelectedNodeByType(editor, CODE_BLOCK)) {
22
+ data.forEach((node, index) => {
23
+ if (node.type === CODE_BLOCK) {
24
+ const newBlock = node.children.map(line => {
25
+ const text = Node.string(line);
26
+ const children = generateElement(PARAGRAPH, { childrenOrText: text });
27
+ return children;
28
+ });
29
+ data.splice(index, 1, ...newBlock);
30
+ }
31
+ });
32
+ return insertFragment(data);
33
+ } else {
34
+ if (getSelectedNodeByType(editor, CODE_BLOCK)) {
35
+ // Paste into code block
36
+
37
+ // Pasted data is code block split with code-line
38
+ data.forEach((node, index) => {
39
+ if (node.type === CODE_BLOCK) {
40
+ const codeLineArr = node.children.map(line => line);
41
+ data.splice(index, 1, ...codeLineArr);
42
+ }
43
+ });
44
+ const insertCodeLines = data.map(node => {
45
+ const text = Node.string(node);
46
+ const codeLine = generateElement(CODE_LINE, { childrenOrText: text });
47
+ return codeLine;
48
+ });
49
+
50
+ // Current focus code-line string not empty
51
+ const string = Editor.string(newEditor, newEditor.selection.focus.path);
52
+ if (string.length !== 0 && Range.isCollapsed(newEditor.selection)) {
53
+ const [node, ...restNode] = insertCodeLines;
54
+ const text = Node.string(node);
55
+ insertText(text);
56
+ if (restNode.length !== 0) {
57
+ insertBreak();
58
+ insertFragment(restNode);
59
+ }
60
+ return;
61
+ }
62
+ return insertFragment(insertCodeLines);
63
+ } else {
64
+ // Paste into not a code block
65
+ return insertFragment(data);
66
+ }
67
+ }
68
+ };
69
+
70
+ // Rewrite normalizeNode
71
+ newEditor.normalizeNode = ([node, path]) => {
72
+ const type = getNodeType(node);
73
+
74
+ if (type === CODE_LINE && path.length <= 1) {
75
+ Transforms.setNodes(newEditor, { type: PARAGRAPH }, { at: path });
76
+ return;
77
+ }
78
+
79
+ if (type === CODE_BLOCK) {
80
+ if (node.children.length === 0) {
81
+ Transforms.delete(newEditor, { at: path });
82
+ return;
83
+ }
84
+
85
+ // code-block is the last node in the editor and needs to be followed by a p node
86
+ const isLast = isLastNode(newEditor, node);
87
+ if (isLast) {
88
+ const paragraph = generateEmptyElement(PARAGRAPH);
89
+ Transforms.insertNodes(newEditor, paragraph, { at: [path[0] + 1] });
90
+ }
91
+
92
+ // Here must be a code node below code-block
93
+ if (getNodeType(node.children[0]) !== CODE_LINE) {
94
+ Transforms.unwrapNodes(newEditor);
95
+ Transforms.setNodes(newEditor, { type: PARAGRAPH }, { mode: 'highest' });
96
+ }
97
+
98
+ if (node.children.length > 1) {
99
+ node.children.forEach((child, index) => {
100
+ if (child.type !== CODE_LINE) {
101
+ Transforms.setNodes(newEditor, { type: CODE_LINE }, { at: [...path, index] });
102
+ }
103
+ });
104
+ }
105
+ }
106
+
107
+ // Perform default behavior
108
+ return normalizeNode([node, path]);
109
+ };
110
+
111
+ newEditor.onHotKeyDown = (event) => {
112
+ const wrapperCodeBlock = getCodeBlockNodeEntry(newEditor);
113
+ if (!wrapperCodeBlock) return onHotKeyDown && onHotKeyDown(event);
114
+
115
+ if (isHotkey('mod+enter', event)) {
116
+ event.preventDefault();
117
+ if (newEditor.selection && !Range.isExpanded(newEditor.selection)) {
118
+ const path = Editor.path(newEditor, newEditor.selection);
119
+ const newParagraphPath = [path[0] + 1];
120
+ const newParagraph = generateEmptyElement(PARAGRAPH);
121
+ Transforms.insertNodes(newEditor, newParagraph, { at: newParagraphPath });
122
+ Transforms.select(newEditor, newParagraphPath);
123
+ }
124
+ return true;
125
+ }
126
+
127
+ if (isHotkey('tab', event)) {
128
+ event.preventDefault();
129
+ const nodeEntries = Editor.nodes(newEditor, {
130
+ mode: 'lowest',
131
+ match: node => node.type === CODE_LINE,
132
+ });
133
+ const nodeEntryList = Array.from(nodeEntries);
134
+ for (const nodeEntry of nodeEntryList) {
135
+ const [, path] = nodeEntry;
136
+ // Insert 4 spaces for easier remove space
137
+ Transforms.insertText(newEditor, ' '.repeat(4), { at: { path: [...path, 0], offset: 0 } });
138
+ }
139
+ const firstCodeLinePath = nodeEntryList.at(0)[1].concat(0);
140
+ const lastCodeLinePath = nodeEntryList.at(-1)[1].concat(0);
141
+ const newRange = Editor.range(newEditor, firstCodeLinePath, lastCodeLinePath);
142
+ nodeEntryList.length > 1 ? Transforms.select(newEditor, newRange) : Transforms.select(newEditor);
143
+ return true;
144
+ }
145
+
146
+ if (isHotkey('shift+tab', event)) {
147
+ event.preventDefault();
148
+ // Match the beginning of the line space, delete up to 4 spaces at a time
149
+ const originSelection = newEditor.selection;
150
+ const matchBeginSpace = /^\s*/;
151
+ const nodeEntries = Editor.nodes(newEditor, {
152
+ mode: 'lowest',
153
+ match: node => node.type === CODE_LINE,
154
+ });
155
+ const nodeEntryList = Array.from(nodeEntries);
156
+ let removedSpaceCount = 0;
157
+
158
+ for (const nodeEntry of nodeEntryList) {
159
+ const [node, path] = nodeEntry;
160
+ const spaceNum = Node.string(node).match(matchBeginSpace);
161
+ // skip empty line and no space begining line
162
+ if (!spaceNum || !spaceNum[0].length) continue;
163
+ const deleteNum = Math.min(spaceNum[0].length, 4);
164
+ removedSpaceCount += deleteNum;
165
+ for (let i = 0; i < deleteNum; i++) {
166
+ Transforms.select(newEditor, { path: [...path, 0], offset: 0 });
167
+ Editor.deleteForward(newEditor, { unit: 'character' });
168
+ }
169
+ }
170
+ // Select multiple rows when operating more then one line
171
+ // Keep cursor location when operating one line
172
+ if (nodeEntryList.length > 1) {
173
+ const firstCodeLinePath = nodeEntryList.at(0)[1].concat(0);
174
+ const lastCodeLinePath = nodeEntryList.at(-1)[1].concat(0);
175
+ const selectLocation = Editor.range(newEditor, firstCodeLinePath, lastCodeLinePath);
176
+ Transforms.select(newEditor, selectLocation);
177
+ } else {
178
+ const { anchor, focus } = originSelection;
179
+ const isCollapsed = Range.isCollapsed(originSelection);
180
+ if (isCollapsed) {
181
+ const selectLocation = { ...originSelection.focus, offset: originSelection.focus.offset - removedSpaceCount };
182
+ Transforms.select(newEditor, selectLocation);
183
+ } else {
184
+ const selectLocation = {
185
+ anchor: { ...anchor, offset: anchor.offset - removedSpaceCount },
186
+ focus: { ...focus, offset: focus.offset - removedSpaceCount }
187
+ };
188
+ Transforms.select(newEditor, selectLocation);
189
+ }
190
+ }
191
+ return true;
192
+ }
193
+
194
+ if (isHotkey('mod+a', event)) {
195
+ event.preventDefault();
196
+ const codeBlockEntry = Editor.nodes(newEditor, {
197
+ mode: 'highest',
198
+ match: node => node.type === CODE_BLOCK,
199
+ });
200
+ if (!codeBlockEntry) return;
201
+ const codeBlockEntryList = Array.from(...codeBlockEntry);
202
+ Transforms.select(newEditor, codeBlockEntryList[1]);
203
+ return true;
204
+ }
205
+ };
206
+
207
+ return newEditor;
208
+ };
209
+
210
+ export default withCodeBlock;
@@ -0,0 +1,17 @@
1
+ // Default language key name
2
+ export const EXPLAIN_TEXT = 'none';
3
+
4
+ export const LANGUAGE_MAP = {
5
+ [EXPLAIN_TEXT]: ' Text',
6
+ html: ' HTML',
7
+ css: ' CSS',
8
+ javascript: ' Javascript',
9
+ c: ' C',
10
+ cpp: ' C++',
11
+ csharp: ' C#',
12
+ java: ' Java',
13
+ python: ' Python',
14
+ sql: ' Sql',
15
+ swift: ' Swift',
16
+ json: ' JSON'
17
+ };
@@ -0,0 +1,55 @@
1
+ /* eslint-disable react-hooks/rules-of-hooks */
2
+ import React, { useCallback, useEffect, useRef, useState } from 'react';
3
+ import { useSlate } from 'slate-react';
4
+ import LanguageSelector from './language-selector';
5
+ import { isInCodeBlock } from '../helpers';
6
+
7
+ const renderCodeBlock = ({ attributes, children, element }) => {
8
+ const editor = useSlate();
9
+ const [isShowLanguageSelector, setIsShowLanguageSelector] = useState(true);
10
+ const codeBlockRef = useRef(null);
11
+
12
+ useEffect(() => {
13
+ if (isShowLanguageSelector && !isInCodeBlock(editor)) onHideLanguageSelector();
14
+ if (!isShowLanguageSelector && isInCodeBlock(editor)) registerEventHandler();
15
+ // eslint-disable-next-line react-hooks/exhaustive-deps
16
+ }, [editor.selection]);
17
+
18
+ const onHideLanguageSelector = useCallback((e) => {
19
+ if (codeBlockRef?.current?.contains(e?.target)) return;
20
+ setIsShowLanguageSelector(false);
21
+ unregisterEventHandler();
22
+ // eslint-disable-next-line react-hooks/exhaustive-deps
23
+ }, [setIsShowLanguageSelector]);
24
+ const registerEventHandler = useCallback(() => {
25
+ setIsShowLanguageSelector(true);
26
+ document.addEventListener('click', onHideLanguageSelector, true);
27
+ }, [onHideLanguageSelector]);
28
+
29
+ const unregisterEventHandler = useCallback(() => {
30
+ document.removeEventListener('click', onHideLanguageSelector);
31
+ }, [onHideLanguageSelector]);
32
+
33
+ return (
34
+ <div
35
+ ref={codeBlockRef}
36
+ className='sf-code-block-container'
37
+ >
38
+ <pre {...attributes}>
39
+ <code>{children}</code>
40
+ </pre>
41
+ {isShowLanguageSelector && <LanguageSelector lang={element.lang} />}
42
+ </div>
43
+ );
44
+ };
45
+
46
+ export default renderCodeBlock;
47
+
48
+ export const renderCodeLine = (props, editor) => {
49
+ const { element, attributes, children } = props;
50
+ return (
51
+ <div data-id={element.id} {...attributes} className={'sf-code-line'}>
52
+ {children}
53
+ </div>
54
+ );
55
+ };
@@ -0,0 +1,38 @@
1
+ import React, { useMemo } from 'react';
2
+ import { useSlate } from 'slate-react';
3
+ import PropTypes from 'prop-types';
4
+ import { EXPLAIN_TEXT, LANGUAGE_MAP } from './constant';
5
+ import { setCodeBlockLanguage } from '../helpers';
6
+
7
+ import './style.css';
8
+
9
+ const LanguageSelector = ({ lang = EXPLAIN_TEXT }) => {
10
+ const editor = useSlate();
11
+ const langOptions = useMemo(() => {
12
+ const options = [];
13
+ for (const value in LANGUAGE_MAP) {
14
+ if (Object.hasOwnProperty.call(LANGUAGE_MAP, value)) {
15
+ const name = LANGUAGE_MAP[value];
16
+ options.push(<option key={value} value={value}>{name}</option>);
17
+ }
18
+ }
19
+ return options;
20
+ }, []);
21
+
22
+ return (
23
+ <select
24
+ name='language'
25
+ className='sf-lang-selector'
26
+ value={lang}
27
+ onChange={e => setCodeBlockLanguage(editor, e.target.value)}
28
+ >
29
+ {langOptions}
30
+ </select>
31
+ );
32
+ };
33
+
34
+ LanguageSelector.propTypes = {
35
+ lang: PropTypes.string.isRequired,
36
+ };
37
+
38
+ export default LanguageSelector;
@@ -0,0 +1,27 @@
1
+ .sf-code-block-container {
2
+ position: relative;
3
+ margin: 0 0 20px;
4
+ padding: 0;
5
+ border-radius: 4px;
6
+ overflow: hidden;
7
+ background: #f5f7fa;
8
+ }
9
+
10
+ .sf-lang-selector {
11
+ position: absolute;
12
+ top: 0;
13
+ right: 0;
14
+ display: flex;
15
+ padding: 2px 5px;
16
+ width: max-content;
17
+ border: none;
18
+ border-radius: 3px;
19
+ appearance: none;
20
+ align-items: center;
21
+ line-height: 1;
22
+ font-size: 12px;
23
+ background: #eeeeee;
24
+ cursor: pointer;
25
+ transition: all 0.3s;
26
+ outline: none;
27
+ }
@@ -0,0 +1,22 @@
1
+ /* formula */
2
+ .sf-selected-formula {
3
+ display: block;
4
+ user-select: none;
5
+ outline: 2px solid #007bff;
6
+ cursor: pointer;
7
+ }
8
+
9
+ .sf-selected-formula .MathJax {
10
+ margin: 0;
11
+ }
12
+
13
+ .formula-dialog-wrapper .form-check-label {
14
+ margin-bottom: 4px;
15
+ }
16
+
17
+ .formula-dialog-wrapper .formula-preview {
18
+ padding: 10px;
19
+ border: 2px dashed #ebebeb;
20
+ min-height: 40px;
21
+ overflow-x: auto;
22
+ }
@@ -0,0 +1,68 @@
1
+ import { Editor, Transforms } from 'slate';
2
+ import slugid from 'slugid';
3
+ import { getNodeType, getSelectedElems } from '../../core/queries';
4
+ import { focusEditor } from '../../core/transforms/focus-editor';
5
+ import { ELementTypes } from '../../constants';
6
+ import { FORMULA } from '../../constants/element-types';
7
+ import { generateDefaultText } from '../../core';
8
+
9
+ export const isMenuDisabled = (editor, readonly = false) => {
10
+ if (readonly) return true;
11
+ const { selection } = editor;
12
+ if (!selection) return false;
13
+ const selectedElems = getSelectedElems(editor);
14
+ // Check if the selected element is illegal
15
+ const isSelectedIllegalElement = selectedElems.some(elem => {
16
+ const { type } = elem;
17
+ if (editor.isVoid(elem)) return true;
18
+ const unSupportTypes = [
19
+ ELementTypes.CODE_BLOCK,
20
+ ELementTypes.CODE_LINE,
21
+ ELementTypes.TABLE,
22
+ ELementTypes.TABLE_CELL,
23
+ ELementTypes.TABLE_ROW,
24
+ ELementTypes.HEADER1,
25
+ ELementTypes.HEADER2,
26
+ ELementTypes.HEADER3,
27
+ ELementTypes.HEADER4,
28
+ ELementTypes.HEADER5,
29
+ ELementTypes.HEADER6,
30
+ ELementTypes.LIST_ITEM,
31
+ ELementTypes.UNORDERED_LIST,
32
+ ELementTypes.ORDERED_LIST,
33
+ ];
34
+ if (unSupportTypes.includes(type)) return true;
35
+ return false;
36
+ });
37
+ if (isSelectedIllegalElement) return true;
38
+ return false;
39
+ };
40
+
41
+ export const isFormulaActive = (editor) => {
42
+ return isFormulaType(editor);
43
+ };
44
+
45
+ export const isFormulaType = (editor) => {
46
+ const [match] = Editor.nodes(editor, {
47
+ match: n => getNodeType(n) === ELementTypes.FORMULA,
48
+ universal: true,
49
+ });
50
+ return !!match;
51
+ };
52
+
53
+ export const insertFormula = (editor, data) => {
54
+ const formula = {
55
+ id: slugid.nice(),
56
+ type: FORMULA,
57
+ data: { formula: data.formula },
58
+ children: [generateDefaultText()],
59
+ };
60
+ Transforms.insertNodes(editor, formula, { at: data.at, void: true });
61
+ focusEditor(editor);
62
+ };
63
+
64
+ export const updateFormula = (editor, data) => {
65
+ const { formula, at } = data;
66
+ Transforms.setNodes(editor, { data: { formula } }, { at: at, void: true });
67
+ focusEditor(editor);
68
+ };
@@ -0,0 +1,14 @@
1
+ import { FORMULA } from '../../constants/element-types';
2
+ import FormulaMenu from './menu';
3
+ import withFormula from './plugin';
4
+ import renderFormula from './render-elem';
5
+
6
+ const FormulaPlugin = {
7
+ type: FORMULA,
8
+ nodeType: 'element',
9
+ editorMenus: [FormulaMenu],
10
+ editorPlugin: withFormula,
11
+ renderElements: [renderFormula],
12
+ };
13
+
14
+ export default FormulaPlugin;
@@ -0,0 +1,73 @@
1
+ import React, { useCallback, useEffect, useRef, useState } from 'react';
2
+ import { Button, Modal, ModalHeader, ModalBody, ModalFooter, Input } from 'reactstrap';
3
+ import { useTranslation } from 'react-i18next';
4
+ import { insertFormula, updateFormula } from '../helper';
5
+ import { getAboveBlockNode } from '../../../core';
6
+
7
+ const FormulaModal = ({ editor, formula, onCloseModal }) => {
8
+ const oldFormulaRef = useRef(formula);
9
+ const [newFormula, setNewFormula] = useState(formula);
10
+ const [modifyPath, setModifyPath] = useState(null);
11
+ const [isDisabled, setIsDisabled] = useState(true);
12
+ const formulaPreviewRef = useRef(null);
13
+ const { t } = useTranslation('seafile-editor');
14
+
15
+ // record current selection position
16
+ useEffect(() => {
17
+ const node = getAboveBlockNode(editor);
18
+ if (node) {
19
+ setModifyPath(node[1]);
20
+ return;
21
+ }
22
+ // eslint-disable-next-line react-hooks/exhaustive-deps
23
+ }, []);
24
+
25
+ useEffect(() => {
26
+ if (formulaPreviewRef.current) {
27
+ formulaPreviewRef.current.innerHTML = '';
28
+ const dom = window.MathJax.tex2svg(newFormula);
29
+ formulaPreviewRef.current.appendChild(dom);
30
+ }
31
+ });
32
+
33
+ const onFormulaChange = useCallback((event) => {
34
+ setIsDisabled(false);
35
+ setNewFormula(event.target.value);
36
+ }, []);
37
+
38
+ const handleUpdateFormula = useCallback(() => {
39
+
40
+ const data = { formula: newFormula, at: modifyPath };
41
+ if (oldFormulaRef.current) {
42
+ updateFormula(editor, data);
43
+ } else {
44
+ insertFormula(editor, data);
45
+ }
46
+ onCloseModal();
47
+ }, [editor, modifyPath, newFormula, onCloseModal]);
48
+
49
+ return (
50
+ <Modal isOpen={true} toggle={onCloseModal} autoFocus={false} wrapClassName='formula-dialog-wrapper'>
51
+ <ModalHeader toggle={onCloseModal}>{t('insert_formula')}</ModalHeader>
52
+ <ModalBody>
53
+ <div className="form-group">
54
+ <label className="form-check-label">{t('Formula')}</label>
55
+ <Input
56
+ type="textarea"
57
+ className="form-control"
58
+ value={newFormula}
59
+ autoFocus={true}
60
+ onChange={onFormulaChange}
61
+ />
62
+ </div>
63
+ <div ref={formulaPreviewRef} className="formula-preview"></div>
64
+ </ModalBody>
65
+ <ModalFooter>
66
+ <Button color="secondary" onClick={onCloseModal}>{t('Cancel')}</Button>
67
+ <Button color="primary" disabled={isDisabled} onClick={handleUpdateFormula}>{t('Insert_formula')}</Button>
68
+ </ModalFooter>
69
+ </Modal>
70
+ );
71
+ };
72
+
73
+ export default FormulaModal;
@@ -0,0 +1,63 @@
1
+ import React, { useCallback, useEffect, useMemo, useState } from 'react';
2
+ import { INTERNAL_EVENTS } from '../../../../constants/event-types';
3
+ import { MENUS_CONFIG_MAP } from '../../../constants/menus-config';
4
+ import { FORMULA } from '../../../constants/element-types';
5
+ import MenuItem from '../../../commons/menu/menu-item';
6
+ import { isFormulaActive, isMenuDisabled } from '../helper';
7
+ import EventBus from '../../../../utils/event-bus';
8
+ import FormulaModal from './formula-modal';
9
+
10
+ const menuConfig = MENUS_CONFIG_MAP[FORMULA];
11
+
12
+ const FormulaMenu = ({ isRichEditor, className, readonly, editor }) => {
13
+ const [isOpenFormulaModal, setIsOpenFormulaModal] = useState(false);
14
+ const [formula, setFormula] = useState('');
15
+ // eslint-disable-next-line react-hooks/exhaustive-deps
16
+ const isMenuActive = useMemo(() => isFormulaActive(editor), [editor.selection]);
17
+
18
+ useEffect(() => {
19
+ const eventBus = EventBus.getInstance();
20
+ const unsubscribe = eventBus.subscribe(INTERNAL_EVENTS.ON_OPEN_FORMULA_DIALOG, handleOpenLinkModal);
21
+ return () => unsubscribe();
22
+ // eslint-disable-next-line react-hooks/exhaustive-deps
23
+ }, []);
24
+
25
+ const onMouseDown = useCallback(() => {
26
+ setIsOpenFormulaModal(true);
27
+ }, []);
28
+
29
+ const handleOpenLinkModal = useCallback((formulaElement) => {
30
+ if (formulaElement) {
31
+ const { formula } = formulaElement.data || {};
32
+ setFormula(formula);
33
+ setIsOpenFormulaModal(true);
34
+ }
35
+ }, []);
36
+
37
+ const onCloseModal = useCallback(() => {
38
+ setIsOpenFormulaModal(false);
39
+ setFormula('');
40
+ }, []);
41
+
42
+ return (
43
+ <>
44
+ <MenuItem
45
+ isRichEditor={isRichEditor}
46
+ className={className}
47
+ disabled={isMenuDisabled(editor, readonly)}
48
+ isActive={isMenuActive}
49
+ onMouseDown={onMouseDown}
50
+ {...menuConfig}
51
+ />
52
+ {isOpenFormulaModal && (
53
+ <FormulaModal
54
+ editor={editor}
55
+ formula={formula}
56
+ onCloseModal={onCloseModal}
57
+ />
58
+ )}
59
+ </>
60
+ );
61
+ };
62
+
63
+ export default FormulaMenu;
@@ -0,0 +1,18 @@
1
+ import { ELementTypes } from '../../constants';
2
+
3
+ const withFormula = (editor) => {
4
+ const { isVoid } = editor;
5
+ const newEditor = editor;
6
+
7
+
8
+ newEditor.isVoid = (element) => {
9
+ const { type } = element;
10
+
11
+ if (type === ELementTypes.FORMULA) return true;
12
+ return isVoid(element);
13
+ };
14
+
15
+ return newEditor;
16
+ };
17
+
18
+ export default withFormula;
@@ -0,0 +1,40 @@
1
+ import React, { useCallback, useEffect, useRef } from 'react';
2
+ import { useSelected } from 'slate-react';
3
+ import { INTERNAL_EVENTS } from '../../../constants/event-types';
4
+ import EventBus from '../../../utils/event-bus';
5
+
6
+ import './formula.css';
7
+
8
+ const Formula = ({ attributes, element, children }) => {
9
+
10
+ const isSelected = useSelected();
11
+ const formulaContainerRef = useRef(null);
12
+
13
+ useEffect(() => {
14
+ const { formula = '' } = element.data || {};
15
+ if (!formula) return;
16
+ if (formulaContainerRef.current && window.MathJax) {
17
+ formulaContainerRef.current.innerHTML = '';
18
+ const dom = window.MathJax.tex2svg(formula);
19
+ formulaContainerRef.current.appendChild(dom);
20
+ }
21
+ }, [element]);
22
+
23
+ const toggleFormulaEditor = useCallback(() => {
24
+ const eventBus = EventBus.getInstance();
25
+ eventBus.dispatch(INTERNAL_EVENTS.ON_OPEN_FORMULA_DIALOG, element);
26
+ }, [element]);
27
+
28
+ return (
29
+ <span onDoubleClick={toggleFormulaEditor} className={'sf-block-formula ' + (isSelected ? ' sf-selected-formula' : '')} {...attributes}>
30
+ <span contentEditable={false} ref={formulaContainerRef}></span>
31
+ <span contentEditable={false}>{children}</span>
32
+ </span>
33
+ );
34
+ };
35
+
36
+ const renderFormula = (props) => {
37
+ return <Formula {...props} />;
38
+ };
39
+
40
+ export default renderFormula;