js-draw 0.18.2 → 0.20.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 (269) hide show
  1. package/.eslintrc.js +1 -0
  2. package/CHANGELOG.md +10 -0
  3. package/dist/bundle.js +2 -2
  4. package/dist/bundledStyles.js +1 -0
  5. package/dist/cjs/src/Color4.d.ts +8 -0
  6. package/dist/cjs/src/Color4.js +67 -0
  7. package/dist/cjs/src/Editor.d.ts +2 -2
  8. package/dist/cjs/src/Editor.js +7 -7
  9. package/dist/cjs/src/SVGLoader.js +77 -12
  10. package/dist/cjs/src/Viewport.d.ts +2 -0
  11. package/dist/cjs/src/Viewport.js +6 -2
  12. package/dist/cjs/src/components/AbstractComponent.d.ts +2 -2
  13. package/dist/cjs/src/components/AbstractComponent.js +3 -3
  14. package/dist/cjs/src/components/{ImageBackground.d.ts → BackgroundComponent.d.ts} +23 -3
  15. package/dist/cjs/src/components/BackgroundComponent.js +309 -0
  16. package/dist/cjs/src/components/RestylableComponent.d.ts +21 -2
  17. package/dist/cjs/src/components/Stroke.d.ts +35 -0
  18. package/dist/cjs/src/components/Stroke.js +37 -3
  19. package/dist/cjs/src/components/TextComponent.d.ts +27 -17
  20. package/dist/cjs/src/components/TextComponent.js +23 -1
  21. package/dist/cjs/src/components/lib.d.ts +4 -3
  22. package/dist/cjs/src/components/lib.js +2 -2
  23. package/dist/cjs/src/components/util/StrokeSmoother.js +25 -15
  24. package/dist/cjs/src/lib.d.ts +30 -0
  25. package/dist/cjs/src/lib.js +30 -0
  26. package/dist/cjs/src/localizations/de.js +1 -1
  27. package/dist/cjs/src/localizations/es.js +1 -1
  28. package/dist/cjs/src/math/Path.js +1 -1
  29. package/dist/cjs/src/math/polynomial/QuadraticBezier.d.ts +28 -0
  30. package/dist/cjs/src/math/polynomial/QuadraticBezier.js +115 -0
  31. package/dist/cjs/src/math/polynomial/solveQuadratic.d.ts +6 -0
  32. package/dist/cjs/src/math/polynomial/solveQuadratic.js +36 -0
  33. package/dist/cjs/src/rendering/RenderingStyle.d.ts +4 -4
  34. package/dist/cjs/src/rendering/TextRenderingStyle.d.ts +10 -10
  35. package/dist/cjs/src/rendering/lib.d.ts +2 -0
  36. package/dist/cjs/src/rendering/renderers/AbstractRenderer.d.ts +2 -2
  37. package/dist/cjs/src/rendering/renderers/CanvasRenderer.d.ts +2 -2
  38. package/dist/cjs/src/rendering/renderers/CanvasRenderer.js +5 -3
  39. package/dist/cjs/src/rendering/renderers/DummyRenderer.d.ts +2 -2
  40. package/dist/cjs/src/rendering/renderers/SVGRenderer.d.ts +2 -2
  41. package/dist/cjs/src/rendering/renderers/SVGRenderer.js +15 -6
  42. package/dist/cjs/src/rendering/renderers/TextOnlyRenderer.d.ts +2 -2
  43. package/dist/cjs/src/toolbar/IconProvider.d.ts +2 -2
  44. package/dist/cjs/src/toolbar/localization.d.ts +2 -1
  45. package/dist/cjs/src/toolbar/localization.js +3 -2
  46. package/dist/cjs/src/toolbar/widgets/BaseWidget.js +1 -1
  47. package/dist/cjs/src/toolbar/widgets/DocumentPropertiesWidget.d.ts +5 -0
  48. package/dist/cjs/src/toolbar/widgets/DocumentPropertiesWidget.js +77 -2
  49. package/dist/cjs/src/toolbar/widgets/PenToolWidget.js +1 -1
  50. package/dist/cjs/src/tools/FindTool.js +1 -1
  51. package/dist/cjs/src/tools/SoundUITool.d.ts +24 -0
  52. package/dist/cjs/src/tools/SoundUITool.js +164 -0
  53. package/dist/cjs/src/tools/TextTool.d.ts +2 -2
  54. package/dist/cjs/src/tools/ToolController.js +6 -1
  55. package/dist/cjs/src/tools/lib.d.ts +1 -0
  56. package/dist/cjs/src/tools/lib.js +3 -1
  57. package/dist/cjs/src/tools/localization.d.ts +3 -0
  58. package/dist/cjs/src/tools/localization.js +3 -0
  59. package/dist/mjs/src/Color4.d.ts +8 -0
  60. package/dist/mjs/src/Color4.mjs +64 -0
  61. package/dist/mjs/src/Editor.d.ts +2 -2
  62. package/dist/mjs/src/Editor.mjs +6 -6
  63. package/dist/mjs/src/SVGLoader.mjs +76 -11
  64. package/dist/mjs/src/Viewport.d.ts +2 -0
  65. package/dist/mjs/src/Viewport.mjs +6 -2
  66. package/dist/mjs/src/components/AbstractComponent.d.ts +2 -2
  67. package/dist/mjs/src/components/AbstractComponent.mjs +3 -3
  68. package/dist/mjs/src/components/{ImageBackground.d.ts → BackgroundComponent.d.ts} +23 -3
  69. package/dist/mjs/src/components/BackgroundComponent.mjs +279 -0
  70. package/dist/mjs/src/components/RestylableComponent.d.ts +21 -2
  71. package/dist/mjs/src/components/Stroke.d.ts +35 -0
  72. package/dist/mjs/src/components/Stroke.mjs +37 -3
  73. package/dist/mjs/src/components/TextComponent.d.ts +27 -17
  74. package/dist/mjs/src/components/TextComponent.mjs +23 -1
  75. package/dist/mjs/src/components/lib.d.ts +4 -3
  76. package/dist/mjs/src/components/lib.mjs +2 -2
  77. package/dist/mjs/src/components/util/StrokeSmoother.mjs +25 -15
  78. package/dist/mjs/src/lib.d.ts +30 -0
  79. package/dist/mjs/src/lib.mjs +30 -0
  80. package/dist/mjs/src/localizations/de.mjs +1 -1
  81. package/dist/mjs/src/localizations/es.mjs +1 -1
  82. package/dist/mjs/src/math/Path.mjs +1 -1
  83. package/dist/mjs/src/math/polynomial/QuadraticBezier.d.ts +28 -0
  84. package/dist/mjs/src/math/polynomial/QuadraticBezier.mjs +109 -0
  85. package/dist/mjs/src/math/polynomial/solveQuadratic.d.ts +6 -0
  86. package/dist/mjs/src/math/polynomial/solveQuadratic.mjs +34 -0
  87. package/dist/mjs/src/rendering/RenderingStyle.d.ts +4 -4
  88. package/dist/mjs/src/rendering/TextRenderingStyle.d.ts +10 -10
  89. package/dist/mjs/src/rendering/lib.d.ts +2 -0
  90. package/dist/mjs/src/rendering/renderers/AbstractRenderer.d.ts +2 -2
  91. package/dist/mjs/src/rendering/renderers/CanvasRenderer.d.ts +2 -2
  92. package/dist/mjs/src/rendering/renderers/CanvasRenderer.mjs +5 -3
  93. package/dist/mjs/src/rendering/renderers/DummyRenderer.d.ts +2 -2
  94. package/dist/mjs/src/rendering/renderers/SVGRenderer.d.ts +2 -2
  95. package/dist/mjs/src/rendering/renderers/SVGRenderer.mjs +15 -6
  96. package/dist/mjs/src/rendering/renderers/TextOnlyRenderer.d.ts +2 -2
  97. package/dist/mjs/src/toolbar/IconProvider.d.ts +2 -2
  98. package/dist/mjs/src/toolbar/localization.d.ts +2 -1
  99. package/dist/mjs/src/toolbar/localization.mjs +3 -2
  100. package/dist/mjs/src/toolbar/widgets/BaseWidget.mjs +1 -1
  101. package/dist/mjs/src/toolbar/widgets/DocumentPropertiesWidget.d.ts +5 -0
  102. package/dist/mjs/src/toolbar/widgets/DocumentPropertiesWidget.mjs +54 -2
  103. package/dist/mjs/src/toolbar/widgets/PenToolWidget.mjs +1 -1
  104. package/dist/mjs/src/tools/FindTool.mjs +1 -1
  105. package/dist/mjs/src/tools/SoundUITool.d.ts +24 -0
  106. package/dist/mjs/src/tools/SoundUITool.mjs +158 -0
  107. package/dist/mjs/src/tools/TextTool.d.ts +2 -2
  108. package/dist/mjs/src/tools/ToolController.mjs +6 -1
  109. package/dist/mjs/src/tools/lib.d.ts +1 -0
  110. package/dist/mjs/src/tools/lib.mjs +1 -0
  111. package/dist/mjs/src/tools/localization.d.ts +3 -0
  112. package/dist/mjs/src/tools/localization.mjs +3 -0
  113. package/jest.config.js +1 -1
  114. package/package.json +19 -17
  115. package/src/Editor.css +2 -2
  116. package/src/tools/SoundUITool.css +15 -0
  117. package/src/tools/tools.css +4 -0
  118. package/dist/cjs/src/components/ImageBackground.js +0 -146
  119. package/dist/mjs/src/components/ImageBackground.mjs +0 -139
  120. package/src/Color4.test.ts +0 -40
  121. package/src/Color4.ts +0 -236
  122. package/src/Editor.loadFrom.test.ts +0 -24
  123. package/src/Editor.toSVG.test.ts +0 -111
  124. package/src/Editor.ts +0 -1122
  125. package/src/EditorImage.test.ts +0 -120
  126. package/src/EditorImage.ts +0 -603
  127. package/src/EventDispatcher.test.ts +0 -123
  128. package/src/EventDispatcher.ts +0 -71
  129. package/src/Pointer.ts +0 -127
  130. package/src/SVGLoader.test.ts +0 -114
  131. package/src/SVGLoader.ts +0 -511
  132. package/src/UndoRedoHistory.test.ts +0 -33
  133. package/src/UndoRedoHistory.ts +0 -102
  134. package/src/Viewport.ts +0 -319
  135. package/src/bundle/bundled.ts +0 -7
  136. package/src/commands/Command.ts +0 -45
  137. package/src/commands/Duplicate.ts +0 -48
  138. package/src/commands/Erase.ts +0 -74
  139. package/src/commands/SerializableCommand.ts +0 -49
  140. package/src/commands/UnresolvedCommand.ts +0 -37
  141. package/src/commands/invertCommand.ts +0 -51
  142. package/src/commands/lib.ts +0 -16
  143. package/src/commands/localization.ts +0 -47
  144. package/src/commands/uniteCommands.test.ts +0 -23
  145. package/src/commands/uniteCommands.ts +0 -135
  146. package/src/components/AbstractComponent.transformBy.test.ts +0 -22
  147. package/src/components/AbstractComponent.ts +0 -364
  148. package/src/components/ImageBackground.test.ts +0 -35
  149. package/src/components/ImageBackground.ts +0 -176
  150. package/src/components/ImageComponent.ts +0 -171
  151. package/src/components/RestylableComponent.ts +0 -142
  152. package/src/components/SVGGlobalAttributesObject.ts +0 -81
  153. package/src/components/Stroke.test.ts +0 -139
  154. package/src/components/Stroke.ts +0 -245
  155. package/src/components/TextComponent.test.ts +0 -99
  156. package/src/components/TextComponent.ts +0 -315
  157. package/src/components/UnknownSVGObject.test.ts +0 -10
  158. package/src/components/UnknownSVGObject.ts +0 -60
  159. package/src/components/builders/ArrowBuilder.ts +0 -107
  160. package/src/components/builders/FreehandLineBuilder.ts +0 -212
  161. package/src/components/builders/LineBuilder.ts +0 -77
  162. package/src/components/builders/PressureSensitiveFreehandLineBuilder.ts +0 -454
  163. package/src/components/builders/RectangleBuilder.ts +0 -74
  164. package/src/components/builders/types.ts +0 -15
  165. package/src/components/lib.ts +0 -25
  166. package/src/components/localization.ts +0 -22
  167. package/src/components/util/StrokeSmoother.ts +0 -293
  168. package/src/components/util/describeComponentList.ts +0 -18
  169. package/src/lib.ts +0 -37
  170. package/src/localization.ts +0 -34
  171. package/src/localizations/de.ts +0 -98
  172. package/src/localizations/en.ts +0 -8
  173. package/src/localizations/es.ts +0 -74
  174. package/src/localizations/getLocalizationTable.test.ts +0 -27
  175. package/src/localizations/getLocalizationTable.ts +0 -55
  176. package/src/math/LineSegment2.test.ts +0 -99
  177. package/src/math/LineSegment2.ts +0 -160
  178. package/src/math/Mat33.test.ts +0 -244
  179. package/src/math/Mat33.ts +0 -437
  180. package/src/math/Path.fromString.test.ts +0 -223
  181. package/src/math/Path.test.ts +0 -198
  182. package/src/math/Path.toString.test.ts +0 -77
  183. package/src/math/Path.ts +0 -790
  184. package/src/math/Rect2.test.ts +0 -204
  185. package/src/math/Rect2.ts +0 -315
  186. package/src/math/Triangle.ts +0 -29
  187. package/src/math/Vec2.test.ts +0 -30
  188. package/src/math/Vec2.ts +0 -18
  189. package/src/math/Vec3.test.ts +0 -44
  190. package/src/math/Vec3.ts +0 -218
  191. package/src/math/lib.ts +0 -15
  192. package/src/math/rounding.test.ts +0 -65
  193. package/src/math/rounding.ts +0 -156
  194. package/src/rendering/Display.ts +0 -249
  195. package/src/rendering/RenderingStyle.test.ts +0 -68
  196. package/src/rendering/RenderingStyle.ts +0 -55
  197. package/src/rendering/TextRenderingStyle.ts +0 -45
  198. package/src/rendering/caching/CacheRecord.test.ts +0 -49
  199. package/src/rendering/caching/CacheRecord.ts +0 -77
  200. package/src/rendering/caching/CacheRecordManager.ts +0 -71
  201. package/src/rendering/caching/RenderingCache.test.ts +0 -44
  202. package/src/rendering/caching/RenderingCache.ts +0 -66
  203. package/src/rendering/caching/RenderingCacheNode.ts +0 -405
  204. package/src/rendering/caching/testUtils.ts +0 -35
  205. package/src/rendering/caching/types.ts +0 -34
  206. package/src/rendering/lib.ts +0 -6
  207. package/src/rendering/localization.ts +0 -20
  208. package/src/rendering/renderers/AbstractRenderer.ts +0 -222
  209. package/src/rendering/renderers/CanvasRenderer.ts +0 -296
  210. package/src/rendering/renderers/DummyRenderer.test.ts +0 -42
  211. package/src/rendering/renderers/DummyRenderer.ts +0 -136
  212. package/src/rendering/renderers/SVGRenderer.ts +0 -354
  213. package/src/rendering/renderers/TextOnlyRenderer.ts +0 -70
  214. package/src/testing/beforeEachFile.ts +0 -8
  215. package/src/testing/createEditor.ts +0 -11
  216. package/src/testing/global.d.ts +0 -17
  217. package/src/testing/lib.ts +0 -3
  218. package/src/testing/loadExpectExtensions.ts +0 -25
  219. package/src/testing/sendPenEvent.ts +0 -31
  220. package/src/testing/sendTouchEvent.ts +0 -78
  221. package/src/toolbar/HTMLToolbar.ts +0 -492
  222. package/src/toolbar/IconProvider.ts +0 -736
  223. package/src/toolbar/lib.ts +0 -4
  224. package/src/toolbar/localization.ts +0 -106
  225. package/src/toolbar/makeColorInput.ts +0 -145
  226. package/src/toolbar/types.ts +0 -5
  227. package/src/toolbar/widgets/ActionButtonWidget.ts +0 -39
  228. package/src/toolbar/widgets/BaseToolWidget.ts +0 -56
  229. package/src/toolbar/widgets/BaseWidget.ts +0 -377
  230. package/src/toolbar/widgets/DocumentPropertiesWidget.ts +0 -167
  231. package/src/toolbar/widgets/EraserToolWidget.ts +0 -85
  232. package/src/toolbar/widgets/HandToolWidget.ts +0 -250
  233. package/src/toolbar/widgets/InsertImageWidget.ts +0 -223
  234. package/src/toolbar/widgets/OverflowWidget.ts +0 -92
  235. package/src/toolbar/widgets/PenToolWidget.ts +0 -288
  236. package/src/toolbar/widgets/SelectionToolWidget.ts +0 -190
  237. package/src/toolbar/widgets/TextToolWidget.ts +0 -145
  238. package/src/toolbar/widgets/lib.ts +0 -13
  239. package/src/tools/BaseTool.ts +0 -76
  240. package/src/tools/Eraser.test.ts +0 -103
  241. package/src/tools/Eraser.ts +0 -139
  242. package/src/tools/FindTool.ts +0 -152
  243. package/src/tools/PanZoom.test.ts +0 -310
  244. package/src/tools/PanZoom.ts +0 -520
  245. package/src/tools/PasteHandler.ts +0 -95
  246. package/src/tools/Pen.test.ts +0 -194
  247. package/src/tools/Pen.ts +0 -226
  248. package/src/tools/PipetteTool.ts +0 -55
  249. package/src/tools/SelectionTool/SelectAllShortcutHandler.ts +0 -28
  250. package/src/tools/SelectionTool/Selection.ts +0 -607
  251. package/src/tools/SelectionTool/SelectionHandle.ts +0 -108
  252. package/src/tools/SelectionTool/SelectionTool.test.ts +0 -261
  253. package/src/tools/SelectionTool/SelectionTool.ts +0 -480
  254. package/src/tools/SelectionTool/TransformMode.ts +0 -114
  255. package/src/tools/SelectionTool/types.ts +0 -11
  256. package/src/tools/TextTool.ts +0 -326
  257. package/src/tools/ToolController.ts +0 -178
  258. package/src/tools/ToolEnabledGroup.ts +0 -14
  259. package/src/tools/ToolSwitcherShortcut.ts +0 -39
  260. package/src/tools/ToolbarShortcutHandler.ts +0 -34
  261. package/src/tools/UndoRedoShortcut.test.ts +0 -56
  262. package/src/tools/UndoRedoShortcut.ts +0 -25
  263. package/src/tools/lib.ts +0 -21
  264. package/src/tools/localization.ts +0 -66
  265. package/src/types.ts +0 -234
  266. package/src/util/assertions.ts +0 -55
  267. package/src/util/fileToBase64.ts +0 -18
  268. package/src/util/untilNextAnimationFrame.ts +0 -9
  269. package/src/util/waitForTimeout.ts +0 -9
@@ -1,492 +0,0 @@
1
- import Editor from '../Editor';
2
- import { EditorEventType } from '../types';
3
-
4
- import { coloris, init as colorisInit } from '@melloware/coloris';
5
- import Color4 from '../Color4';
6
- import { defaultToolbarLocalization, ToolbarLocalization } from './localization';
7
- import { ActionButtonIcon } from './types';
8
- import SelectionTool from '../tools/SelectionTool/SelectionTool';
9
- import PanZoomTool from '../tools/PanZoom';
10
- import TextTool from '../tools/TextTool';
11
- import EraserTool from '../tools/Eraser';
12
- import PenTool from '../tools/Pen';
13
- import PenToolWidget from './widgets/PenToolWidget';
14
- import EraserWidget from './widgets/EraserToolWidget';
15
- import SelectionToolWidget from './widgets/SelectionToolWidget';
16
- import TextToolWidget from './widgets/TextToolWidget';
17
- import HandToolWidget from './widgets/HandToolWidget';
18
- import BaseWidget from './widgets/BaseWidget';
19
- import ActionButtonWidget from './widgets/ActionButtonWidget';
20
- import InsertImageWidget from './widgets/InsertImageWidget';
21
- import DocumentPropertiesWidget from './widgets/DocumentPropertiesWidget';
22
- import OverflowWidget from './widgets/OverflowWidget';
23
- import { DispatcherEventListener } from '../EventDispatcher';
24
-
25
- export const toolbarCSSPrefix = 'toolbar-';
26
-
27
- type UpdateColorisCallback = ()=>void;
28
-
29
- interface SpacerOptions {
30
- // Defaults to 0. If a non-zero number, determines the rate at which the
31
- // spacer should grow (like flexGrow).
32
- grow: number;
33
-
34
- // Minimum size (e.g. "23px")
35
- minSize: string;
36
-
37
- // Maximum size (e.g. "50px")
38
- maxSize: string;
39
- }
40
-
41
- export default class HTMLToolbar {
42
- private container: HTMLElement;
43
- private resizeObserver: ResizeObserver;
44
- private listeners: DispatcherEventListener[] = [];
45
-
46
- // Flex-order of the next widget to be added.
47
- private widgetOrderCounter: number = 0;
48
-
49
- private widgetsById: Record<string, BaseWidget> = {};
50
- private widgetList: Array<BaseWidget> = [];
51
-
52
- // Widget to toggle overflow menu.
53
- private overflowWidget: OverflowWidget|null = null;
54
-
55
- private static colorisStarted: boolean = false;
56
- private updateColoris: UpdateColorisCallback|null = null;
57
-
58
- /** @internal */
59
- public constructor(
60
- private editor: Editor, parent: HTMLElement,
61
- private localizationTable: ToolbarLocalization = defaultToolbarLocalization,
62
- ) {
63
- this.container = document.createElement('div');
64
- this.container.classList.add(`${toolbarCSSPrefix}root`);
65
- this.container.setAttribute('role', 'toolbar');
66
- parent.appendChild(this.container);
67
-
68
- if (!HTMLToolbar.colorisStarted) {
69
- colorisInit();
70
- HTMLToolbar.colorisStarted = true;
71
- }
72
- this.setupColorPickers();
73
-
74
- if ('ResizeObserver' in window) {
75
- this.resizeObserver = new ResizeObserver((_entries) => {
76
- this.reLayout();
77
- });
78
- this.resizeObserver.observe(this.container);
79
- } else {
80
- console.warn('ResizeObserver not supported. Toolbar will not resize.');
81
- }
82
- }
83
-
84
- // @internal
85
- public setupColorPickers() {
86
- // Much of the setup only needs to be done once.
87
- if (this.updateColoris) {
88
- this.updateColoris();
89
- return;
90
- }
91
-
92
- const closePickerOverlay = document.createElement('div');
93
- closePickerOverlay.className = `${toolbarCSSPrefix}closeColorPickerOverlay`;
94
- this.editor.createHTMLOverlay(closePickerOverlay);
95
-
96
- const maxSwatchLen = 12;
97
- const swatches = [
98
- Color4.red.toHexString(),
99
- Color4.purple.toHexString(),
100
- Color4.blue.toHexString(),
101
- Color4.clay.toHexString(),
102
- Color4.black.toHexString(),
103
- Color4.white.toHexString(),
104
- ];
105
- const presetColorEnd = swatches.length;
106
-
107
- // (Re)init Coloris -- update the swatches list.
108
- const initColoris = () => {
109
- coloris({
110
- el: '.coloris_input',
111
- format: 'hex',
112
- selectInput: false,
113
- focusInput: false,
114
- themeMode: 'auto',
115
-
116
- swatches
117
- });
118
- };
119
- initColoris();
120
- this.updateColoris = initColoris;
121
-
122
- const addColorToSwatch = (newColor: string) => {
123
- let alreadyPresent = false;
124
-
125
- for (const color of swatches) {
126
- if (color === newColor) {
127
- alreadyPresent = true;
128
- }
129
- }
130
-
131
- if (!alreadyPresent) {
132
- swatches.push(newColor);
133
- if (swatches.length > maxSwatchLen) {
134
- swatches.splice(presetColorEnd, 1);
135
- }
136
- initColoris();
137
- }
138
- };
139
-
140
- this.listeners.push(this.editor.notifier.on(EditorEventType.ColorPickerToggled, event => {
141
- if (event.kind !== EditorEventType.ColorPickerToggled) {
142
- return;
143
- }
144
-
145
- // Show/hide the overlay. Making the overlay visible gives users a surface to click
146
- // on that shows/hides the color picker.
147
- closePickerOverlay.style.display = event.open ? 'block' : 'none';
148
- }));
149
-
150
- // Add newly-selected colors to the swatch.
151
- this.listeners.push(this.editor.notifier.on(EditorEventType.ColorPickerColorSelected, event => {
152
- if (event.kind === EditorEventType.ColorPickerColorSelected) {
153
- addColorToSwatch(event.color.toHexString());
154
- }
155
- }));
156
- }
157
-
158
- private reLayoutQueued: boolean = false;
159
- private queueReLayout() {
160
- if (!this.reLayoutQueued) {
161
- this.reLayoutQueued = true;
162
- requestAnimationFrame(() => this.reLayout());
163
- }
164
- }
165
-
166
- private reLayout() {
167
- this.reLayoutQueued = false;
168
-
169
- if (!this.overflowWidget) {
170
- return;
171
- }
172
-
173
- const getTotalWidth = (widgetList: Array<BaseWidget>) => {
174
- let totalWidth = 0;
175
- for (const widget of widgetList) {
176
- if (!widget.isHidden()) {
177
- totalWidth += widget.getButtonWidth();
178
- }
179
- }
180
-
181
- return totalWidth;
182
- };
183
-
184
- // Returns true if there is enough empty space to move the first child
185
- // from the overflow menu to the main menu.
186
- const canRemoveFirstChildFromOverflow = (freeSpaceInMainMenu: number) => {
187
- const overflowChildren = this.overflowWidget?.getChildWidgets() ?? [];
188
-
189
- if (overflowChildren.length === 0) {
190
- return false;
191
- }
192
-
193
- return overflowChildren[0].getButtonWidth() <= freeSpaceInMainMenu;
194
- };
195
-
196
- let overflowWidgetsWidth = getTotalWidth(this.overflowWidget.getChildWidgets());
197
- let shownWidgetWidth = getTotalWidth(this.widgetList) - overflowWidgetsWidth;
198
- let availableWidth = this.container.clientWidth * 0.87;
199
-
200
- // If on a device that has enough vertical space, allow
201
- // showing two rows of buttons.
202
- // TODO: Fix magic numbers
203
- if (window.innerHeight > availableWidth * 1.75) {
204
- availableWidth *= 1.75;
205
- }
206
-
207
- let updatedChildren = false;
208
-
209
- // If we can remove at least one child from the overflow menu,
210
- if (canRemoveFirstChildFromOverflow(availableWidth - shownWidgetWidth)) {
211
- // Move widgets to the main menu.
212
- const overflowChildren = this.overflowWidget.clearChildren();
213
-
214
- for (const child of overflowChildren) {
215
- child.addTo(this.container);
216
- child.setIsToplevel(true);
217
-
218
- if (!child.isHidden()) {
219
- shownWidgetWidth += child.getButtonWidth();
220
- }
221
- }
222
-
223
- overflowWidgetsWidth = 0;
224
- updatedChildren = true;
225
- }
226
-
227
- if (shownWidgetWidth >= availableWidth) {
228
- // Move widgets to the overflow menu.
229
-
230
- // Start with the rightmost widget, move to the leftmost
231
- for (
232
- let i = this.widgetList.length - 1;
233
- i >= 0 && shownWidgetWidth >= availableWidth;
234
- i--
235
- ) {
236
- const child = this.widgetList[i];
237
-
238
- if (this.overflowWidget.hasAsChild(child)) {
239
- continue;
240
- }
241
-
242
- if (child.canBeInOverflowMenu()) {
243
- shownWidgetWidth -= child.getButtonWidth();
244
- this.overflowWidget.addToOverflow(child);
245
- }
246
- }
247
-
248
- updatedChildren = true;
249
- }
250
-
251
- // Hide/show the overflow widget.
252
- this.overflowWidget.setHidden(this.overflowWidget.getChildWidgets().length === 0);
253
-
254
- if (updatedChildren) {
255
- this.setupColorPickers();
256
- }
257
- }
258
-
259
-
260
- /**
261
- * Adds an `ActionButtonWidget` or `BaseToolWidget`. The widget should not have already have a parent
262
- * (i.e. its `addTo` method should not have been called).
263
- *
264
- * @example
265
- * ```ts
266
- * const toolbar = editor.addToolbar();
267
- * const insertImageWidget = new InsertImageWidget(editor);
268
- * toolbar.addWidget(insertImageWidget);
269
- * ```
270
- */
271
- public addWidget(widget: BaseWidget) {
272
- // Prevent name collisions
273
- const id = widget.getUniqueIdIn(this.widgetsById);
274
-
275
- // Add the widget
276
- this.widgetsById[id] = widget;
277
- this.widgetList.push(widget);
278
-
279
- // Add HTML elements.
280
- const container = widget.addTo(this.container);
281
- this.setupColorPickers();
282
-
283
- // Ensure that the widget gets displayed in the correct
284
- // place in the toolbar, even if it's removed and re-added.
285
- container.style.order = `${this.widgetOrderCounter++}`;
286
-
287
- this.queueReLayout();
288
- }
289
-
290
- /**
291
- * Adds a spacer.
292
- *
293
- * @example
294
- * Adding a save button that moves to the very right edge of the toolbar
295
- * while keeping the other buttons centered:
296
- * ```ts
297
- * const toolbar = editor.addToolbar(false);
298
- *
299
- * toolbar.addSpacer({ grow: 1 });
300
- * toolbar.addDefaults();
301
- * toolbar.addSpacer({ grow: 1 });
302
- *
303
- * toolbar.addActionButton({
304
- * label: 'Save',
305
- * icon: editor.icons.makeSaveIcon(),
306
- * }, () => {
307
- * saveCallback();
308
- * });
309
- * ```
310
- */
311
- public addSpacer(options: Partial<SpacerOptions> = {}) {
312
- const spacer = document.createElement('div');
313
- spacer.classList.add(`${toolbarCSSPrefix}spacer`);
314
-
315
- if (options.grow) {
316
- spacer.style.flexGrow = `${options.grow}`;
317
- }
318
-
319
- if (options.minSize) {
320
- spacer.style.minWidth = options.minSize;
321
- }
322
-
323
- if (options.maxSize) {
324
- spacer.style.maxWidth = options.maxSize;
325
- }
326
-
327
- spacer.style.order = `${this.widgetOrderCounter++}`;
328
- this.container.appendChild(spacer);
329
- }
330
-
331
- public serializeState(): string {
332
- const result: Record<string, any> = {};
333
-
334
- for (const widgetId in this.widgetsById) {
335
- result[widgetId] = this.widgetsById[widgetId].serializeState();
336
- }
337
-
338
- return JSON.stringify(result);
339
- }
340
-
341
- /**
342
- * Deserialize toolbar widgets from the given state.
343
- * Assumes that toolbar widgets are in the same order as when state was serialized.
344
- */
345
- public deserializeState(state: string) {
346
- const data = JSON.parse(state);
347
-
348
- for (const widgetId in data) {
349
- if (!(widgetId in this.widgetsById)) {
350
- console.warn(`Unable to deserialize widget ${widgetId} ­— no such widget.`);
351
- }
352
-
353
- this.widgetsById[widgetId].deserializeFrom(data[widgetId]);
354
- }
355
- }
356
-
357
- /**
358
- * Adds an action button with `title` to this toolbar (or to the given `parent` element).
359
- *
360
- * @return The added button.
361
- */
362
- public addActionButton(
363
- title: string|ActionButtonIcon,
364
- command: ()=> void,
365
- mustBeToplevel: boolean = true
366
- ): BaseWidget {
367
- const titleString = typeof title === 'string' ? title : title.label;
368
- const widgetId = 'action-button';
369
-
370
- const makeIcon = () => {
371
- if (typeof title === 'string') {
372
- return null;
373
- }
374
-
375
- return title.icon;
376
- };
377
-
378
- const widget = new ActionButtonWidget(
379
- this.editor,
380
- widgetId,
381
- makeIcon,
382
- titleString,
383
- command,
384
- this.editor.localization,
385
- mustBeToplevel,
386
- );
387
-
388
- this.addWidget(widget);
389
- return widget;
390
- }
391
-
392
- public addUndoRedoButtons() {
393
- const undoButton = this.addActionButton({
394
- label: this.localizationTable.undo,
395
- icon: this.editor.icons.makeUndoIcon()
396
- }, () => {
397
- this.editor.history.undo();
398
- });
399
- const redoButton = this.addActionButton({
400
- label: this.localizationTable.redo,
401
- icon: this.editor.icons.makeRedoIcon(),
402
- }, () => {
403
- this.editor.history.redo();
404
- });
405
-
406
- undoButton.setDisabled(true);
407
- redoButton.setDisabled(true);
408
- this.editor.notifier.on(EditorEventType.UndoRedoStackUpdated, event => {
409
- if (event.kind !== EditorEventType.UndoRedoStackUpdated) {
410
- throw new Error('Wrong event type!');
411
- }
412
-
413
- undoButton.setDisabled(event.undoStackSize === 0);
414
- redoButton.setDisabled(event.redoStackSize === 0);
415
- });
416
- }
417
-
418
- public addDefaultToolWidgets() {
419
- const toolController = this.editor.toolController;
420
- for (const tool of toolController.getMatchingTools(PenTool)) {
421
- const widget = new PenToolWidget(
422
- this.editor, tool, this.localizationTable,
423
- );
424
- this.addWidget(widget);
425
- }
426
-
427
- for (const tool of toolController.getMatchingTools(EraserTool)) {
428
- this.addWidget(new EraserWidget(this.editor, tool, this.localizationTable));
429
- }
430
-
431
- for (const tool of toolController.getMatchingTools(SelectionTool)) {
432
- this.addWidget(new SelectionToolWidget(this.editor, tool, this.localizationTable));
433
- }
434
-
435
- for (const tool of toolController.getMatchingTools(TextTool)) {
436
- this.addWidget(new TextToolWidget(this.editor, tool, this.localizationTable));
437
- }
438
-
439
- const panZoomTool = toolController.getMatchingTools(PanZoomTool)[0];
440
- if (panZoomTool) {
441
- this.addWidget(new HandToolWidget(this.editor, panZoomTool, this.localizationTable));
442
- }
443
-
444
- this.addWidget(new InsertImageWidget(this.editor, this.localizationTable));
445
- }
446
-
447
- public addDefaultActionButtons() {
448
- this.addWidget(new DocumentPropertiesWidget(this.editor, this.localizationTable));
449
- this.addUndoRedoButtons();
450
- }
451
-
452
- /**
453
- * Adds a widget that toggles the overflow menu. Call `addOverflowWidget` to ensure
454
- * that this widget is in the correct space (if shown).
455
- *
456
- * @example
457
- * ```ts
458
- * toolbar.addDefaultToolWidgets();
459
- * toolbar.addOverflowWidget();
460
- * toolbar.addDefaultActionButtons();
461
- * ```
462
- * shows the overflow widget between the default tool widgets and the default action buttons,
463
- * if shown.
464
- */
465
- public addOverflowWidget() {
466
- this.overflowWidget = new OverflowWidget(this.editor, this.localizationTable);
467
- this.addWidget(this.overflowWidget);
468
- }
469
-
470
- /**
471
- * Adds both the default tool widgets and action buttons. Equivalent to
472
- * ```ts
473
- * toolbar.addDefaultToolWidgets();
474
- * toolbar.addOverflowWidget();
475
- * toolbar.addDefaultActionButtons();
476
- * ```
477
- */
478
- public addDefaults() {
479
- this.addDefaultToolWidgets();
480
- this.addOverflowWidget();
481
- this.addDefaultActionButtons();
482
- }
483
-
484
- public remove() {
485
- this.container.remove();
486
- this.resizeObserver.disconnect();
487
-
488
- for (const listener of this.listeners) {
489
- listener.remove();
490
- }
491
- }
492
- }