@webiny/lexical-editor 0.0.0-unstable.06b2ede40f

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 (308) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +17 -0
  3. package/commands/image.d.ts +6 -0
  4. package/commands/image.js +4 -0
  5. package/commands/image.js.map +1 -0
  6. package/commands/index.d.ts +5 -0
  7. package/commands/index.js +7 -0
  8. package/commands/index.js.map +1 -0
  9. package/commands/list.d.ts +7 -0
  10. package/commands/list.js +6 -0
  11. package/commands/list.js.map +1 -0
  12. package/commands/quote.d.ts +5 -0
  13. package/commands/quote.js +4 -0
  14. package/commands/quote.js.map +1 -0
  15. package/commands/toolbar.d.ts +1 -0
  16. package/commands/toolbar.js +4 -0
  17. package/commands/toolbar.js.map +1 -0
  18. package/commands/typography.d.ts +8 -0
  19. package/commands/typography.js +4 -0
  20. package/commands/typography.js.map +1 -0
  21. package/components/Editor/EnsureHeadingTagPlugin.d.ts +6 -0
  22. package/components/Editor/EnsureHeadingTagPlugin.js +20 -0
  23. package/components/Editor/EnsureHeadingTagPlugin.js.map +1 -0
  24. package/components/Editor/HeadingEditor.d.ts +7 -0
  25. package/components/Editor/HeadingEditor.js +22 -0
  26. package/components/Editor/HeadingEditor.js.map +1 -0
  27. package/components/Editor/ParagraphEditor.d.ts +7 -0
  28. package/components/Editor/ParagraphEditor.js +22 -0
  29. package/components/Editor/ParagraphEditor.js.map +1 -0
  30. package/components/Editor/RichTextEditor.d.ts +49 -0
  31. package/components/Editor/RichTextEditor.js +133 -0
  32. package/components/Editor/RichTextEditor.js.map +1 -0
  33. package/components/Editor/normalizeInputValue.d.ts +6 -0
  34. package/components/Editor/normalizeInputValue.js +16 -0
  35. package/components/Editor/normalizeInputValue.js.map +1 -0
  36. package/components/LexicalEditorConfig/LexicalEditorConfig.d.ts +20 -0
  37. package/components/LexicalEditorConfig/LexicalEditorConfig.js +63 -0
  38. package/components/LexicalEditorConfig/LexicalEditorConfig.js.map +1 -0
  39. package/components/LexicalEditorConfig/components/Node.d.ts +14 -0
  40. package/components/LexicalEditorConfig/components/Node.js +30 -0
  41. package/components/LexicalEditorConfig/components/Node.js.map +1 -0
  42. package/components/LexicalEditorConfig/components/Plugin.d.ts +13 -0
  43. package/components/LexicalEditorConfig/components/Plugin.js +30 -0
  44. package/components/LexicalEditorConfig/components/Plugin.js.map +1 -0
  45. package/components/LexicalEditorConfig/components/ToolbarElement.d.ts +13 -0
  46. package/components/LexicalEditorConfig/components/ToolbarElement.js +30 -0
  47. package/components/LexicalEditorConfig/components/ToolbarElement.js.map +1 -0
  48. package/components/LexicalHtmlRenderer.d.ts +14 -0
  49. package/components/LexicalHtmlRenderer.js +55 -0
  50. package/components/LexicalHtmlRenderer.js.map +1 -0
  51. package/components/Toolbar/StaticToolbar.css +260 -0
  52. package/components/Toolbar/StaticToolbar.d.ts +5 -0
  53. package/components/Toolbar/StaticToolbar.js +22 -0
  54. package/components/Toolbar/StaticToolbar.js.map +1 -0
  55. package/components/Toolbar/Toolbar.css +643 -0
  56. package/components/Toolbar/Toolbar.d.ts +6 -0
  57. package/components/Toolbar/Toolbar.js +148 -0
  58. package/components/Toolbar/Toolbar.js.map +1 -0
  59. package/components/ToolbarActions/BoldAction.d.ts +2 -0
  60. package/components/ToolbarActions/BoldAction.js +25 -0
  61. package/components/ToolbarActions/BoldAction.js.map +1 -0
  62. package/components/ToolbarActions/BulletListAction.d.ts +2 -0
  63. package/components/ToolbarActions/BulletListAction.js +40 -0
  64. package/components/ToolbarActions/BulletListAction.js.map +1 -0
  65. package/components/ToolbarActions/CodeHighlightAction.d.ts +2 -0
  66. package/components/ToolbarActions/CodeHighlightAction.js +25 -0
  67. package/components/ToolbarActions/CodeHighlightAction.js.map +1 -0
  68. package/components/ToolbarActions/FontColorAction.d.ts +29 -0
  69. package/components/ToolbarActions/FontColorAction.js +50 -0
  70. package/components/ToolbarActions/FontColorAction.js.map +1 -0
  71. package/components/ToolbarActions/FontSizeAction.d.ts +14 -0
  72. package/components/ToolbarActions/FontSizeAction.js +91 -0
  73. package/components/ToolbarActions/FontSizeAction.js.map +1 -0
  74. package/components/ToolbarActions/ImageAction.d.ts +2 -0
  75. package/components/ToolbarActions/ImageAction.js +37 -0
  76. package/components/ToolbarActions/ImageAction.js.map +1 -0
  77. package/components/ToolbarActions/ItalicAction.d.ts +2 -0
  78. package/components/ToolbarActions/ItalicAction.js +25 -0
  79. package/components/ToolbarActions/ItalicAction.js.map +1 -0
  80. package/components/ToolbarActions/LinkAction.d.ts +2 -0
  81. package/components/ToolbarActions/LinkAction.js +37 -0
  82. package/components/ToolbarActions/LinkAction.js.map +1 -0
  83. package/components/ToolbarActions/NumberedListAction.d.ts +2 -0
  84. package/components/ToolbarActions/NumberedListAction.js +48 -0
  85. package/components/ToolbarActions/NumberedListAction.js.map +1 -0
  86. package/components/ToolbarActions/QuoteAction.d.ts +2 -0
  87. package/components/ToolbarActions/QuoteAction.js +33 -0
  88. package/components/ToolbarActions/QuoteAction.js.map +1 -0
  89. package/components/ToolbarActions/TextAlignmentAction.d.ts +29 -0
  90. package/components/ToolbarActions/TextAlignmentAction.js +61 -0
  91. package/components/ToolbarActions/TextAlignmentAction.js.map +1 -0
  92. package/components/ToolbarActions/TypographyAction.d.ts +29 -0
  93. package/components/ToolbarActions/TypographyAction.js +108 -0
  94. package/components/ToolbarActions/TypographyAction.js.map +1 -0
  95. package/components/ToolbarActions/UnderlineAction.d.ts +2 -0
  96. package/components/ToolbarActions/UnderlineAction.js +24 -0
  97. package/components/ToolbarActions/UnderlineAction.js.map +1 -0
  98. package/context/FontColorActionContext.d.ts +6 -0
  99. package/context/FontColorActionContext.js +4 -0
  100. package/context/FontColorActionContext.js.map +1 -0
  101. package/context/RichTextEditorContext.d.ts +19 -0
  102. package/context/RichTextEditorContext.js +21 -0
  103. package/context/RichTextEditorContext.js.map +1 -0
  104. package/context/SharedHistoryContext.d.ts +10 -0
  105. package/context/SharedHistoryContext.js +19 -0
  106. package/context/SharedHistoryContext.js.map +1 -0
  107. package/context/TextAlignmentActionContextProps.d.ts +9 -0
  108. package/context/TextAlignmentActionContextProps.js +4 -0
  109. package/context/TextAlignmentActionContextProps.js.map +1 -0
  110. package/context/TypographyActionContext.d.ts +8 -0
  111. package/context/TypographyActionContext.js +4 -0
  112. package/context/TypographyActionContext.js.map +1 -0
  113. package/hooks/index.d.ts +8 -0
  114. package/hooks/index.js +10 -0
  115. package/hooks/index.js.map +1 -0
  116. package/hooks/useCurrentElement.d.ts +7 -0
  117. package/hooks/useCurrentElement.js +27 -0
  118. package/hooks/useCurrentElement.js.map +1 -0
  119. package/hooks/useCurrentSelection.d.ts +13 -0
  120. package/hooks/useCurrentSelection.js +57 -0
  121. package/hooks/useCurrentSelection.js.map +1 -0
  122. package/hooks/useFontColorPicker.d.ts +2 -0
  123. package/hooks/useFontColorPicker.js +11 -0
  124. package/hooks/useFontColorPicker.js.map +1 -0
  125. package/hooks/useIsMounted.d.ts +1 -0
  126. package/hooks/useIsMounted.js +12 -0
  127. package/hooks/useIsMounted.js.map +1 -0
  128. package/hooks/useList.d.ts +2 -0
  129. package/hooks/useList.js +50 -0
  130. package/hooks/useList.js.map +1 -0
  131. package/hooks/useQuote.d.ts +2 -0
  132. package/hooks/useQuote.js +17 -0
  133. package/hooks/useQuote.js.map +1 -0
  134. package/hooks/useRichTextEditor.d.ts +2 -0
  135. package/hooks/useRichTextEditor.js +11 -0
  136. package/hooks/useRichTextEditor.js.map +1 -0
  137. package/hooks/useTextAlignmentAction.d.ts +1 -0
  138. package/hooks/useTextAlignmentAction.js +11 -0
  139. package/hooks/useTextAlignmentAction.js.map +1 -0
  140. package/hooks/useTypographyAction.d.ts +1 -0
  141. package/hooks/useTypographyAction.js +11 -0
  142. package/hooks/useTypographyAction.js.map +1 -0
  143. package/images/icons/LICENSE.md +5 -0
  144. package/images/icons/chat-square-quote.svg +1 -0
  145. package/images/icons/chevron-down.svg +1 -0
  146. package/images/icons/code.svg +1 -0
  147. package/images/icons/font-color.svg +1 -0
  148. package/images/icons/indent.svg +3 -0
  149. package/images/icons/insert-image.svg +4 -0
  150. package/images/icons/justify.svg +3 -0
  151. package/images/icons/link.svg +1 -0
  152. package/images/icons/list-ol.svg +1 -0
  153. package/images/icons/list-ul.svg +1 -0
  154. package/images/icons/outdent.svg +3 -0
  155. package/images/icons/pencil-fill.svg +1 -0
  156. package/images/icons/text-center.svg +1 -0
  157. package/images/icons/text-left.svg +1 -0
  158. package/images/icons/text-paragraph.svg +1 -0
  159. package/images/icons/text-right.svg +1 -0
  160. package/images/icons/type-bold.svg +1 -0
  161. package/images/icons/type-h1.svg +1 -0
  162. package/images/icons/type-h2.svg +1 -0
  163. package/images/icons/type-h3.svg +1 -0
  164. package/images/icons/type-h4.svg +1 -0
  165. package/images/icons/type-h5.svg +1 -0
  166. package/images/icons/type-h6.svg +1 -0
  167. package/images/icons/type-italic.svg +1 -0
  168. package/images/icons/type-strikethrough.svg +1 -0
  169. package/images/icons/type-underline.svg +1 -0
  170. package/images/icons/unlink_icon.svg +1 -0
  171. package/index.d.ts +37 -0
  172. package/index.js +50 -0
  173. package/index.js.map +1 -0
  174. package/package.json +40 -0
  175. package/plugins/BlurEventPlugin/BlurEventPlugin.d.ts +7 -0
  176. package/plugins/BlurEventPlugin/BlurEventPlugin.js +20 -0
  177. package/plugins/BlurEventPlugin/BlurEventPlugin.js.map +1 -0
  178. package/plugins/CodeHighlightPlugin/CodeHighlightPlugin.d.ts +1 -0
  179. package/plugins/CodeHighlightPlugin/CodeHighlightPlugin.js +14 -0
  180. package/plugins/CodeHighlightPlugin/CodeHighlightPlugin.js.map +1 -0
  181. package/plugins/CodeHighlightPlugin/index.d.ts +1 -0
  182. package/plugins/CodeHighlightPlugin/index.js +3 -0
  183. package/plugins/CodeHighlightPlugin/index.js.map +1 -0
  184. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditor.d.ts +16 -0
  185. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditor.js +158 -0
  186. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditor.js.map +1 -0
  187. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditorController.d.ts +8 -0
  188. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditorController.js +62 -0
  189. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditorController.js.map +1 -0
  190. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditorPlugin.css +176 -0
  191. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditorPlugin.d.ts +7 -0
  192. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditorPlugin.js +14 -0
  193. package/plugins/FloatingLinkEditorPlugin/FloatingLinkEditorPlugin.js.map +1 -0
  194. package/plugins/FloatingLinkEditorPlugin/LinkEditForm.d.ts +8 -0
  195. package/plugins/FloatingLinkEditorPlugin/LinkEditForm.js +104 -0
  196. package/plugins/FloatingLinkEditorPlugin/LinkEditForm.js.map +1 -0
  197. package/plugins/FloatingLinkEditorPlugin/LinkPreviewForm.d.ts +9 -0
  198. package/plugins/FloatingLinkEditorPlugin/LinkPreviewForm.js +34 -0
  199. package/plugins/FloatingLinkEditorPlugin/LinkPreviewForm.js.map +1 -0
  200. package/plugins/FloatingLinkEditorPlugin/index.d.ts +1 -0
  201. package/plugins/FloatingLinkEditorPlugin/index.js +3 -0
  202. package/plugins/FloatingLinkEditorPlugin/index.js.map +1 -0
  203. package/plugins/FloatingLinkEditorPlugin/isChildOfLinkEditor.d.ts +1 -0
  204. package/plugins/FloatingLinkEditorPlugin/isChildOfLinkEditor.js +12 -0
  205. package/plugins/FloatingLinkEditorPlugin/isChildOfLinkEditor.js.map +1 -0
  206. package/plugins/FontColorPlugin/FontColorPlugin.d.ts +1 -0
  207. package/plugins/FontColorPlugin/FontColorPlugin.js +27 -0
  208. package/plugins/FontColorPlugin/FontColorPlugin.js.map +1 -0
  209. package/plugins/FontColorPlugin/applyColorToNode.d.ts +3 -0
  210. package/plugins/FontColorPlugin/applyColorToNode.js +8 -0
  211. package/plugins/FontColorPlugin/applyColorToNode.js.map +1 -0
  212. package/plugins/FontColorPlugin/applyColorToSelection.d.ts +3 -0
  213. package/plugins/FontColorPlugin/applyColorToSelection.js +66 -0
  214. package/plugins/FontColorPlugin/applyColorToSelection.js.map +1 -0
  215. package/plugins/ImagesPlugin/ImagesPlugin.d.ts +12 -0
  216. package/plugins/ImagesPlugin/ImagesPlugin.js +151 -0
  217. package/plugins/ImagesPlugin/ImagesPlugin.js.map +1 -0
  218. package/plugins/LinkPlugin/LinkPlugin.d.ts +5 -0
  219. package/plugins/LinkPlugin/LinkPlugin.js +59 -0
  220. package/plugins/LinkPlugin/LinkPlugin.js.map +1 -0
  221. package/plugins/ListPLugin/ListPlugin.d.ts +1 -0
  222. package/plugins/ListPLugin/ListPlugin.js +17 -0
  223. package/plugins/ListPLugin/ListPlugin.js.map +1 -0
  224. package/plugins/QuoteNodePlugin/QuoteNodePlugin.d.ts +1 -0
  225. package/plugins/QuoteNodePlugin/QuoteNodePlugin.js +18 -0
  226. package/plugins/QuoteNodePlugin/QuoteNodePlugin.js.map +1 -0
  227. package/plugins/StateHandlingPlugin.d.ts +8 -0
  228. package/plugins/StateHandlingPlugin.js +75 -0
  229. package/plugins/StateHandlingPlugin.js.map +1 -0
  230. package/plugins/TypographyPlugin/TypographyPlugin.d.ts +1 -0
  231. package/plugins/TypographyPlugin/TypographyPlugin.js +27 -0
  232. package/plugins/TypographyPlugin/TypographyPlugin.js.map +1 -0
  233. package/types.d.ts +12 -0
  234. package/types.js +9 -0
  235. package/types.js.map +1 -0
  236. package/ui/ContentEditable.css +22 -0
  237. package/ui/ContentEditable.d.ts +12 -0
  238. package/ui/ContentEditable.js +19 -0
  239. package/ui/ContentEditable.js.map +1 -0
  240. package/ui/Divider.d.ts +2 -0
  241. package/ui/Divider.js +8 -0
  242. package/ui/Divider.js.map +1 -0
  243. package/ui/DropDown.d.ts +25 -0
  244. package/ui/DropDown.js +182 -0
  245. package/ui/DropDown.js.map +1 -0
  246. package/ui/ImageResizer.d.ts +24 -0
  247. package/ui/ImageResizer.js +211 -0
  248. package/ui/ImageResizer.js.map +1 -0
  249. package/ui/Input.css +32 -0
  250. package/ui/LinkPreview.css +69 -0
  251. package/ui/LinkPreview.d.ts +12 -0
  252. package/ui/LinkPreview.js +97 -0
  253. package/ui/LinkPreview.js.map +1 -0
  254. package/ui/Placeholder.css +20 -0
  255. package/ui/Placeholder.d.ts +15 -0
  256. package/ui/Placeholder.js +24 -0
  257. package/ui/Placeholder.js.map +1 -0
  258. package/ui/TextInput.d.ts +18 -0
  259. package/ui/TextInput.js +34 -0
  260. package/ui/TextInput.js.map +1 -0
  261. package/ui/ToolbarActionDialog.d.ts +11 -0
  262. package/ui/ToolbarActionDialog.js +77 -0
  263. package/ui/ToolbarActionDialog.js.map +1 -0
  264. package/utils/canUseDOM.d.ts +1 -0
  265. package/utils/canUseDOM.js +3 -0
  266. package/utils/canUseDOM.js.map +1 -0
  267. package/utils/files.d.ts +11 -0
  268. package/utils/files.js +21 -0
  269. package/utils/files.js.map +1 -0
  270. package/utils/getDOMRangeRect.d.ts +8 -0
  271. package/utils/getDOMRangeRect.js +23 -0
  272. package/utils/getDOMRangeRect.js.map +1 -0
  273. package/utils/getSelectedNode.d.ts +2 -0
  274. package/utils/getSelectedNode.js +25 -0
  275. package/utils/getSelectedNode.js.map +1 -0
  276. package/utils/getTransparentImage.d.ts +1 -0
  277. package/utils/getTransparentImage.js +5 -0
  278. package/utils/getTransparentImage.js.map +1 -0
  279. package/utils/insertImage.d.ts +2 -0
  280. package/utils/insertImage.js +16 -0
  281. package/utils/insertImage.js.map +1 -0
  282. package/utils/isAnchorLink.d.ts +1 -0
  283. package/utils/isAnchorLink.js +5 -0
  284. package/utils/isAnchorLink.js.map +1 -0
  285. package/utils/isChildOfFloatingToolbar.d.ts +1 -0
  286. package/utils/isChildOfFloatingToolbar.js +12 -0
  287. package/utils/isChildOfFloatingToolbar.js.map +1 -0
  288. package/utils/isHTMLElement.d.ts +8 -0
  289. package/utils/isHTMLElement.js +12 -0
  290. package/utils/isHTMLElement.js.map +1 -0
  291. package/utils/isValidJSON.d.ts +1 -0
  292. package/utils/isValidJSON.js +13 -0
  293. package/utils/isValidJSON.js.map +1 -0
  294. package/utils/isValidLexicalData.d.ts +4 -0
  295. package/utils/isValidLexicalData.js +24 -0
  296. package/utils/isValidLexicalData.js.map +1 -0
  297. package/utils/point.d.ts +21 -0
  298. package/utils/point.js +49 -0
  299. package/utils/point.js.map +1 -0
  300. package/utils/rect.d.ts +45 -0
  301. package/utils/rect.js +130 -0
  302. package/utils/rect.js.map +1 -0
  303. package/utils/sanitizeUrl.d.ts +1 -0
  304. package/utils/sanitizeUrl.js +25 -0
  305. package/utils/sanitizeUrl.js.map +1 -0
  306. package/utils/setFloatingElemPosition.d.ts +1 -0
  307. package/utils/setFloatingElemPosition.js +30 -0
  308. package/utils/setFloatingElemPosition.js.map +1 -0
@@ -0,0 +1,148 @@
1
+ import React, { Fragment, useCallback, useEffect, useRef, useState } from "react";
2
+ import { $getSelection, BLUR_COMMAND, COMMAND_PRIORITY_LOW, SELECTION_CHANGE_COMMAND } from "lexical";
3
+ import { createPortal } from "react-dom";
4
+ import { mergeRegister } from "@lexical/utils";
5
+ import "./Toolbar.css";
6
+ import { getDOMRangeRect } from "../../utils/getDOMRangeRect";
7
+ import { setFloatingElemPosition } from "../../utils/setFloatingElemPosition";
8
+ import { useLexicalEditorConfig } from "../LexicalEditorConfig/LexicalEditorConfig";
9
+ import { useDeriveValueFromSelection } from "../../hooks/useCurrentSelection";
10
+ import { useRichTextEditor } from "../../hooks";
11
+ import { isChildOfFloatingToolbar } from "../../utils/isChildOfFloatingToolbar";
12
+ import { HIDE_FLOATING_TOOLBAR } from "../../commands";
13
+ const FloatingToolbar = ({
14
+ anchorElem,
15
+ editor
16
+ }) => {
17
+ const [isVisible, setIsVisible] = useState(true);
18
+ const popupCharStylesEditorRef = useRef(null);
19
+ const {
20
+ toolbarElements
21
+ } = useLexicalEditorConfig();
22
+ const mouseMoveListener = useCallback(e => {
23
+ /* Indicates which mouse button(s) was pressed.
24
+ / 1 = mouse left button
25
+ / 3 = mouse left and right button in the same time
26
+ / More info: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons
27
+ */
28
+ if (popupCharStylesEditorRef?.current && (e.buttons === 1 || e.buttons === 3)) {
29
+ if (popupCharStylesEditorRef.current.style.pointerEvents !== "none") {
30
+ const x = e.clientX;
31
+ const y = e.clientY;
32
+ const elementUnderMouse = document.elementFromPoint(x, y);
33
+ if (!popupCharStylesEditorRef.current.contains(elementUnderMouse)) {
34
+ // Mouse is not over the target element => not a normal click, but probably a drag
35
+ popupCharStylesEditorRef.current.style.pointerEvents = "none";
36
+ }
37
+ }
38
+ }
39
+ }, [popupCharStylesEditorRef]);
40
+ const mouseUpListener = useCallback(() => {
41
+ if (popupCharStylesEditorRef?.current) {
42
+ if (popupCharStylesEditorRef.current.style.pointerEvents !== "auto") {
43
+ popupCharStylesEditorRef.current.style.pointerEvents = "auto";
44
+ }
45
+ }
46
+ }, [popupCharStylesEditorRef]);
47
+ useEffect(() => {
48
+ if (popupCharStylesEditorRef?.current) {
49
+ document.addEventListener("mousemove", mouseMoveListener);
50
+ document.addEventListener("mouseup", mouseUpListener);
51
+ return () => {
52
+ document.removeEventListener("mousemove", mouseMoveListener);
53
+ document.removeEventListener("mouseup", mouseUpListener);
54
+ };
55
+ }
56
+ return;
57
+ }, [popupCharStylesEditorRef]);
58
+ const updateTextFormatFloatingToolbar = useCallback(() => {
59
+ const selection = $getSelection();
60
+ const popupCharStylesEditorElem = popupCharStylesEditorRef.current;
61
+ const nativeSelection = window.getSelection();
62
+ if (popupCharStylesEditorElem === null) {
63
+ return;
64
+ }
65
+ const rootElement = editor.getRootElement();
66
+ if (selection !== null && nativeSelection !== null && !nativeSelection.isCollapsed && rootElement !== null && rootElement.contains(nativeSelection.anchorNode)) {
67
+ const rangeRect = getDOMRangeRect(nativeSelection, rootElement);
68
+ setFloatingElemPosition(rangeRect, popupCharStylesEditorElem, anchorElem);
69
+ }
70
+ }, [editor, anchorElem]);
71
+ useEffect(() => {
72
+ const scrollerElem = anchorElem.parentElement;
73
+ const update = () => {
74
+ editor.getEditorState().read(() => {
75
+ updateTextFormatFloatingToolbar();
76
+ });
77
+ };
78
+ window.addEventListener("resize", update);
79
+ if (scrollerElem) {
80
+ scrollerElem.addEventListener("scroll", update);
81
+ }
82
+ return () => {
83
+ window.removeEventListener("resize", update);
84
+ if (scrollerElem) {
85
+ scrollerElem.removeEventListener("scroll", update);
86
+ }
87
+ };
88
+ }, [editor, updateTextFormatFloatingToolbar, anchorElem]);
89
+ useEffect(() => {
90
+ editor.getEditorState().read(() => {
91
+ updateTextFormatFloatingToolbar();
92
+ });
93
+ return mergeRegister(editor.registerUpdateListener(({
94
+ editorState
95
+ }) => {
96
+ editorState.read(() => {
97
+ updateTextFormatFloatingToolbar();
98
+ });
99
+ }), editor.registerCommand(HIDE_FLOATING_TOOLBAR, () => {
100
+ setTimeout(() => {
101
+ setIsVisible(false);
102
+ }, 10);
103
+ return true;
104
+ }, COMMAND_PRIORITY_LOW), editor.registerCommand(SELECTION_CHANGE_COMMAND, () => {
105
+ setIsVisible(true);
106
+ updateTextFormatFloatingToolbar();
107
+ return false;
108
+ }, COMMAND_PRIORITY_LOW), editor.registerCommand(BLUR_COMMAND, payload => {
109
+ if (!isChildOfFloatingToolbar(payload.relatedTarget)) {
110
+ setIsVisible(false);
111
+ }
112
+ return false;
113
+ }, COMMAND_PRIORITY_LOW));
114
+ }, [editor, updateTextFormatFloatingToolbar]);
115
+ if (!isVisible) {
116
+ return null;
117
+ }
118
+ return /*#__PURE__*/React.createElement("div", {
119
+ ref: popupCharStylesEditorRef,
120
+ className: "floating-toolbar"
121
+ }, editor.isEditable() && /*#__PURE__*/React.createElement(React.Fragment, null, toolbarElements.map(action => /*#__PURE__*/React.createElement(Fragment, {
122
+ key: action.name
123
+ }, action.element))));
124
+ };
125
+ export const Toolbar = ({
126
+ anchorElem = document.body
127
+ }) => {
128
+ const {
129
+ editor
130
+ } = useRichTextEditor();
131
+ const showToolbar = useDeriveValueFromSelection(({
132
+ rangeSelection
133
+ }) => {
134
+ if (!rangeSelection) {
135
+ return false;
136
+ }
137
+ return !rangeSelection.isCollapsed();
138
+ });
139
+ if (!showToolbar) {
140
+ return null;
141
+ }
142
+ return /*#__PURE__*/createPortal(/*#__PURE__*/React.createElement(FloatingToolbar, {
143
+ anchorElem: anchorElem,
144
+ editor: editor
145
+ }), anchorElem);
146
+ };
147
+
148
+ //# sourceMappingURL=Toolbar.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","Fragment","useCallback","useEffect","useRef","useState","$getSelection","BLUR_COMMAND","COMMAND_PRIORITY_LOW","SELECTION_CHANGE_COMMAND","createPortal","mergeRegister","getDOMRangeRect","setFloatingElemPosition","useLexicalEditorConfig","useDeriveValueFromSelection","useRichTextEditor","isChildOfFloatingToolbar","HIDE_FLOATING_TOOLBAR","FloatingToolbar","anchorElem","editor","isVisible","setIsVisible","popupCharStylesEditorRef","toolbarElements","mouseMoveListener","e","current","buttons","style","pointerEvents","x","clientX","y","clientY","elementUnderMouse","document","elementFromPoint","contains","mouseUpListener","addEventListener","removeEventListener","updateTextFormatFloatingToolbar","selection","popupCharStylesEditorElem","nativeSelection","window","getSelection","rootElement","getRootElement","isCollapsed","anchorNode","rangeRect","scrollerElem","parentElement","update","getEditorState","read","registerUpdateListener","editorState","registerCommand","setTimeout","payload","relatedTarget","createElement","ref","className","isEditable","map","action","key","name","element","Toolbar","body","showToolbar","rangeSelection"],"sources":["Toolbar.tsx"],"sourcesContent":["import type { FC } from \"react\";\nimport React, { Fragment, useCallback, useEffect, useRef, useState } from \"react\";\nimport type { LexicalEditor } from \"lexical\";\nimport {\n $getSelection,\n BLUR_COMMAND,\n COMMAND_PRIORITY_LOW,\n SELECTION_CHANGE_COMMAND\n} from \"lexical\";\nimport { createPortal } from \"react-dom\";\nimport { mergeRegister } from \"@lexical/utils\";\nimport \"./Toolbar.css\";\nimport { getDOMRangeRect } from \"~/utils/getDOMRangeRect\";\nimport { setFloatingElemPosition } from \"~/utils/setFloatingElemPosition\";\nimport { useLexicalEditorConfig } from \"~/components/LexicalEditorConfig/LexicalEditorConfig\";\nimport { useDeriveValueFromSelection } from \"~/hooks/useCurrentSelection\";\nimport { useRichTextEditor } from \"~/hooks\";\nimport { isChildOfFloatingToolbar } from \"~/utils/isChildOfFloatingToolbar\";\nimport { HIDE_FLOATING_TOOLBAR } from \"~/commands\";\n\ninterface FloatingToolbarProps {\n anchorElem: HTMLElement;\n editor: LexicalEditor;\n}\n\nconst FloatingToolbar: FC<FloatingToolbarProps> = ({ anchorElem, editor }) => {\n const [isVisible, setIsVisible] = useState(true);\n const popupCharStylesEditorRef = useRef<HTMLDivElement | null>(null);\n const { toolbarElements } = useLexicalEditorConfig();\n\n const mouseMoveListener = useCallback(\n (e: MouseEvent) => {\n /* Indicates which mouse button(s) was pressed.\n / 1 = mouse left button\n / 3 = mouse left and right button in the same time\n / More info: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons\n */\n if (popupCharStylesEditorRef?.current && (e.buttons === 1 || e.buttons === 3)) {\n if (popupCharStylesEditorRef.current.style.pointerEvents !== \"none\") {\n const x = e.clientX;\n const y = e.clientY;\n const elementUnderMouse = document.elementFromPoint(x, y);\n\n if (!popupCharStylesEditorRef.current.contains(elementUnderMouse)) {\n // Mouse is not over the target element => not a normal click, but probably a drag\n popupCharStylesEditorRef.current.style.pointerEvents = \"none\";\n }\n }\n }\n },\n [popupCharStylesEditorRef]\n );\n\n const mouseUpListener = useCallback(() => {\n if (popupCharStylesEditorRef?.current) {\n if (popupCharStylesEditorRef.current.style.pointerEvents !== \"auto\") {\n popupCharStylesEditorRef.current.style.pointerEvents = \"auto\";\n }\n }\n }, [popupCharStylesEditorRef]);\n\n useEffect(() => {\n if (popupCharStylesEditorRef?.current) {\n document.addEventListener(\"mousemove\", mouseMoveListener);\n document.addEventListener(\"mouseup\", mouseUpListener);\n\n return () => {\n document.removeEventListener(\"mousemove\", mouseMoveListener);\n document.removeEventListener(\"mouseup\", mouseUpListener);\n };\n }\n return;\n }, [popupCharStylesEditorRef]);\n\n const updateTextFormatFloatingToolbar = useCallback(() => {\n const selection = $getSelection();\n\n const popupCharStylesEditorElem = popupCharStylesEditorRef.current;\n const nativeSelection = window.getSelection();\n\n if (popupCharStylesEditorElem === null) {\n return;\n }\n\n const rootElement = editor.getRootElement();\n if (\n selection !== null &&\n nativeSelection !== null &&\n !nativeSelection.isCollapsed &&\n rootElement !== null &&\n rootElement.contains(nativeSelection.anchorNode)\n ) {\n const rangeRect = getDOMRangeRect(nativeSelection, rootElement);\n\n setFloatingElemPosition(rangeRect, popupCharStylesEditorElem, anchorElem);\n }\n }, [editor, anchorElem]);\n\n useEffect(() => {\n const scrollerElem = anchorElem.parentElement;\n\n const update = () => {\n editor.getEditorState().read(() => {\n updateTextFormatFloatingToolbar();\n });\n };\n\n window.addEventListener(\"resize\", update);\n if (scrollerElem) {\n scrollerElem.addEventListener(\"scroll\", update);\n }\n\n return () => {\n window.removeEventListener(\"resize\", update);\n if (scrollerElem) {\n scrollerElem.removeEventListener(\"scroll\", update);\n }\n };\n }, [editor, updateTextFormatFloatingToolbar, anchorElem]);\n\n useEffect(() => {\n editor.getEditorState().read(() => {\n updateTextFormatFloatingToolbar();\n });\n return mergeRegister(\n editor.registerUpdateListener(({ editorState }) => {\n editorState.read(() => {\n updateTextFormatFloatingToolbar();\n });\n }),\n\n editor.registerCommand(\n HIDE_FLOATING_TOOLBAR,\n () => {\n setTimeout(() => {\n setIsVisible(false);\n }, 10);\n return true;\n },\n COMMAND_PRIORITY_LOW\n ),\n\n editor.registerCommand(\n SELECTION_CHANGE_COMMAND,\n () => {\n setIsVisible(true);\n updateTextFormatFloatingToolbar();\n return false;\n },\n COMMAND_PRIORITY_LOW\n ),\n\n editor.registerCommand(\n BLUR_COMMAND,\n payload => {\n if (!isChildOfFloatingToolbar(payload.relatedTarget as HTMLElement)) {\n setIsVisible(false);\n }\n\n return false;\n },\n COMMAND_PRIORITY_LOW\n )\n );\n }, [editor, updateTextFormatFloatingToolbar]);\n\n if (!isVisible) {\n return null;\n }\n\n return (\n <div ref={popupCharStylesEditorRef} className=\"floating-toolbar\">\n {editor.isEditable() && (\n <>\n {toolbarElements.map(action => (\n <Fragment key={action.name}>{action.element}</Fragment>\n ))}\n </>\n )}\n </div>\n );\n};\n\nexport interface ToolbarProps {\n anchorElem?: HTMLElement;\n}\n\nexport const Toolbar = ({ anchorElem = document.body }: ToolbarProps) => {\n const { editor } = useRichTextEditor();\n const showToolbar = useDeriveValueFromSelection(({ rangeSelection }) => {\n if (!rangeSelection) {\n return false;\n }\n\n return !rangeSelection.isCollapsed();\n });\n\n if (!showToolbar) {\n return null;\n }\n\n return createPortal(<FloatingToolbar anchorElem={anchorElem} editor={editor} />, anchorElem);\n};\n"],"mappings":"AACA,OAAOA,KAAK,IAAIC,QAAQ,EAAEC,WAAW,EAAEC,SAAS,EAAEC,MAAM,EAAEC,QAAQ,QAAQ,OAAO;AAEjF,SACIC,aAAa,EACbC,YAAY,EACZC,oBAAoB,EACpBC,wBAAwB,QACrB,SAAS;AAChB,SAASC,YAAY,QAAQ,WAAW;AACxC,SAASC,aAAa,QAAQ,gBAAgB;AAC9C;AACA,SAASC,eAAe;AACxB,SAASC,uBAAuB;AAChC,SAASC,sBAAsB;AAC/B,SAASC,2BAA2B;AACpC,SAASC,iBAAiB;AAC1B,SAASC,wBAAwB;AACjC,SAASC,qBAAqB;AAO9B,MAAMC,eAAyC,GAAGA,CAAC;EAAEC,UAAU;EAAEC;AAAO,CAAC,KAAK;EAC1E,MAAM,CAACC,SAAS,EAAEC,YAAY,CAAC,GAAGlB,QAAQ,CAAC,IAAI,CAAC;EAChD,MAAMmB,wBAAwB,GAAGpB,MAAM,CAAwB,IAAI,CAAC;EACpE,MAAM;IAAEqB;EAAgB,CAAC,GAAGX,sBAAsB,CAAC,CAAC;EAEpD,MAAMY,iBAAiB,GAAGxB,WAAW,CAChCyB,CAAa,IAAK;IACf;AACZ;AACA;AACA;AACA;IACY,IAAIH,wBAAwB,EAAEI,OAAO,KAAKD,CAAC,CAACE,OAAO,KAAK,CAAC,IAAIF,CAAC,CAACE,OAAO,KAAK,CAAC,CAAC,EAAE;MAC3E,IAAIL,wBAAwB,CAACI,OAAO,CAACE,KAAK,CAACC,aAAa,KAAK,MAAM,EAAE;QACjE,MAAMC,CAAC,GAAGL,CAAC,CAACM,OAAO;QACnB,MAAMC,CAAC,GAAGP,CAAC,CAACQ,OAAO;QACnB,MAAMC,iBAAiB,GAAGC,QAAQ,CAACC,gBAAgB,CAACN,CAAC,EAAEE,CAAC,CAAC;QAEzD,IAAI,CAACV,wBAAwB,CAACI,OAAO,CAACW,QAAQ,CAACH,iBAAiB,CAAC,EAAE;UAC/D;UACAZ,wBAAwB,CAACI,OAAO,CAACE,KAAK,CAACC,aAAa,GAAG,MAAM;QACjE;MACJ;IACJ;EACJ,CAAC,EACD,CAACP,wBAAwB,CAC7B,CAAC;EAED,MAAMgB,eAAe,GAAGtC,WAAW,CAAC,MAAM;IACtC,IAAIsB,wBAAwB,EAAEI,OAAO,EAAE;MACnC,IAAIJ,wBAAwB,CAACI,OAAO,CAACE,KAAK,CAACC,aAAa,KAAK,MAAM,EAAE;QACjEP,wBAAwB,CAACI,OAAO,CAACE,KAAK,CAACC,aAAa,GAAG,MAAM;MACjE;IACJ;EACJ,CAAC,EAAE,CAACP,wBAAwB,CAAC,CAAC;EAE9BrB,SAAS,CAAC,MAAM;IACZ,IAAIqB,wBAAwB,EAAEI,OAAO,EAAE;MACnCS,QAAQ,CAACI,gBAAgB,CAAC,WAAW,EAAEf,iBAAiB,CAAC;MACzDW,QAAQ,CAACI,gBAAgB,CAAC,SAAS,EAAED,eAAe,CAAC;MAErD,OAAO,MAAM;QACTH,QAAQ,CAACK,mBAAmB,CAAC,WAAW,EAAEhB,iBAAiB,CAAC;QAC5DW,QAAQ,CAACK,mBAAmB,CAAC,SAAS,EAAEF,eAAe,CAAC;MAC5D,CAAC;IACL;IACA;EACJ,CAAC,EAAE,CAAChB,wBAAwB,CAAC,CAAC;EAE9B,MAAMmB,+BAA+B,GAAGzC,WAAW,CAAC,MAAM;IACtD,MAAM0C,SAAS,GAAGtC,aAAa,CAAC,CAAC;IAEjC,MAAMuC,yBAAyB,GAAGrB,wBAAwB,CAACI,OAAO;IAClE,MAAMkB,eAAe,GAAGC,MAAM,CAACC,YAAY,CAAC,CAAC;IAE7C,IAAIH,yBAAyB,KAAK,IAAI,EAAE;MACpC;IACJ;IAEA,MAAMI,WAAW,GAAG5B,MAAM,CAAC6B,cAAc,CAAC,CAAC;IAC3C,IACIN,SAAS,KAAK,IAAI,IAClBE,eAAe,KAAK,IAAI,IACxB,CAACA,eAAe,CAACK,WAAW,IAC5BF,WAAW,KAAK,IAAI,IACpBA,WAAW,CAACV,QAAQ,CAACO,eAAe,CAACM,UAAU,CAAC,EAClD;MACE,MAAMC,SAAS,GAAGzC,eAAe,CAACkC,eAAe,EAAEG,WAAW,CAAC;MAE/DpC,uBAAuB,CAACwC,SAAS,EAAER,yBAAyB,EAAEzB,UAAU,CAAC;IAC7E;EACJ,CAAC,EAAE,CAACC,MAAM,EAAED,UAAU,CAAC,CAAC;EAExBjB,SAAS,CAAC,MAAM;IACZ,MAAMmD,YAAY,GAAGlC,UAAU,CAACmC,aAAa;IAE7C,MAAMC,MAAM,GAAGA,CAAA,KAAM;MACjBnC,MAAM,CAACoC,cAAc,CAAC,CAAC,CAACC,IAAI,CAAC,MAAM;QAC/Bf,+BAA+B,CAAC,CAAC;MACrC,CAAC,CAAC;IACN,CAAC;IAEDI,MAAM,CAACN,gBAAgB,CAAC,QAAQ,EAAEe,MAAM,CAAC;IACzC,IAAIF,YAAY,EAAE;MACdA,YAAY,CAACb,gBAAgB,CAAC,QAAQ,EAAEe,MAAM,CAAC;IACnD;IAEA,OAAO,MAAM;MACTT,MAAM,CAACL,mBAAmB,CAAC,QAAQ,EAAEc,MAAM,CAAC;MAC5C,IAAIF,YAAY,EAAE;QACdA,YAAY,CAACZ,mBAAmB,CAAC,QAAQ,EAAEc,MAAM,CAAC;MACtD;IACJ,CAAC;EACL,CAAC,EAAE,CAACnC,MAAM,EAAEsB,+BAA+B,EAAEvB,UAAU,CAAC,CAAC;EAEzDjB,SAAS,CAAC,MAAM;IACZkB,MAAM,CAACoC,cAAc,CAAC,CAAC,CAACC,IAAI,CAAC,MAAM;MAC/Bf,+BAA+B,CAAC,CAAC;IACrC,CAAC,CAAC;IACF,OAAOhC,aAAa,CAChBU,MAAM,CAACsC,sBAAsB,CAAC,CAAC;MAAEC;IAAY,CAAC,KAAK;MAC/CA,WAAW,CAACF,IAAI,CAAC,MAAM;QACnBf,+BAA+B,CAAC,CAAC;MACrC,CAAC,CAAC;IACN,CAAC,CAAC,EAEFtB,MAAM,CAACwC,eAAe,CAClB3C,qBAAqB,EACrB,MAAM;MACF4C,UAAU,CAAC,MAAM;QACbvC,YAAY,CAAC,KAAK,CAAC;MACvB,CAAC,EAAE,EAAE,CAAC;MACN,OAAO,IAAI;IACf,CAAC,EACDf,oBACJ,CAAC,EAEDa,MAAM,CAACwC,eAAe,CAClBpD,wBAAwB,EACxB,MAAM;MACFc,YAAY,CAAC,IAAI,CAAC;MAClBoB,+BAA+B,CAAC,CAAC;MACjC,OAAO,KAAK;IAChB,CAAC,EACDnC,oBACJ,CAAC,EAEDa,MAAM,CAACwC,eAAe,CAClBtD,YAAY,EACZwD,OAAO,IAAI;MACP,IAAI,CAAC9C,wBAAwB,CAAC8C,OAAO,CAACC,aAA4B,CAAC,EAAE;QACjEzC,YAAY,CAAC,KAAK,CAAC;MACvB;MAEA,OAAO,KAAK;IAChB,CAAC,EACDf,oBACJ,CACJ,CAAC;EACL,CAAC,EAAE,CAACa,MAAM,EAAEsB,+BAA+B,CAAC,CAAC;EAE7C,IAAI,CAACrB,SAAS,EAAE;IACZ,OAAO,IAAI;EACf;EAEA,oBACItB,KAAA,CAAAiE,aAAA;IAAKC,GAAG,EAAE1C,wBAAyB;IAAC2C,SAAS,EAAC;EAAkB,GAC3D9C,MAAM,CAAC+C,UAAU,CAAC,CAAC,iBAChBpE,KAAA,CAAAiE,aAAA,CAAAjE,KAAA,CAAAC,QAAA,QACKwB,eAAe,CAAC4C,GAAG,CAACC,MAAM,iBACvBtE,KAAA,CAAAiE,aAAA,CAAChE,QAAQ;IAACsE,GAAG,EAAED,MAAM,CAACE;EAAK,GAAEF,MAAM,CAACG,OAAkB,CACzD,CACH,CAEL,CAAC;AAEd,CAAC;AAMD,OAAO,MAAMC,OAAO,GAAGA,CAAC;EAAEtD,UAAU,GAAGiB,QAAQ,CAACsC;AAAmB,CAAC,KAAK;EACrE,MAAM;IAAEtD;EAAO,CAAC,GAAGL,iBAAiB,CAAC,CAAC;EACtC,MAAM4D,WAAW,GAAG7D,2BAA2B,CAAC,CAAC;IAAE8D;EAAe,CAAC,KAAK;IACpE,IAAI,CAACA,cAAc,EAAE;MACjB,OAAO,KAAK;IAChB;IAEA,OAAO,CAACA,cAAc,CAAC1B,WAAW,CAAC,CAAC;EACxC,CAAC,CAAC;EAEF,IAAI,CAACyB,WAAW,EAAE;IACd,OAAO,IAAI;EACf;EAEA,oBAAOlE,YAAY,cAACV,KAAA,CAAAiE,aAAA,CAAC9C,eAAe;IAACC,UAAU,EAAEA,UAAW;IAACC,MAAM,EAAEA;EAAO,CAAE,CAAC,EAAED,UAAU,CAAC;AAChG,CAAC","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ import React from "react";
2
+ export declare const BoldAction: () => React.JSX.Element;
@@ -0,0 +1,25 @@
1
+ import React from "react";
2
+ import { FORMAT_TEXT_COMMAND } from "lexical";
3
+ import { useCurrentSelection } from "../../hooks/useCurrentSelection";
4
+ import { useRichTextEditor } from "../../hooks";
5
+ export const BoldAction = () => {
6
+ const {
7
+ editor
8
+ } = useRichTextEditor();
9
+ const {
10
+ rangeSelection
11
+ } = useCurrentSelection();
12
+ const isBoldSelected = rangeSelection ? rangeSelection.hasFormat("bold") : false;
13
+ const handleClick = () => {
14
+ editor.dispatchCommand(FORMAT_TEXT_COMMAND, "bold");
15
+ };
16
+ return /*#__PURE__*/React.createElement("button", {
17
+ onClick: handleClick,
18
+ className: "popup-item spaced " + (isBoldSelected ? "active" : ""),
19
+ "aria-label": "Format text as bold"
20
+ }, /*#__PURE__*/React.createElement("i", {
21
+ className: "format bold"
22
+ }));
23
+ };
24
+
25
+ //# sourceMappingURL=BoldAction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","FORMAT_TEXT_COMMAND","useCurrentSelection","useRichTextEditor","BoldAction","editor","rangeSelection","isBoldSelected","hasFormat","handleClick","dispatchCommand","createElement","onClick","className"],"sources":["BoldAction.tsx"],"sourcesContent":["import React from \"react\";\nimport { FORMAT_TEXT_COMMAND } from \"lexical\";\nimport { useCurrentSelection } from \"~/hooks/useCurrentSelection\";\nimport { useRichTextEditor } from \"~/hooks\";\n\nexport const BoldAction = () => {\n const { editor } = useRichTextEditor();\n const { rangeSelection } = useCurrentSelection();\n const isBoldSelected = rangeSelection ? rangeSelection.hasFormat(\"bold\") : false;\n\n const handleClick = () => {\n editor.dispatchCommand(FORMAT_TEXT_COMMAND, \"bold\");\n };\n\n return (\n <button\n onClick={handleClick}\n className={\"popup-item spaced \" + (isBoldSelected ? \"active\" : \"\")}\n aria-label=\"Format text as bold\"\n >\n <i className=\"format bold\" />\n </button>\n );\n};\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AACzB,SAASC,mBAAmB,QAAQ,SAAS;AAC7C,SAASC,mBAAmB;AAC5B,SAASC,iBAAiB;AAE1B,OAAO,MAAMC,UAAU,GAAGA,CAAA,KAAM;EAC5B,MAAM;IAAEC;EAAO,CAAC,GAAGF,iBAAiB,CAAC,CAAC;EACtC,MAAM;IAAEG;EAAe,CAAC,GAAGJ,mBAAmB,CAAC,CAAC;EAChD,MAAMK,cAAc,GAAGD,cAAc,GAAGA,cAAc,CAACE,SAAS,CAAC,MAAM,CAAC,GAAG,KAAK;EAEhF,MAAMC,WAAW,GAAGA,CAAA,KAAM;IACtBJ,MAAM,CAACK,eAAe,CAACT,mBAAmB,EAAE,MAAM,CAAC;EACvD,CAAC;EAED,oBACID,KAAA,CAAAW,aAAA;IACIC,OAAO,EAAEH,WAAY;IACrBI,SAAS,EAAE,oBAAoB,IAAIN,cAAc,GAAG,QAAQ,GAAG,EAAE,CAAE;IACnE,cAAW;EAAqB,gBAEhCP,KAAA,CAAAW,aAAA;IAAGE,SAAS,EAAC;EAAa,CAAE,CACxB,CAAC;AAEjB,CAAC","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ import React from "react";
2
+ export declare const BulletListAction: () => React.JSX.Element;
@@ -0,0 +1,40 @@
1
+ import React from "react";
2
+ import { INSERT_UNORDERED_LIST_COMMAND, REMOVE_LIST_COMMAND } from "../../commands";
3
+ import { useRichTextEditor } from "../../hooks/useRichTextEditor";
4
+ import { findTypographyStyleByHtmlTag } from "@webiny/lexical-theme";
5
+ import { $isListNode } from "@webiny/lexical-nodes";
6
+ import { useCurrentElement } from "../../hooks/useCurrentElement";
7
+ export const BulletListAction = () => {
8
+ const {
9
+ editor
10
+ } = useRichTextEditor();
11
+ const {
12
+ element
13
+ } = useCurrentElement();
14
+ const {
15
+ themeEmotionMap
16
+ } = useRichTextEditor();
17
+ const isList = $isListNode(element);
18
+ const isBullet = isList && element.getListType() === "bullet";
19
+ const formatBulletList = () => {
20
+ if (!isBullet) {
21
+ const styleId = themeEmotionMap ? findTypographyStyleByHtmlTag("ul", themeEmotionMap)?.id : undefined;
22
+
23
+ // will update the active state in the useEffect
24
+ editor.dispatchCommand(INSERT_UNORDERED_LIST_COMMAND, {
25
+ themeStyleId: styleId
26
+ });
27
+ } else {
28
+ editor.dispatchCommand(REMOVE_LIST_COMMAND, undefined);
29
+ }
30
+ };
31
+ return /*#__PURE__*/React.createElement("button", {
32
+ onClick: () => formatBulletList(),
33
+ className: "popup-item spaced " + (isBullet ? "active" : ""),
34
+ "aria-label": "Format text as bullet list"
35
+ }, /*#__PURE__*/React.createElement("i", {
36
+ className: "icon bullet-list"
37
+ }));
38
+ };
39
+
40
+ //# sourceMappingURL=BulletListAction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","INSERT_UNORDERED_LIST_COMMAND","REMOVE_LIST_COMMAND","useRichTextEditor","findTypographyStyleByHtmlTag","$isListNode","useCurrentElement","BulletListAction","editor","element","themeEmotionMap","isList","isBullet","getListType","formatBulletList","styleId","id","undefined","dispatchCommand","themeStyleId","createElement","onClick","className"],"sources":["BulletListAction.tsx"],"sourcesContent":["import React from \"react\";\nimport { INSERT_UNORDERED_LIST_COMMAND, REMOVE_LIST_COMMAND } from \"~/commands\";\nimport { useRichTextEditor } from \"~/hooks/useRichTextEditor\";\nimport { findTypographyStyleByHtmlTag } from \"@webiny/lexical-theme\";\nimport type { ListNode } from \"@webiny/lexical-nodes\";\nimport { $isListNode } from \"@webiny/lexical-nodes\";\nimport { useCurrentElement } from \"~/hooks/useCurrentElement\";\n\nexport const BulletListAction = () => {\n const { editor } = useRichTextEditor();\n const { element } = useCurrentElement();\n const { themeEmotionMap } = useRichTextEditor();\n const isList = $isListNode(element);\n\n const isBullet = isList && (element as ListNode).getListType() === \"bullet\";\n\n const formatBulletList = () => {\n if (!isBullet) {\n const styleId = themeEmotionMap\n ? findTypographyStyleByHtmlTag(\"ul\", themeEmotionMap)?.id\n : undefined;\n\n // will update the active state in the useEffect\n editor.dispatchCommand(INSERT_UNORDERED_LIST_COMMAND, { themeStyleId: styleId });\n } else {\n editor.dispatchCommand(REMOVE_LIST_COMMAND, undefined);\n }\n };\n\n return (\n <button\n onClick={() => formatBulletList()}\n className={\"popup-item spaced \" + (isBullet ? \"active\" : \"\")}\n aria-label=\"Format text as bullet list\"\n >\n <i className=\"icon bullet-list\" />\n </button>\n );\n};\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AACzB,SAASC,6BAA6B,EAAEC,mBAAmB;AAC3D,SAASC,iBAAiB;AAC1B,SAASC,4BAA4B,QAAQ,uBAAuB;AAEpE,SAASC,WAAW,QAAQ,uBAAuB;AACnD,SAASC,iBAAiB;AAE1B,OAAO,MAAMC,gBAAgB,GAAGA,CAAA,KAAM;EAClC,MAAM;IAAEC;EAAO,CAAC,GAAGL,iBAAiB,CAAC,CAAC;EACtC,MAAM;IAAEM;EAAQ,CAAC,GAAGH,iBAAiB,CAAC,CAAC;EACvC,MAAM;IAAEI;EAAgB,CAAC,GAAGP,iBAAiB,CAAC,CAAC;EAC/C,MAAMQ,MAAM,GAAGN,WAAW,CAACI,OAAO,CAAC;EAEnC,MAAMG,QAAQ,GAAGD,MAAM,IAAKF,OAAO,CAAcI,WAAW,CAAC,CAAC,KAAK,QAAQ;EAE3E,MAAMC,gBAAgB,GAAGA,CAAA,KAAM;IAC3B,IAAI,CAACF,QAAQ,EAAE;MACX,MAAMG,OAAO,GAAGL,eAAe,GACzBN,4BAA4B,CAAC,IAAI,EAAEM,eAAe,CAAC,EAAEM,EAAE,GACvDC,SAAS;;MAEf;MACAT,MAAM,CAACU,eAAe,CAACjB,6BAA6B,EAAE;QAAEkB,YAAY,EAAEJ;MAAQ,CAAC,CAAC;IACpF,CAAC,MAAM;MACHP,MAAM,CAACU,eAAe,CAAChB,mBAAmB,EAAEe,SAAS,CAAC;IAC1D;EACJ,CAAC;EAED,oBACIjB,KAAA,CAAAoB,aAAA;IACIC,OAAO,EAAEA,CAAA,KAAMP,gBAAgB,CAAC,CAAE;IAClCQ,SAAS,EAAE,oBAAoB,IAAIV,QAAQ,GAAG,QAAQ,GAAG,EAAE,CAAE;IAC7D,cAAW;EAA4B,gBAEvCZ,KAAA,CAAAoB,aAAA;IAAGE,SAAS,EAAC;EAAkB,CAAE,CAC7B,CAAC;AAEjB,CAAC","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ import React from "react";
2
+ export declare const CodeHighlightAction: () => React.JSX.Element;
@@ -0,0 +1,25 @@
1
+ import React from "react";
2
+ import { FORMAT_TEXT_COMMAND } from "lexical";
3
+ import { useCurrentSelection } from "../../hooks/useCurrentSelection";
4
+ import { useRichTextEditor } from "../../hooks";
5
+ export const CodeHighlightAction = () => {
6
+ const {
7
+ editor
8
+ } = useRichTextEditor();
9
+ const {
10
+ rangeSelection
11
+ } = useCurrentSelection();
12
+ const isCodeSelected = rangeSelection ? rangeSelection.hasFormat("code") : false;
13
+ const handleClick = () => {
14
+ editor.dispatchCommand(FORMAT_TEXT_COMMAND, "code");
15
+ };
16
+ return /*#__PURE__*/React.createElement("button", {
17
+ onClick: handleClick,
18
+ className: "popup-item spaced " + (isCodeSelected ? "active" : ""),
19
+ "aria-label": "Text code highlight"
20
+ }, /*#__PURE__*/React.createElement("i", {
21
+ className: "format code"
22
+ }));
23
+ };
24
+
25
+ //# sourceMappingURL=CodeHighlightAction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","FORMAT_TEXT_COMMAND","useCurrentSelection","useRichTextEditor","CodeHighlightAction","editor","rangeSelection","isCodeSelected","hasFormat","handleClick","dispatchCommand","createElement","onClick","className"],"sources":["CodeHighlightAction.tsx"],"sourcesContent":["import React from \"react\";\nimport { FORMAT_TEXT_COMMAND } from \"lexical\";\nimport { useCurrentSelection } from \"~/hooks/useCurrentSelection\";\nimport { useRichTextEditor } from \"~/hooks\";\n\nexport const CodeHighlightAction = () => {\n const { editor } = useRichTextEditor();\n const { rangeSelection } = useCurrentSelection();\n const isCodeSelected = rangeSelection ? rangeSelection.hasFormat(\"code\") : false;\n\n const handleClick = () => {\n editor.dispatchCommand(FORMAT_TEXT_COMMAND, \"code\");\n };\n\n return (\n <button\n onClick={handleClick}\n className={\"popup-item spaced \" + (isCodeSelected ? \"active\" : \"\")}\n aria-label=\"Text code highlight\"\n >\n <i className=\"format code\" />\n </button>\n );\n};\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AACzB,SAASC,mBAAmB,QAAQ,SAAS;AAC7C,SAASC,mBAAmB;AAC5B,SAASC,iBAAiB;AAE1B,OAAO,MAAMC,mBAAmB,GAAGA,CAAA,KAAM;EACrC,MAAM;IAAEC;EAAO,CAAC,GAAGF,iBAAiB,CAAC,CAAC;EACtC,MAAM;IAAEG;EAAe,CAAC,GAAGJ,mBAAmB,CAAC,CAAC;EAChD,MAAMK,cAAc,GAAGD,cAAc,GAAGA,cAAc,CAACE,SAAS,CAAC,MAAM,CAAC,GAAG,KAAK;EAEhF,MAAMC,WAAW,GAAGA,CAAA,KAAM;IACtBJ,MAAM,CAACK,eAAe,CAACT,mBAAmB,EAAE,MAAM,CAAC;EACvD,CAAC;EAED,oBACID,KAAA,CAAAW,aAAA;IACIC,OAAO,EAAEH,WAAY;IACrBI,SAAS,EAAE,oBAAoB,IAAIN,cAAc,GAAG,QAAQ,GAAG,EAAE,CAAE;IACnE,cAAW;EAAqB,gBAEhCP,KAAA,CAAAW,aAAA;IAAGE,SAAS,EAAC;EAAa,CAAE,CACxB,CAAC;AAEjB,CAAC","ignoreList":[]}
@@ -0,0 +1,29 @@
1
+ import React from "react";
2
+ export declare const FontColorPicker: (() => JSX.Element | null) & {
3
+ original: () => JSX.Element | null;
4
+ originalName: string;
5
+ displayName: string;
6
+ } & {
7
+ original: (() => JSX.Element | null) & {
8
+ original: () => JSX.Element | null;
9
+ originalName: string;
10
+ displayName: string;
11
+ };
12
+ originalName: string;
13
+ displayName: string;
14
+ } & {
15
+ createDecorator: (decorator: import("@webiny/react-composition").ComponentDecorator<(() => JSX.Element | null) & {
16
+ original: () => JSX.Element | null;
17
+ originalName: string;
18
+ displayName: string;
19
+ }>) => (props: unknown) => React.JSX.Element;
20
+ };
21
+ interface FontActionColorPicker {
22
+ element: JSX.Element;
23
+ }
24
+ declare const FontActionColorPicker: ({ element }: FontActionColorPicker) => JSX.Element;
25
+ export type FontColorAction = React.ComponentType<unknown> & {
26
+ ColorPicker: typeof FontActionColorPicker;
27
+ };
28
+ export declare const FontColorAction: FontColorAction;
29
+ export {};
@@ -0,0 +1,50 @@
1
+ import React, { useCallback, useEffect, useMemo } from "react";
2
+ import { Compose, makeDecoratable } from "@webiny/react-composition";
3
+ import { FontColorActionContext } from "../../context/FontColorActionContext";
4
+ import { $isFontColorNode, ADD_FONT_COLOR_COMMAND, ThemeColorValue } from "@webiny/lexical-nodes";
5
+ import { getSelectedNode } from "../../utils/getSelectedNode";
6
+ import { useDeriveValueFromSelection } from "../../hooks/useCurrentSelection";
7
+ import { useRichTextEditor } from "../../hooks";
8
+ export const FontColorPicker = makeDecoratable("FontColorPicker", () => {
9
+ useEffect(() => {
10
+ console.log("Default FontColorPicker, please add your own component");
11
+ }, []);
12
+ return null;
13
+ });
14
+ const FontActionColorPicker = ({
15
+ element
16
+ }) => {
17
+ return /*#__PURE__*/React.createElement(Compose, {
18
+ component: FontColorPicker,
19
+ with: () => () => element
20
+ });
21
+ };
22
+ export const FontColorAction = () => {
23
+ const {
24
+ editor
25
+ } = useRichTextEditor();
26
+ const fontColor = useDeriveValueFromSelection(({
27
+ rangeSelection
28
+ }) => {
29
+ if (!rangeSelection) {
30
+ return "#000";
31
+ }
32
+ const node = getSelectedNode(rangeSelection);
33
+ return $isFontColorNode(node) ? node.getColorStyle().color : "#000";
34
+ });
35
+ const onFontColorSelect = useCallback((colorValue, themeColorName) => {
36
+ editor.dispatchCommand(ADD_FONT_COLOR_COMMAND, {
37
+ color: new ThemeColorValue(colorValue, themeColorName)
38
+ });
39
+ }, []);
40
+ const context = useMemo(() => ({
41
+ value: fontColor,
42
+ applyColor: onFontColorSelect
43
+ }), [onFontColorSelect, fontColor]);
44
+ return /*#__PURE__*/React.createElement(FontColorActionContext.Provider, {
45
+ value: context
46
+ }, /*#__PURE__*/React.createElement(FontColorPicker, null));
47
+ };
48
+ FontColorAction.ColorPicker = FontActionColorPicker;
49
+
50
+ //# sourceMappingURL=FontColorAction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","useCallback","useEffect","useMemo","Compose","makeDecoratable","FontColorActionContext","$isFontColorNode","ADD_FONT_COLOR_COMMAND","ThemeColorValue","getSelectedNode","useDeriveValueFromSelection","useRichTextEditor","FontColorPicker","console","log","FontActionColorPicker","element","createElement","component","with","FontColorAction","editor","fontColor","rangeSelection","node","getColorStyle","color","onFontColorSelect","colorValue","themeColorName","dispatchCommand","context","value","applyColor","Provider","ColorPicker"],"sources":["FontColorAction.tsx"],"sourcesContent":["import React, { useCallback, useEffect, useMemo } from \"react\";\nimport type { LexicalCommand } from \"lexical\";\nimport { Compose, makeDecoratable } from \"@webiny/react-composition\";\nimport { FontColorActionContext } from \"~/context/FontColorActionContext\";\nimport type { FontColorPayload } from \"@webiny/lexical-nodes\";\nimport { $isFontColorNode, ADD_FONT_COLOR_COMMAND, ThemeColorValue } from \"@webiny/lexical-nodes\";\nimport { getSelectedNode } from \"~/utils/getSelectedNode\";\nimport { useDeriveValueFromSelection } from \"~/hooks/useCurrentSelection\";\nimport { useRichTextEditor } from \"~/hooks\";\n\nexport const FontColorPicker = makeDecoratable(\"FontColorPicker\", (): JSX.Element | null => {\n useEffect(() => {\n console.log(\"Default FontColorPicker, please add your own component\");\n }, []);\n return null;\n});\n\ninterface FontActionColorPicker {\n element: JSX.Element;\n}\n\nconst FontActionColorPicker = ({ element }: FontActionColorPicker): JSX.Element => {\n return <Compose component={FontColorPicker} with={() => () => element} />;\n};\n\nexport type FontColorAction = React.ComponentType<unknown> & {\n ColorPicker: typeof FontActionColorPicker;\n};\n\nexport const FontColorAction: FontColorAction = () => {\n const { editor } = useRichTextEditor();\n const fontColor = useDeriveValueFromSelection(({ rangeSelection }) => {\n if (!rangeSelection) {\n return \"#000\";\n }\n\n const node = getSelectedNode(rangeSelection);\n return $isFontColorNode(node) ? node.getColorStyle().color : \"#000\";\n });\n\n const onFontColorSelect = useCallback(\n (colorValue: string, themeColorName: string | undefined) => {\n editor.dispatchCommand<LexicalCommand<FontColorPayload>>(ADD_FONT_COLOR_COMMAND, {\n color: new ThemeColorValue(colorValue, themeColorName)\n });\n },\n []\n );\n\n const context = useMemo(\n () => ({\n value: fontColor,\n applyColor: onFontColorSelect\n }),\n [onFontColorSelect, fontColor]\n );\n\n return (\n <FontColorActionContext.Provider value={context}>\n <FontColorPicker />\n </FontColorActionContext.Provider>\n );\n};\n\nFontColorAction.ColorPicker = FontActionColorPicker;\n"],"mappings":"AAAA,OAAOA,KAAK,IAAIC,WAAW,EAAEC,SAAS,EAAEC,OAAO,QAAQ,OAAO;AAE9D,SAASC,OAAO,EAAEC,eAAe,QAAQ,2BAA2B;AACpE,SAASC,sBAAsB;AAE/B,SAASC,gBAAgB,EAAEC,sBAAsB,EAAEC,eAAe,QAAQ,uBAAuB;AACjG,SAASC,eAAe;AACxB,SAASC,2BAA2B;AACpC,SAASC,iBAAiB;AAE1B,OAAO,MAAMC,eAAe,GAAGR,eAAe,CAAC,iBAAiB,EAAE,MAA0B;EACxFH,SAAS,CAAC,MAAM;IACZY,OAAO,CAACC,GAAG,CAAC,wDAAwD,CAAC;EACzE,CAAC,EAAE,EAAE,CAAC;EACN,OAAO,IAAI;AACf,CAAC,CAAC;AAMF,MAAMC,qBAAqB,GAAGA,CAAC;EAAEC;AAA+B,CAAC,KAAkB;EAC/E,oBAAOjB,KAAA,CAAAkB,aAAA,CAACd,OAAO;IAACe,SAAS,EAAEN,eAAgB;IAACO,IAAI,EAAEA,CAAA,KAAM,MAAMH;EAAQ,CAAE,CAAC;AAC7E,CAAC;AAMD,OAAO,MAAMI,eAAgC,GAAGA,CAAA,KAAM;EAClD,MAAM;IAAEC;EAAO,CAAC,GAAGV,iBAAiB,CAAC,CAAC;EACtC,MAAMW,SAAS,GAAGZ,2BAA2B,CAAC,CAAC;IAAEa;EAAe,CAAC,KAAK;IAClE,IAAI,CAACA,cAAc,EAAE;MACjB,OAAO,MAAM;IACjB;IAEA,MAAMC,IAAI,GAAGf,eAAe,CAACc,cAAc,CAAC;IAC5C,OAAOjB,gBAAgB,CAACkB,IAAI,CAAC,GAAGA,IAAI,CAACC,aAAa,CAAC,CAAC,CAACC,KAAK,GAAG,MAAM;EACvE,CAAC,CAAC;EAEF,MAAMC,iBAAiB,GAAG3B,WAAW,CACjC,CAAC4B,UAAkB,EAAEC,cAAkC,KAAK;IACxDR,MAAM,CAACS,eAAe,CAAmCvB,sBAAsB,EAAE;MAC7EmB,KAAK,EAAE,IAAIlB,eAAe,CAACoB,UAAU,EAAEC,cAAc;IACzD,CAAC,CAAC;EACN,CAAC,EACD,EACJ,CAAC;EAED,MAAME,OAAO,GAAG7B,OAAO,CACnB,OAAO;IACH8B,KAAK,EAAEV,SAAS;IAChBW,UAAU,EAAEN;EAChB,CAAC,CAAC,EACF,CAACA,iBAAiB,EAAEL,SAAS,CACjC,CAAC;EAED,oBACIvB,KAAA,CAAAkB,aAAA,CAACZ,sBAAsB,CAAC6B,QAAQ;IAACF,KAAK,EAAED;EAAQ,gBAC5ChC,KAAA,CAAAkB,aAAA,CAACL,eAAe,MAAE,CACW,CAAC;AAE1C,CAAC;AAEDQ,eAAe,CAACe,WAAW,GAAGpB,qBAAqB","ignoreList":[]}
@@ -0,0 +1,14 @@
1
+ import React from "react";
2
+ export interface FontSize {
3
+ id: string;
4
+ name: string;
5
+ value: string;
6
+ }
7
+ export declare const FONT_SIZES_FALLBACK: FontSize[];
8
+ interface FontSizeActionProps {
9
+ fontSizes?: FontSize[];
10
+ }
11
+ export declare const FontSizeAction: (({ fontSizes }: FontSizeActionProps) => React.JSX.Element) & {
12
+ FONT_SIZES_FALLBACK: FontSize[];
13
+ };
14
+ export {};
@@ -0,0 +1,91 @@
1
+ import React, { useCallback, useEffect, useState } from "react";
2
+ import { $getSelectionStyleValueForProperty, $patchStyleText } from "@lexical/selection";
3
+ import { mergeRegister } from "@lexical/utils";
4
+ import { $getSelection, $isRangeSelection } from "lexical";
5
+ import { DropDown, DropDownItem } from "../../ui/DropDown";
6
+ import { useDeriveValueFromSelection } from "../../hooks/useCurrentSelection";
7
+ import { useRichTextEditor } from "../../hooks";
8
+ export const FONT_SIZES_FALLBACK = ["8px", "9px", "10px", "11px", "12px", "14px", "15px", "16px", "18px", "21px", "24px", "30px", "36px", "48px", "60px", "72px", "96px"].map(size => ({
9
+ id: size,
10
+ name: size,
11
+ value: size,
12
+ default: size === "15px"
13
+ }));
14
+ const emptyOption = {
15
+ value: "",
16
+ name: "Font Size",
17
+ id: "empty"
18
+ };
19
+ function dropDownActiveClass(active) {
20
+ if (active) {
21
+ return "active dropdown-item-active";
22
+ }
23
+ return "";
24
+ }
25
+ function FontSizeDropDown(props) {
26
+ const {
27
+ editor,
28
+ value,
29
+ fontSizes,
30
+ disabled = false
31
+ } = props;
32
+ const handleClick = useCallback(option => {
33
+ editor.update(() => {
34
+ const selection = $getSelection();
35
+ if ($isRangeSelection(selection)) {
36
+ $patchStyleText(selection, {
37
+ ["font-size"]: option.value
38
+ });
39
+ }
40
+ });
41
+ }, [editor]);
42
+ const selectedOption = fontSizes.find(opt => opt.value === value) ?? emptyOption;
43
+ return /*#__PURE__*/React.createElement(DropDown, {
44
+ disabled: disabled,
45
+ buttonClassName: "toolbar-item font-size",
46
+ buttonLabel: selectedOption.name,
47
+ buttonAriaLabel: "Formatting options for font size"
48
+ }, fontSizes.map(option => /*#__PURE__*/React.createElement(DropDownItem, {
49
+ className: `item fontsize-item ${dropDownActiveClass(value === option.id)}`,
50
+ onClick: () => handleClick(option),
51
+ key: option.id
52
+ }, /*#__PURE__*/React.createElement("span", {
53
+ className: "text"
54
+ }, option.name))));
55
+ }
56
+ const FontSize = ({
57
+ fontSizes = FONT_SIZES_FALLBACK
58
+ }) => {
59
+ const {
60
+ editor
61
+ } = useRichTextEditor();
62
+ const [isEditable, setIsEditable] = useState(() => editor.isEditable());
63
+ const fontSize = useDeriveValueFromSelection(({
64
+ rangeSelection
65
+ }) => {
66
+ if (!rangeSelection) {
67
+ return undefined;
68
+ }
69
+ try {
70
+ return $getSelectionStyleValueForProperty(rangeSelection, "font-size");
71
+ } catch {
72
+ return undefined;
73
+ }
74
+ });
75
+ useEffect(() => {
76
+ return mergeRegister(editor.registerEditableListener(editable => {
77
+ setIsEditable(editable);
78
+ }));
79
+ }, [editor]);
80
+ return /*#__PURE__*/React.createElement(React.Fragment, null, /*#__PURE__*/React.createElement(FontSizeDropDown, {
81
+ disabled: !isEditable,
82
+ value: fontSize,
83
+ editor: editor,
84
+ fontSizes: fontSizes
85
+ }));
86
+ };
87
+ export const FontSizeAction = Object.assign(FontSize, {
88
+ FONT_SIZES_FALLBACK
89
+ });
90
+
91
+ //# sourceMappingURL=FontSizeAction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","useCallback","useEffect","useState","$getSelectionStyleValueForProperty","$patchStyleText","mergeRegister","$getSelection","$isRangeSelection","DropDown","DropDownItem","useDeriveValueFromSelection","useRichTextEditor","FONT_SIZES_FALLBACK","map","size","id","name","value","default","emptyOption","dropDownActiveClass","active","FontSizeDropDown","props","editor","fontSizes","disabled","handleClick","option","update","selection","selectedOption","find","opt","createElement","buttonClassName","buttonLabel","buttonAriaLabel","className","onClick","key","FontSize","isEditable","setIsEditable","fontSize","rangeSelection","undefined","registerEditableListener","editable","Fragment","FontSizeAction","Object","assign"],"sources":["FontSizeAction.tsx"],"sourcesContent":["import React, { useCallback, useEffect, useState } from \"react\";\nimport { $getSelectionStyleValueForProperty, $patchStyleText } from \"@lexical/selection\";\nimport { mergeRegister } from \"@lexical/utils\";\nimport type { LexicalEditor } from \"lexical\";\nimport { $getSelection, $isRangeSelection } from \"lexical\";\nimport { DropDown, DropDownItem } from \"~/ui/DropDown\";\nimport { useDeriveValueFromSelection } from \"~/hooks/useCurrentSelection\";\nimport { useRichTextEditor } from \"~/hooks\";\n\nexport interface FontSize {\n id: string;\n name: string;\n value: string;\n}\n\nexport const FONT_SIZES_FALLBACK: FontSize[] = [\n \"8px\",\n \"9px\",\n \"10px\",\n \"11px\",\n \"12px\",\n \"14px\",\n \"15px\",\n \"16px\",\n \"18px\",\n \"21px\",\n \"24px\",\n \"30px\",\n \"36px\",\n \"48px\",\n \"60px\",\n \"72px\",\n \"96px\"\n].map(size => ({\n id: size,\n name: size,\n value: size,\n default: size === \"15px\"\n}));\n\nconst emptyOption: FontSize = {\n value: \"\",\n name: \"Font Size\",\n id: \"empty\"\n};\n\nfunction dropDownActiveClass(active: boolean) {\n if (active) {\n return \"active dropdown-item-active\";\n }\n return \"\";\n}\n\ninterface FontSizeDropDownProps {\n fontSizes: FontSize[];\n editor: LexicalEditor;\n value: string | undefined;\n disabled?: boolean;\n}\n\nfunction FontSizeDropDown(props: FontSizeDropDownProps): JSX.Element {\n const { editor, value, fontSizes, disabled = false } = props;\n\n const handleClick = useCallback(\n (option: FontSize) => {\n editor.update(() => {\n const selection = $getSelection();\n if ($isRangeSelection(selection)) {\n $patchStyleText(selection, {\n [\"font-size\"]: option.value\n });\n }\n });\n },\n [editor]\n );\n\n const selectedOption = fontSizes.find(opt => opt.value === value) ?? emptyOption;\n\n return (\n <DropDown\n disabled={disabled}\n buttonClassName=\"toolbar-item font-size\"\n buttonLabel={selectedOption.name}\n buttonAriaLabel={\"Formatting options for font size\"}\n >\n {fontSizes.map(option => (\n <DropDownItem\n className={`item fontsize-item ${dropDownActiveClass(value === option.id)}`}\n onClick={() => handleClick(option)}\n key={option.id}\n >\n <span className=\"text\">{option.name}</span>\n </DropDownItem>\n ))}\n </DropDown>\n );\n}\n\ninterface FontSizeActionProps {\n fontSizes?: FontSize[];\n}\n\nconst FontSize = ({ fontSizes = FONT_SIZES_FALLBACK }: FontSizeActionProps) => {\n const { editor } = useRichTextEditor();\n const [isEditable, setIsEditable] = useState(() => editor.isEditable());\n const fontSize = useDeriveValueFromSelection(({ rangeSelection }) => {\n if (!rangeSelection) {\n return undefined;\n }\n try {\n return $getSelectionStyleValueForProperty(rangeSelection, \"font-size\");\n } catch {\n return undefined;\n }\n });\n\n useEffect(() => {\n return mergeRegister(\n editor.registerEditableListener(editable => {\n setIsEditable(editable);\n })\n );\n }, [editor]);\n\n return (\n <>\n <FontSizeDropDown\n disabled={!isEditable}\n value={fontSize}\n editor={editor}\n fontSizes={fontSizes}\n />\n </>\n );\n};\n\nexport const FontSizeAction = Object.assign(FontSize, { FONT_SIZES_FALLBACK });\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ import React from "react";
2
+ export declare const ImageAction: () => React.JSX.Element;
@@ -0,0 +1,37 @@
1
+ import React, { useEffect, useState } from "react";
2
+ import { useRichTextEditor } from "../../hooks/useRichTextEditor";
3
+ import { fileToImagePayload } from "../../utils/files";
4
+ import { INSERT_IMAGE_COMMAND } from "../../commands";
5
+ export const ImageAction = () => {
6
+ const {
7
+ editor,
8
+ toolbarActionPlugins
9
+ } = useRichTextEditor();
10
+ const [imageActionPlugin, setImageActionPlugin] = useState();
11
+ useEffect(() => {
12
+ if (!!toolbarActionPlugins?.length) {
13
+ const actionPlugin = toolbarActionPlugins.find(action => action.targetAction === "image-action");
14
+ setImageActionPlugin(actionPlugin);
15
+ }
16
+ }, [toolbarActionPlugins]);
17
+ const handleClick = () => {
18
+ if (typeof imageActionPlugin?.plugin === "function") {
19
+ const cb = data => {
20
+ const imagePayload = fileToImagePayload(data);
21
+ if (imagePayload) {
22
+ editor.dispatchCommand(INSERT_IMAGE_COMMAND, imagePayload);
23
+ }
24
+ };
25
+ imageActionPlugin.plugin(cb);
26
+ }
27
+ };
28
+ return /*#__PURE__*/React.createElement("button", {
29
+ onClick: () => handleClick(),
30
+ className: "popup-item",
31
+ "aria-label": "Insert image"
32
+ }, /*#__PURE__*/React.createElement("i", {
33
+ className: "icon insert-image"
34
+ }));
35
+ };
36
+
37
+ //# sourceMappingURL=ImageAction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","useEffect","useState","useRichTextEditor","fileToImagePayload","INSERT_IMAGE_COMMAND","ImageAction","editor","toolbarActionPlugins","imageActionPlugin","setImageActionPlugin","length","actionPlugin","find","action","targetAction","handleClick","plugin","cb","data","imagePayload","dispatchCommand","createElement","onClick","className"],"sources":["ImageAction.tsx"],"sourcesContent":["import React, { useEffect, useState } from \"react\";\nimport type { LexicalCommand } from \"lexical\";\nimport { useRichTextEditor } from \"~/hooks/useRichTextEditor\";\nimport type { FileManagerFileItem } from \"~/utils/files\";\nimport { fileToImagePayload } from \"~/utils/files\";\nimport type { ImagePayload } from \"~/commands\";\nimport { INSERT_IMAGE_COMMAND } from \"~/commands\";\nimport type { ToolbarActionPlugin } from \"~/types\";\n\nexport const ImageAction = () => {\n const { editor, toolbarActionPlugins } = useRichTextEditor();\n const [imageActionPlugin, setImageActionPlugin] = useState<ToolbarActionPlugin | undefined>();\n\n useEffect(() => {\n if (!!toolbarActionPlugins?.length) {\n const actionPlugin = toolbarActionPlugins.find(\n action => action.targetAction === \"image-action\"\n );\n setImageActionPlugin(actionPlugin);\n }\n }, [toolbarActionPlugins]);\n\n const handleClick = () => {\n if (typeof imageActionPlugin?.plugin === \"function\") {\n const cb = (data: FileManagerFileItem) => {\n const imagePayload = fileToImagePayload(data);\n if (imagePayload) {\n editor.dispatchCommand<LexicalCommand<ImagePayload>>(\n INSERT_IMAGE_COMMAND,\n imagePayload\n );\n }\n };\n imageActionPlugin.plugin(cb);\n }\n };\n\n return (\n <button onClick={() => handleClick()} className={\"popup-item\"} aria-label=\"Insert image\">\n <i className=\"icon insert-image\" />\n </button>\n );\n};\n"],"mappings":"AAAA,OAAOA,KAAK,IAAIC,SAAS,EAAEC,QAAQ,QAAQ,OAAO;AAElD,SAASC,iBAAiB;AAE1B,SAASC,kBAAkB;AAE3B,SAASC,oBAAoB;AAG7B,OAAO,MAAMC,WAAW,GAAGA,CAAA,KAAM;EAC7B,MAAM;IAAEC,MAAM;IAAEC;EAAqB,CAAC,GAAGL,iBAAiB,CAAC,CAAC;EAC5D,MAAM,CAACM,iBAAiB,EAAEC,oBAAoB,CAAC,GAAGR,QAAQ,CAAkC,CAAC;EAE7FD,SAAS,CAAC,MAAM;IACZ,IAAI,CAAC,CAACO,oBAAoB,EAAEG,MAAM,EAAE;MAChC,MAAMC,YAAY,GAAGJ,oBAAoB,CAACK,IAAI,CAC1CC,MAAM,IAAIA,MAAM,CAACC,YAAY,KAAK,cACtC,CAAC;MACDL,oBAAoB,CAACE,YAAY,CAAC;IACtC;EACJ,CAAC,EAAE,CAACJ,oBAAoB,CAAC,CAAC;EAE1B,MAAMQ,WAAW,GAAGA,CAAA,KAAM;IACtB,IAAI,OAAOP,iBAAiB,EAAEQ,MAAM,KAAK,UAAU,EAAE;MACjD,MAAMC,EAAE,GAAIC,IAAyB,IAAK;QACtC,MAAMC,YAAY,GAAGhB,kBAAkB,CAACe,IAAI,CAAC;QAC7C,IAAIC,YAAY,EAAE;UACdb,MAAM,CAACc,eAAe,CAClBhB,oBAAoB,EACpBe,YACJ,CAAC;QACL;MACJ,CAAC;MACDX,iBAAiB,CAACQ,MAAM,CAACC,EAAE,CAAC;IAChC;EACJ,CAAC;EAED,oBACIlB,KAAA,CAAAsB,aAAA;IAAQC,OAAO,EAAEA,CAAA,KAAMP,WAAW,CAAC,CAAE;IAACQ,SAAS,EAAE,YAAa;IAAC,cAAW;EAAc,gBACpFxB,KAAA,CAAAsB,aAAA;IAAGE,SAAS,EAAC;EAAmB,CAAE,CAC9B,CAAC;AAEjB,CAAC","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ import React from "react";
2
+ export declare const ItalicAction: () => React.JSX.Element;
@@ -0,0 +1,25 @@
1
+ import React from "react";
2
+ import { FORMAT_TEXT_COMMAND } from "lexical";
3
+ import { useCurrentSelection } from "../../hooks/useCurrentSelection";
4
+ import { useRichTextEditor } from "../../hooks";
5
+ export const ItalicAction = () => {
6
+ const {
7
+ editor
8
+ } = useRichTextEditor();
9
+ const {
10
+ rangeSelection
11
+ } = useCurrentSelection();
12
+ const isItalicSelected = rangeSelection ? rangeSelection.hasFormat("italic") : false;
13
+ const handleClick = () => {
14
+ editor.dispatchCommand(FORMAT_TEXT_COMMAND, "italic");
15
+ };
16
+ return /*#__PURE__*/React.createElement("button", {
17
+ onClick: handleClick,
18
+ className: "popup-item spaced " + (isItalicSelected ? "active" : ""),
19
+ "aria-label": "Format text as italic"
20
+ }, /*#__PURE__*/React.createElement("i", {
21
+ className: "format italic"
22
+ }));
23
+ };
24
+
25
+ //# sourceMappingURL=ItalicAction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["React","FORMAT_TEXT_COMMAND","useCurrentSelection","useRichTextEditor","ItalicAction","editor","rangeSelection","isItalicSelected","hasFormat","handleClick","dispatchCommand","createElement","onClick","className"],"sources":["ItalicAction.tsx"],"sourcesContent":["import React from \"react\";\nimport { FORMAT_TEXT_COMMAND } from \"lexical\";\nimport { useCurrentSelection } from \"~/hooks/useCurrentSelection\";\nimport { useRichTextEditor } from \"~/hooks\";\n\nexport const ItalicAction = () => {\n const { editor } = useRichTextEditor();\n const { rangeSelection } = useCurrentSelection();\n const isItalicSelected = rangeSelection ? rangeSelection.hasFormat(\"italic\") : false;\n\n const handleClick = () => {\n editor.dispatchCommand(FORMAT_TEXT_COMMAND, \"italic\");\n };\n\n return (\n <button\n onClick={handleClick}\n className={\"popup-item spaced \" + (isItalicSelected ? \"active\" : \"\")}\n aria-label=\"Format text as italic\"\n >\n <i className=\"format italic\" />\n </button>\n );\n};\n"],"mappings":"AAAA,OAAOA,KAAK,MAAM,OAAO;AACzB,SAASC,mBAAmB,QAAQ,SAAS;AAC7C,SAASC,mBAAmB;AAC5B,SAASC,iBAAiB;AAE1B,OAAO,MAAMC,YAAY,GAAGA,CAAA,KAAM;EAC9B,MAAM;IAAEC;EAAO,CAAC,GAAGF,iBAAiB,CAAC,CAAC;EACtC,MAAM;IAAEG;EAAe,CAAC,GAAGJ,mBAAmB,CAAC,CAAC;EAChD,MAAMK,gBAAgB,GAAGD,cAAc,GAAGA,cAAc,CAACE,SAAS,CAAC,QAAQ,CAAC,GAAG,KAAK;EAEpF,MAAMC,WAAW,GAAGA,CAAA,KAAM;IACtBJ,MAAM,CAACK,eAAe,CAACT,mBAAmB,EAAE,QAAQ,CAAC;EACzD,CAAC;EAED,oBACID,KAAA,CAAAW,aAAA;IACIC,OAAO,EAAEH,WAAY;IACrBI,SAAS,EAAE,oBAAoB,IAAIN,gBAAgB,GAAG,QAAQ,GAAG,EAAE,CAAE;IACrE,cAAW;EAAuB,gBAElCP,KAAA,CAAAW,aAAA;IAAGE,SAAS,EAAC;EAAe,CAAE,CAC1B,CAAC;AAEjB,CAAC","ignoreList":[]}
@@ -0,0 +1,2 @@
1
+ import React from "react";
2
+ export declare const LinkAction: () => React.JSX.Element;