js-draw 0.18.2 → 0.19.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 (227) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/dist/bundle.js +2 -2
  3. package/dist/bundledStyles.js +1 -0
  4. package/dist/cjs/src/Color4.d.ts +8 -0
  5. package/dist/cjs/src/Color4.js +67 -0
  6. package/dist/cjs/src/Editor.d.ts +2 -2
  7. package/dist/cjs/src/Editor.js +2 -2
  8. package/dist/cjs/src/SVGLoader.js +8 -5
  9. package/dist/cjs/src/components/AbstractComponent.d.ts +2 -2
  10. package/dist/cjs/src/components/AbstractComponent.js +3 -3
  11. package/dist/cjs/src/components/ImageBackground.js +1 -1
  12. package/dist/cjs/src/components/RestylableComponent.d.ts +21 -2
  13. package/dist/cjs/src/components/Stroke.d.ts +35 -0
  14. package/dist/cjs/src/components/Stroke.js +36 -2
  15. package/dist/cjs/src/components/TextComponent.d.ts +26 -4
  16. package/dist/cjs/src/components/TextComponent.js +22 -0
  17. package/dist/cjs/src/components/lib.d.ts +3 -2
  18. package/dist/cjs/src/lib.d.ts +30 -0
  19. package/dist/cjs/src/lib.js +30 -0
  20. package/dist/cjs/src/rendering/RenderingStyle.d.ts +4 -4
  21. package/dist/cjs/src/rendering/TextRenderingStyle.d.ts +10 -10
  22. package/dist/cjs/src/rendering/lib.d.ts +2 -0
  23. package/dist/cjs/src/rendering/renderers/AbstractRenderer.d.ts +2 -2
  24. package/dist/cjs/src/rendering/renderers/CanvasRenderer.d.ts +2 -2
  25. package/dist/cjs/src/rendering/renderers/DummyRenderer.d.ts +2 -2
  26. package/dist/cjs/src/rendering/renderers/SVGRenderer.d.ts +2 -2
  27. package/dist/cjs/src/rendering/renderers/TextOnlyRenderer.d.ts +2 -2
  28. package/dist/cjs/src/toolbar/IconProvider.d.ts +2 -2
  29. package/dist/cjs/src/toolbar/localization.js +1 -1
  30. package/dist/cjs/src/toolbar/widgets/BaseWidget.js +1 -1
  31. package/dist/cjs/src/tools/SoundUITool.d.ts +24 -0
  32. package/dist/cjs/src/tools/SoundUITool.js +164 -0
  33. package/dist/cjs/src/tools/TextTool.d.ts +2 -2
  34. package/dist/cjs/src/tools/ToolController.js +6 -1
  35. package/dist/cjs/src/tools/lib.d.ts +1 -0
  36. package/dist/cjs/src/tools/lib.js +3 -1
  37. package/dist/cjs/src/tools/localization.d.ts +3 -0
  38. package/dist/cjs/src/tools/localization.js +3 -0
  39. package/dist/mjs/src/Color4.d.ts +8 -0
  40. package/dist/mjs/src/Color4.mjs +64 -0
  41. package/dist/mjs/src/Editor.d.ts +2 -2
  42. package/dist/mjs/src/Editor.mjs +2 -2
  43. package/dist/mjs/src/SVGLoader.mjs +8 -5
  44. package/dist/mjs/src/components/AbstractComponent.d.ts +2 -2
  45. package/dist/mjs/src/components/AbstractComponent.mjs +3 -3
  46. package/dist/mjs/src/components/ImageBackground.mjs +1 -1
  47. package/dist/mjs/src/components/RestylableComponent.d.ts +21 -2
  48. package/dist/mjs/src/components/Stroke.d.ts +35 -0
  49. package/dist/mjs/src/components/Stroke.mjs +36 -2
  50. package/dist/mjs/src/components/TextComponent.d.ts +26 -4
  51. package/dist/mjs/src/components/TextComponent.mjs +22 -0
  52. package/dist/mjs/src/components/lib.d.ts +3 -2
  53. package/dist/mjs/src/lib.d.ts +30 -0
  54. package/dist/mjs/src/lib.mjs +30 -0
  55. package/dist/mjs/src/rendering/RenderingStyle.d.ts +4 -4
  56. package/dist/mjs/src/rendering/TextRenderingStyle.d.ts +10 -10
  57. package/dist/mjs/src/rendering/lib.d.ts +2 -0
  58. package/dist/mjs/src/rendering/renderers/AbstractRenderer.d.ts +2 -2
  59. package/dist/mjs/src/rendering/renderers/CanvasRenderer.d.ts +2 -2
  60. package/dist/mjs/src/rendering/renderers/DummyRenderer.d.ts +2 -2
  61. package/dist/mjs/src/rendering/renderers/SVGRenderer.d.ts +2 -2
  62. package/dist/mjs/src/rendering/renderers/TextOnlyRenderer.d.ts +2 -2
  63. package/dist/mjs/src/toolbar/IconProvider.d.ts +2 -2
  64. package/dist/mjs/src/toolbar/localization.mjs +1 -1
  65. package/dist/mjs/src/toolbar/widgets/BaseWidget.mjs +1 -1
  66. package/dist/mjs/src/tools/SoundUITool.d.ts +24 -0
  67. package/dist/mjs/src/tools/SoundUITool.mjs +158 -0
  68. package/dist/mjs/src/tools/TextTool.d.ts +2 -2
  69. package/dist/mjs/src/tools/ToolController.mjs +6 -1
  70. package/dist/mjs/src/tools/lib.d.ts +1 -0
  71. package/dist/mjs/src/tools/lib.mjs +1 -0
  72. package/dist/mjs/src/tools/localization.d.ts +3 -0
  73. package/dist/mjs/src/tools/localization.mjs +3 -0
  74. package/package.json +6 -4
  75. package/src/Editor.css +2 -2
  76. package/src/tools/SoundUITool.css +15 -0
  77. package/src/tools/tools.css +4 -0
  78. package/src/Color4.test.ts +0 -40
  79. package/src/Color4.ts +0 -236
  80. package/src/Editor.loadFrom.test.ts +0 -24
  81. package/src/Editor.toSVG.test.ts +0 -111
  82. package/src/Editor.ts +0 -1122
  83. package/src/EditorImage.test.ts +0 -120
  84. package/src/EditorImage.ts +0 -603
  85. package/src/EventDispatcher.test.ts +0 -123
  86. package/src/EventDispatcher.ts +0 -71
  87. package/src/Pointer.ts +0 -127
  88. package/src/SVGLoader.test.ts +0 -114
  89. package/src/SVGLoader.ts +0 -511
  90. package/src/UndoRedoHistory.test.ts +0 -33
  91. package/src/UndoRedoHistory.ts +0 -102
  92. package/src/Viewport.ts +0 -319
  93. package/src/bundle/bundled.ts +0 -7
  94. package/src/commands/Command.ts +0 -45
  95. package/src/commands/Duplicate.ts +0 -48
  96. package/src/commands/Erase.ts +0 -74
  97. package/src/commands/SerializableCommand.ts +0 -49
  98. package/src/commands/UnresolvedCommand.ts +0 -37
  99. package/src/commands/invertCommand.ts +0 -51
  100. package/src/commands/lib.ts +0 -16
  101. package/src/commands/localization.ts +0 -47
  102. package/src/commands/uniteCommands.test.ts +0 -23
  103. package/src/commands/uniteCommands.ts +0 -135
  104. package/src/components/AbstractComponent.transformBy.test.ts +0 -22
  105. package/src/components/AbstractComponent.ts +0 -364
  106. package/src/components/ImageBackground.test.ts +0 -35
  107. package/src/components/ImageBackground.ts +0 -176
  108. package/src/components/ImageComponent.ts +0 -171
  109. package/src/components/RestylableComponent.ts +0 -142
  110. package/src/components/SVGGlobalAttributesObject.ts +0 -81
  111. package/src/components/Stroke.test.ts +0 -139
  112. package/src/components/Stroke.ts +0 -245
  113. package/src/components/TextComponent.test.ts +0 -99
  114. package/src/components/TextComponent.ts +0 -315
  115. package/src/components/UnknownSVGObject.test.ts +0 -10
  116. package/src/components/UnknownSVGObject.ts +0 -60
  117. package/src/components/builders/ArrowBuilder.ts +0 -107
  118. package/src/components/builders/FreehandLineBuilder.ts +0 -212
  119. package/src/components/builders/LineBuilder.ts +0 -77
  120. package/src/components/builders/PressureSensitiveFreehandLineBuilder.ts +0 -454
  121. package/src/components/builders/RectangleBuilder.ts +0 -74
  122. package/src/components/builders/types.ts +0 -15
  123. package/src/components/lib.ts +0 -25
  124. package/src/components/localization.ts +0 -22
  125. package/src/components/util/StrokeSmoother.ts +0 -293
  126. package/src/components/util/describeComponentList.ts +0 -18
  127. package/src/lib.ts +0 -37
  128. package/src/localization.ts +0 -34
  129. package/src/localizations/de.ts +0 -98
  130. package/src/localizations/en.ts +0 -8
  131. package/src/localizations/es.ts +0 -74
  132. package/src/localizations/getLocalizationTable.test.ts +0 -27
  133. package/src/localizations/getLocalizationTable.ts +0 -55
  134. package/src/math/LineSegment2.test.ts +0 -99
  135. package/src/math/LineSegment2.ts +0 -160
  136. package/src/math/Mat33.test.ts +0 -244
  137. package/src/math/Mat33.ts +0 -437
  138. package/src/math/Path.fromString.test.ts +0 -223
  139. package/src/math/Path.test.ts +0 -198
  140. package/src/math/Path.toString.test.ts +0 -77
  141. package/src/math/Path.ts +0 -790
  142. package/src/math/Rect2.test.ts +0 -204
  143. package/src/math/Rect2.ts +0 -315
  144. package/src/math/Triangle.ts +0 -29
  145. package/src/math/Vec2.test.ts +0 -30
  146. package/src/math/Vec2.ts +0 -18
  147. package/src/math/Vec3.test.ts +0 -44
  148. package/src/math/Vec3.ts +0 -218
  149. package/src/math/lib.ts +0 -15
  150. package/src/math/rounding.test.ts +0 -65
  151. package/src/math/rounding.ts +0 -156
  152. package/src/rendering/Display.ts +0 -249
  153. package/src/rendering/RenderingStyle.test.ts +0 -68
  154. package/src/rendering/RenderingStyle.ts +0 -55
  155. package/src/rendering/TextRenderingStyle.ts +0 -45
  156. package/src/rendering/caching/CacheRecord.test.ts +0 -49
  157. package/src/rendering/caching/CacheRecord.ts +0 -77
  158. package/src/rendering/caching/CacheRecordManager.ts +0 -71
  159. package/src/rendering/caching/RenderingCache.test.ts +0 -44
  160. package/src/rendering/caching/RenderingCache.ts +0 -66
  161. package/src/rendering/caching/RenderingCacheNode.ts +0 -405
  162. package/src/rendering/caching/testUtils.ts +0 -35
  163. package/src/rendering/caching/types.ts +0 -34
  164. package/src/rendering/lib.ts +0 -6
  165. package/src/rendering/localization.ts +0 -20
  166. package/src/rendering/renderers/AbstractRenderer.ts +0 -222
  167. package/src/rendering/renderers/CanvasRenderer.ts +0 -296
  168. package/src/rendering/renderers/DummyRenderer.test.ts +0 -42
  169. package/src/rendering/renderers/DummyRenderer.ts +0 -136
  170. package/src/rendering/renderers/SVGRenderer.ts +0 -354
  171. package/src/rendering/renderers/TextOnlyRenderer.ts +0 -70
  172. package/src/testing/beforeEachFile.ts +0 -8
  173. package/src/testing/createEditor.ts +0 -11
  174. package/src/testing/global.d.ts +0 -17
  175. package/src/testing/lib.ts +0 -3
  176. package/src/testing/loadExpectExtensions.ts +0 -25
  177. package/src/testing/sendPenEvent.ts +0 -31
  178. package/src/testing/sendTouchEvent.ts +0 -78
  179. package/src/toolbar/HTMLToolbar.ts +0 -492
  180. package/src/toolbar/IconProvider.ts +0 -736
  181. package/src/toolbar/lib.ts +0 -4
  182. package/src/toolbar/localization.ts +0 -106
  183. package/src/toolbar/makeColorInput.ts +0 -145
  184. package/src/toolbar/types.ts +0 -5
  185. package/src/toolbar/widgets/ActionButtonWidget.ts +0 -39
  186. package/src/toolbar/widgets/BaseToolWidget.ts +0 -56
  187. package/src/toolbar/widgets/BaseWidget.ts +0 -377
  188. package/src/toolbar/widgets/DocumentPropertiesWidget.ts +0 -167
  189. package/src/toolbar/widgets/EraserToolWidget.ts +0 -85
  190. package/src/toolbar/widgets/HandToolWidget.ts +0 -250
  191. package/src/toolbar/widgets/InsertImageWidget.ts +0 -223
  192. package/src/toolbar/widgets/OverflowWidget.ts +0 -92
  193. package/src/toolbar/widgets/PenToolWidget.ts +0 -288
  194. package/src/toolbar/widgets/SelectionToolWidget.ts +0 -190
  195. package/src/toolbar/widgets/TextToolWidget.ts +0 -145
  196. package/src/toolbar/widgets/lib.ts +0 -13
  197. package/src/tools/BaseTool.ts +0 -76
  198. package/src/tools/Eraser.test.ts +0 -103
  199. package/src/tools/Eraser.ts +0 -139
  200. package/src/tools/FindTool.ts +0 -152
  201. package/src/tools/PanZoom.test.ts +0 -310
  202. package/src/tools/PanZoom.ts +0 -520
  203. package/src/tools/PasteHandler.ts +0 -95
  204. package/src/tools/Pen.test.ts +0 -194
  205. package/src/tools/Pen.ts +0 -226
  206. package/src/tools/PipetteTool.ts +0 -55
  207. package/src/tools/SelectionTool/SelectAllShortcutHandler.ts +0 -28
  208. package/src/tools/SelectionTool/Selection.ts +0 -607
  209. package/src/tools/SelectionTool/SelectionHandle.ts +0 -108
  210. package/src/tools/SelectionTool/SelectionTool.test.ts +0 -261
  211. package/src/tools/SelectionTool/SelectionTool.ts +0 -480
  212. package/src/tools/SelectionTool/TransformMode.ts +0 -114
  213. package/src/tools/SelectionTool/types.ts +0 -11
  214. package/src/tools/TextTool.ts +0 -326
  215. package/src/tools/ToolController.ts +0 -178
  216. package/src/tools/ToolEnabledGroup.ts +0 -14
  217. package/src/tools/ToolSwitcherShortcut.ts +0 -39
  218. package/src/tools/ToolbarShortcutHandler.ts +0 -34
  219. package/src/tools/UndoRedoShortcut.test.ts +0 -56
  220. package/src/tools/UndoRedoShortcut.ts +0 -25
  221. package/src/tools/lib.ts +0 -21
  222. package/src/tools/localization.ts +0 -66
  223. package/src/types.ts +0 -234
  224. package/src/util/assertions.ts +0 -55
  225. package/src/util/fileToBase64.ts +0 -18
  226. package/src/util/untilNextAnimationFrame.ts +0 -9
  227. package/src/util/waitForTimeout.ts +0 -9
@@ -1,47 +0,0 @@
1
- import Rect2 from '../math/Rect2';
2
-
3
- export interface CommandLocalization {
4
- movedLeft: string;
5
- movedUp: string;
6
- movedDown: string;
7
- movedRight: string;
8
- rotatedBy: (degrees: number) => string;
9
- zoomedOut: string;
10
- zoomedIn: string;
11
- erasedNoElements: string;
12
- duplicatedNoElements: string;
13
- elements: string;
14
- updatedViewport: string;
15
- transformedElements: (elemCount: number) => string;
16
- resizeOutputCommand: (newSize: Rect2) => string;
17
- addElementAction: (elemDescription: string) => string;
18
- eraseAction: (elemDescription: string, numElems: number) => string;
19
- duplicateAction: (elemDescription: string, count: number)=> string;
20
- inverseOf: (actionDescription: string)=> string;
21
- unionOf: (actionDescription: string, actionCount: number)=> string;
22
-
23
- selectedElements: (count: number)=>string;
24
- }
25
-
26
- export const defaultCommandLocalization: CommandLocalization = {
27
- updatedViewport: 'Transformed Viewport',
28
- transformedElements: (elemCount) => `Transformed ${elemCount} element${elemCount === 1 ? '' : 's'}`,
29
- resizeOutputCommand: (newSize: Rect2) => `Resized image to ${newSize.w}x${newSize.h}`,
30
- addElementAction: (componentDescription: string) => `Added ${componentDescription}`,
31
- eraseAction: (componentDescription: string, numElems: number) => `Erased ${numElems} ${componentDescription}`,
32
- duplicateAction: (componentDescription: string, numElems: number) => `Duplicated ${numElems} ${componentDescription}`,
33
- unionOf: (actionDescription: string, actionCount: number) => `Union: ${actionCount} ${actionDescription}`,
34
- inverseOf: (actionDescription: string) => `Inverse of ${actionDescription}`,
35
- elements: 'Elements',
36
- erasedNoElements: 'Erased nothing',
37
- duplicatedNoElements: 'Duplicated nothing',
38
-
39
- rotatedBy: (degrees) => `Rotated by ${Math.abs(degrees)} degrees ${degrees < 0 ? 'clockwise' : 'counter-clockwise'}`,
40
- movedLeft: 'Moved left',
41
- movedUp: 'Moved up',
42
- movedDown: 'Moved down',
43
- movedRight: 'Moved right',
44
- zoomedOut: 'Zoomed out',
45
- zoomedIn: 'Zoomed in',
46
- selectedElements: (count) => `Selected ${count} element${count === 1 ? '' : 's'}`,
47
- };
@@ -1,23 +0,0 @@
1
-
2
- import { Color4, EditorImage, Mat33, Path, SerializableCommand, StrokeComponent, Vec2 } from '../lib';
3
- import uniteCommands from './uniteCommands';
4
- import createEditor from '../testing/createEditor';
5
-
6
- describe('uniteCommands', () => {
7
- it('should be serializable and deserializable', () => {
8
- const editor = createEditor();
9
- const stroke = new StrokeComponent([ Path.fromString('m0,0 l10,10 h-2 z').toRenderable({ fill: Color4.red }) ]);
10
- const union = uniteCommands([
11
- EditorImage.addElement(stroke),
12
- stroke.transformBy(Mat33.translation(Vec2.of(1, 10))),
13
- ]);
14
- const deserialized = SerializableCommand.deserialize(union.serialize(), editor);
15
-
16
- deserialized.apply(editor);
17
-
18
- const lookupResult = editor.image.lookupElement(stroke.getId());
19
- expect(lookupResult).not.toBeNull();
20
- expect(lookupResult?.getBBox().topLeft).toMatchObject(Vec2.of(1, 10));
21
- expect(lookupResult?.getBBox().bottomRight).toMatchObject(Vec2.of(11, 20));
22
- });
23
- });
@@ -1,135 +0,0 @@
1
- import Editor from '../Editor';
2
- import { EditorLocalization } from '../localization';
3
- import Command from './Command';
4
- import SerializableCommand from './SerializableCommand';
5
-
6
-
7
- class NonSerializableUnion extends Command {
8
- public constructor(private commands: Command[], private applyChunkSize: number|undefined) {
9
- super();
10
- }
11
-
12
- private static waitForAll(commands: (Promise<void>|void)[]): Promise<void>|void {
13
- // If any are Promises...
14
- if (commands.some(command => command && command['then'])) {
15
- console.log('waiting...');
16
- // Wait for all commands to finish.
17
- return Promise.all(commands)
18
- // Ensure we return a Promise<void> and not a Promise<void[]>
19
- .then(() => {});
20
- }
21
-
22
- return;
23
- }
24
-
25
- public apply(editor: Editor) {
26
- if (this.applyChunkSize === undefined) {
27
- const results = this.commands.map(cmd => cmd.apply(editor));
28
- return NonSerializableUnion.waitForAll(results);
29
- } else {
30
- return editor.asyncApplyCommands(this.commands, this.applyChunkSize);
31
- }
32
- }
33
-
34
- public unapply(editor: Editor) {
35
- const commands = [ ...this.commands ];
36
- commands.reverse();
37
-
38
- if (this.applyChunkSize === undefined) {
39
- const results = commands.map(cmd => cmd.unapply(editor));
40
- return NonSerializableUnion.waitForAll(results);
41
- } else {
42
- return editor.asyncUnapplyCommands(commands, this.applyChunkSize, false);
43
- }
44
- }
45
-
46
- public description(editor: Editor, localizationTable: EditorLocalization) {
47
- const descriptions: string[] = [];
48
-
49
- let lastDescription: string|null = null;
50
- let duplicateDescriptionCount: number = 0;
51
- for (const part of this.commands) {
52
- const description = part.description(editor, localizationTable);
53
- if (description !== lastDescription && lastDescription !== null) {
54
- descriptions.push(localizationTable.unionOf(lastDescription, duplicateDescriptionCount));
55
- lastDescription = null;
56
- duplicateDescriptionCount = 0;
57
- }
58
-
59
- duplicateDescriptionCount ++;
60
- lastDescription ??= description;
61
- }
62
-
63
- if (duplicateDescriptionCount > 1) {
64
- descriptions.push(localizationTable.unionOf(lastDescription!, duplicateDescriptionCount));
65
- } else if (duplicateDescriptionCount === 1) {
66
- descriptions.push(lastDescription!);
67
- }
68
-
69
- return descriptions.join(', ');
70
- }
71
- }
72
-
73
- class SerializableUnion extends SerializableCommand {
74
- private nonserializableCommand: NonSerializableUnion;
75
- public constructor(private commands: SerializableCommand[], private applyChunkSize: number|undefined) {
76
- super('union');
77
- this.nonserializableCommand = new NonSerializableUnion(commands, applyChunkSize);
78
- }
79
-
80
- protected serializeToJSON() {
81
- return {
82
- applyChunkSize: this.applyChunkSize,
83
- data: this.commands.map(command => command.serialize()),
84
- };
85
- }
86
-
87
- public apply(editor: Editor) {
88
- return this.nonserializableCommand.apply(editor);
89
- }
90
-
91
- public unapply(editor: Editor) {
92
- return this.nonserializableCommand.unapply(editor);
93
- }
94
-
95
- public description(editor: Editor, localizationTable: EditorLocalization): string {
96
- return this.nonserializableCommand.description(editor, localizationTable);
97
- }
98
- }
99
-
100
- const uniteCommands = <T extends Command> (commands: T[], applyChunkSize?: number): T extends SerializableCommand ? SerializableCommand : Command => {
101
- let allSerializable = true;
102
- for (const command of commands) {
103
- if (!(command instanceof SerializableCommand)) {
104
- allSerializable = false;
105
- break;
106
- }
107
- }
108
-
109
- if (!allSerializable) {
110
- return new NonSerializableUnion(commands, applyChunkSize) as any;
111
- } else {
112
- const castedCommands = commands as any[] as SerializableCommand[];
113
- return new SerializableUnion(castedCommands, applyChunkSize);
114
- }
115
- };
116
-
117
- SerializableCommand.register('union', (data: any, editor) => {
118
- if (typeof data.data.length !== 'number') {
119
- throw new Error('Unions of commands must serialize to lists of serialization data.');
120
- }
121
- const applyChunkSize: number|undefined = data.applyChunkSize;
122
- if (typeof applyChunkSize !== 'number' && applyChunkSize !== undefined) {
123
- throw new Error('serialized applyChunkSize is neither undefined nor a number.');
124
- }
125
-
126
- const commands: SerializableCommand[] = [];
127
- for (const part of data.data as any[]) {
128
- commands.push(SerializableCommand.deserialize(part, editor));
129
- }
130
-
131
- return uniteCommands(commands, applyChunkSize);
132
- });
133
-
134
-
135
- export default uniteCommands;
@@ -1,22 +0,0 @@
1
- import { Color4, EditorImage, Mat33, Path, Rect2, Vec2 } from '../lib';
2
- import createEditor from '../testing/createEditor';
3
- import Stroke from './Stroke';
4
-
5
- describe('AbstractComponent.transformBy', () => {
6
- it('should restore the component\'s z-index on undo', () => {
7
- const editor = createEditor();
8
- const component = new Stroke([ Path.fromRect(Rect2.unitSquare).toRenderable({ fill: Color4.red }) ]);
9
- EditorImage.addElement(component).apply(editor);
10
-
11
- const origZIndex = component.getZIndex();
12
-
13
- const transformCommand = component.transformBy(Mat33.translation(Vec2.unitX));
14
- transformCommand.apply(editor);
15
-
16
- // Should increase the z-index on applying a transform
17
- expect(component.getZIndex()).toBeGreaterThan(origZIndex);
18
-
19
- transformCommand.unapply(editor);
20
- expect(component.getZIndex()).toBe(origZIndex);
21
- });
22
- });
@@ -1,364 +0,0 @@
1
- import SerializableCommand from '../commands/SerializableCommand';
2
- import Editor from '../Editor';
3
- import EditorImage from '../EditorImage';
4
- import LineSegment2 from '../math/LineSegment2';
5
- import Mat33, { Mat33Array } from '../math/Mat33';
6
- import Rect2 from '../math/Rect2';
7
- import { EditorLocalization } from '../localization';
8
- import AbstractRenderer from '../rendering/renderers/AbstractRenderer';
9
- import { ImageComponentLocalization } from './localization';
10
- import UnresolvedSerializableCommand from '../commands/UnresolvedCommand';
11
-
12
- export type LoadSaveData = (string[]|Record<symbol, string|number>);
13
- export type LoadSaveDataTable = Record<string, Array<LoadSaveData>>;
14
- export type DeserializeCallback = (data: string)=>AbstractComponent;
15
- type ComponentId = string;
16
-
17
- /**
18
- * A base class for everything that can be added to an {@link EditorImage}.
19
- */
20
- export default abstract class AbstractComponent {
21
- // The timestamp (milliseconds) at which the component was
22
- // last changed (i.e. created/translated).
23
- // @deprecated
24
- protected lastChangedTime: number;
25
-
26
- // The bounding box of this component.
27
- // {@link getBBox}, by default, returns `contentBBox`.
28
- // This must be set by components.
29
- protected abstract contentBBox: Rect2;
30
-
31
- private zIndex: number;
32
- private id: string;
33
-
34
- // Topmost z-index
35
- private static zIndexCounter: number = 0;
36
-
37
- protected constructor(
38
- // A unique identifier for the type of component
39
- private readonly componentKind: string,
40
- initialZIndex?: number,
41
- ) {
42
- this.lastChangedTime = (new Date()).getTime();
43
-
44
- if (initialZIndex !== undefined) {
45
- this.zIndex = initialZIndex;
46
- } else {
47
- this.zIndex = AbstractComponent.zIndexCounter++;
48
- }
49
-
50
- // Create a unique ID.
51
- this.id = `${new Date().getTime()}-${Math.random()}`;
52
-
53
- if (AbstractComponent.deserializationCallbacks[componentKind] === undefined) {
54
- throw new Error(`Component ${componentKind} has not been registered using AbstractComponent.registerComponent`);
55
- }
56
- }
57
-
58
- // Returns a unique ID for this element.
59
- // @see { @link lib!EditorImage.lookupElement }
60
- public getId() {
61
- return this.id;
62
- }
63
-
64
- private static deserializationCallbacks: Record<ComponentId, DeserializeCallback|null> = {};
65
-
66
- // Store the deserialization callback (or lack of it) for [componentKind].
67
- // If components are registered multiple times (as may be done in automated tests),
68
- // the most recent deserialization callback is used.
69
- public static registerComponent(
70
- componentKind: string,
71
- deserialize: DeserializeCallback|null,
72
- ) {
73
- this.deserializationCallbacks[componentKind] = deserialize ?? null;
74
- }
75
-
76
- // Stores data attached by a loader.
77
- private loadSaveData: LoadSaveDataTable = {};
78
-
79
- /**
80
- * Attach data that can be used while exporting the component (e.g. to SVG).
81
- *
82
- * This is intended for use by a {@link ImageLoader}.
83
- */
84
- public attachLoadSaveData(key: string, data: LoadSaveData) {
85
- if (!this.loadSaveData[key]) {
86
- this.loadSaveData[key] = [];
87
- }
88
- this.loadSaveData[key].push(data);
89
- }
90
-
91
- /** See {@link attachLoadSaveData} */
92
- public getLoadSaveData(): LoadSaveDataTable {
93
- return this.loadSaveData;
94
- }
95
-
96
- public getZIndex(): number {
97
- return this.zIndex;
98
- }
99
-
100
- /** @returns the bounding box of */
101
- public getBBox(): Rect2 {
102
- return this.contentBBox;
103
- }
104
-
105
- /** Called when this component is added to the given image. */
106
- public onAddToImage(_image: EditorImage): void { }
107
- public onRemoveFromImage(): void { }
108
-
109
- public abstract render(canvas: AbstractRenderer, visibleRect?: Rect2): void;
110
-
111
- /** @return true if `lineSegment` intersects this component. */
112
- public abstract intersects(lineSegment: LineSegment2): boolean;
113
-
114
- /**
115
- * @returns true if this component intersects `rect` -- it is entirely contained
116
- * within the rectangle or one of the rectangle's edges intersects this component.
117
- */
118
- public intersectsRect(rect: Rect2): boolean {
119
- // If this component intersects rect,
120
- // it is either contained entirely within rect or intersects one of rect's edges.
121
-
122
- // If contained within,
123
- if (rect.containsRect(this.getBBox())) {
124
- return true;
125
- }
126
-
127
- // Calculated bounding boxes can be slightly larger than their actual contents' bounding box.
128
- // As such, test with more lines than just the rect's edges.
129
- const testLines = [];
130
- for (const subregion of rect.divideIntoGrid(2, 2)) {
131
- testLines.push(...subregion.getEdges());
132
- }
133
-
134
- return testLines.some(edge => this.intersects(edge));
135
- }
136
-
137
- // Return null iff this object cannot be safely serialized/deserialized.
138
- protected abstract serializeToJSON(): any[]|Record<string, any>|number|string|null;
139
-
140
- // Private helper for transformBy: Apply the given transformation to all points of this.
141
- protected abstract applyTransformation(affineTransfm: Mat33): void;
142
-
143
- // Returns a command that, when applied, transforms this by [affineTransfm] and
144
- // updates the editor.
145
- public transformBy(affineTransfm: Mat33): SerializableCommand {
146
- return new AbstractComponent.TransformElementCommand(affineTransfm, this.getId(), this);
147
- }
148
-
149
- // Returns a command that updates this component's z-index.
150
- public setZIndex(newZIndex: number): SerializableCommand {
151
- return new AbstractComponent.TransformElementCommand(Mat33.identity, this.getId(), this, newZIndex, this.getZIndex());
152
- }
153
-
154
- // @returns true iff this component can be selected (e.g. by the selection tool.)
155
- public isSelectable(): boolean {
156
- return true;
157
- }
158
-
159
- // @returns true iff this component should be added to the background, rather than the
160
- // foreground of the image.
161
- public isBackground(): boolean {
162
- return false;
163
- }
164
-
165
- // @returns an approximation of the proportional time it takes to render this component.
166
- // This is intended to be a rough estimate, but, for example, a stroke with two points sould have
167
- // a renderingWeight approximately twice that of a stroke with one point.
168
- public getProportionalRenderingTime(): number {
169
- return 1;
170
- }
171
-
172
- private static transformElementCommandId = 'transform-element';
173
-
174
- private static TransformElementCommand = class extends UnresolvedSerializableCommand {
175
- private targetZIndex: number;
176
-
177
- // Construct a new TransformElementCommand. `component`, while optional, should
178
- // be provided if available. If not provided, it will be fetched from the editor's
179
- // document when the command is applied.
180
- public constructor(
181
- private affineTransfm: Mat33,
182
- componentID: string,
183
- component?: AbstractComponent,
184
- targetZIndex?: number,
185
- private origZIndex?: number,
186
- ) {
187
- super(AbstractComponent.transformElementCommandId, componentID, component);
188
- this.targetZIndex = targetZIndex ?? AbstractComponent.zIndexCounter++;
189
-
190
- // Ensure that we keep drawing on top even after changing the z-index.
191
- if (this.targetZIndex >= AbstractComponent.zIndexCounter) {
192
- AbstractComponent.zIndexCounter = this.targetZIndex + 1;
193
- }
194
-
195
- if (component && origZIndex === undefined) {
196
- this.origZIndex = component.getZIndex();
197
- }
198
- }
199
-
200
- protected resolveComponent(image: EditorImage): void {
201
- if (this.component) {
202
- return;
203
- }
204
-
205
- super.resolveComponent(image);
206
- this.origZIndex ??= this.component!.getZIndex();
207
- }
208
-
209
- private updateTransform(editor: Editor, newTransfm: Mat33) {
210
- if (!this.component) {
211
- throw new Error('this.component is undefined or null!');
212
- }
213
-
214
- // Any parent should have only one direct child.
215
- const parent = editor.image.findParent(this.component);
216
- let hadParent = false;
217
- if (parent) {
218
- parent.remove();
219
- hadParent = true;
220
- }
221
-
222
- this.component.applyTransformation(newTransfm);
223
- this.component.lastChangedTime = (new Date()).getTime();
224
-
225
- // Add the element back to the document.
226
- if (hadParent) {
227
- EditorImage.addElement(this.component).apply(editor);
228
- }
229
- }
230
-
231
- public apply(editor: Editor) {
232
- this.resolveComponent(editor.image);
233
-
234
- this.component!.zIndex = this.targetZIndex;
235
- this.updateTransform(editor, this.affineTransfm);
236
- editor.queueRerender();
237
- }
238
-
239
- public unapply(editor: Editor) {
240
- this.resolveComponent(editor.image);
241
-
242
- this.component!.zIndex = this.origZIndex!;
243
- this.updateTransform(editor, this.affineTransfm.inverse());
244
- editor.queueRerender();
245
- }
246
-
247
- public description(_editor: Editor, localizationTable: EditorLocalization) {
248
- return localizationTable.transformedElements(1);
249
- }
250
-
251
- static {
252
- SerializableCommand.register(AbstractComponent.transformElementCommandId, (json: any, editor: Editor) => {
253
- const elem = editor.image.lookupElement(json.id) ?? undefined;
254
- const transform = new Mat33(...(json.transfm as Mat33Array));
255
- const targetZIndex = json.targetZIndex;
256
- const origZIndex = json.origZIndex ?? undefined;
257
-
258
- return new AbstractComponent.TransformElementCommand(
259
- transform,
260
- json.id,
261
- elem,
262
- targetZIndex,
263
- origZIndex,
264
- );
265
- });
266
- }
267
-
268
- protected serializeToJSON() {
269
- return {
270
- id: this.componentID,
271
- transfm: this.affineTransfm.toArray(),
272
- targetZIndex: this.targetZIndex,
273
- origZIndex: this.origZIndex,
274
- };
275
- }
276
- };
277
-
278
- /**
279
- * @return a description that could be read by a screen reader
280
- * (e.g. when adding/erasing the component)
281
- */
282
- public abstract description(localizationTable: ImageComponentLocalization): string;
283
-
284
- // Component-specific implementation of {@link clone}.
285
- protected abstract createClone(): AbstractComponent;
286
-
287
- // Returns a copy of this component.
288
- public clone() {
289
- const clone = this.createClone();
290
-
291
- for (const attachmentKey in this.loadSaveData) {
292
- for (const val of this.loadSaveData[attachmentKey]) {
293
- clone.attachLoadSaveData(attachmentKey, val);
294
- }
295
- }
296
-
297
- return clone;
298
- }
299
-
300
- // Convert the component to an object that can be passed to
301
- // `JSON.stringify`.
302
- //
303
- // Do not rely on the output of this function to take a particular form —
304
- // this function's output can change form without a major version increase.
305
- public serialize() {
306
- const data = this.serializeToJSON();
307
-
308
- if (data === null) {
309
- throw new Error(`${this} cannot be serialized.`);
310
- }
311
-
312
- return {
313
- name: this.componentKind,
314
- zIndex: this.zIndex,
315
- id: this.id,
316
- loadSaveData: this.loadSaveData,
317
- data,
318
- };
319
- }
320
-
321
- // Returns true if `data` is not deserializable. May return false even if [data]
322
- // is not deserializable.
323
- private static isNotDeserializable(json: any|string) {
324
- if (typeof json === 'string') {
325
- json = JSON.parse(json);
326
- }
327
-
328
- if (typeof json !== 'object') {
329
- return true;
330
- }
331
-
332
- if (!this.deserializationCallbacks[json?.name]) {
333
- return true;
334
- }
335
-
336
- if (!json.data) {
337
- return true;
338
- }
339
-
340
- return false;
341
- }
342
-
343
- // Convert a string or an object produced by `JSON.parse` into an `AbstractComponent`.
344
- public static deserialize(json: string|any): AbstractComponent {
345
- if (typeof json === 'string') {
346
- json = JSON.parse(json);
347
- }
348
-
349
- if (AbstractComponent.isNotDeserializable(json)) {
350
- throw new Error(`Element with data ${json} cannot be deserialized.`);
351
- }
352
-
353
- const instance = this.deserializationCallbacks[json.name]!(json.data);
354
- instance.zIndex = json.zIndex;
355
- instance.id = json.id;
356
-
357
- // TODO: What should we do with json.loadSaveData?
358
- // If we attach it to [instance], we create a potential security risk — loadSaveData
359
- // is often used to store unrecognised attributes so they can be preserved on output.
360
- // ...but what if we're deserializing data sent across the network?
361
-
362
- return instance;
363
- }
364
- }
@@ -1,35 +0,0 @@
1
- import Color4 from '../Color4';
2
- import { Path, Rect2 } from '../math/lib';
3
- import createEditor from '../testing/createEditor';
4
- import ImageBackground, { BackgroundType, imageBackgroundCSSClassName } from './ImageBackground';
5
-
6
- describe('ImageBackground', () => {
7
- it('should render to fill exported SVG', () => {
8
- const editor = createEditor();
9
- const background = new ImageBackground(BackgroundType.SolidColor, Color4.green);
10
- editor.image.addElement(
11
- background
12
- ).apply(editor);
13
-
14
- const expectedImportExportRect = new Rect2(-10, 10, 15, 20);
15
- editor.setImportExportRect(expectedImportExportRect).apply(editor);
16
- expect(editor.getImportExportRect()).objEq(expectedImportExportRect);
17
-
18
- expect(background.getBBox()).objEq(expectedImportExportRect);
19
-
20
- const rendered = editor.toSVG();
21
- const renderedBackground = rendered.querySelector(`.${imageBackgroundCSSClassName}`);
22
-
23
- if (renderedBackground === null) {
24
- throw new Error('ImageBackground did not render in exported SVG');
25
- }
26
-
27
- expect(renderedBackground.tagName.toLowerCase()).toBe('path');
28
-
29
- const pathString = renderedBackground.getAttribute('d')!;
30
- expect(pathString).not.toBeNull();
31
-
32
- const path = Path.fromString(pathString);
33
- expect(path.bbox).objEq(editor.getImportExportRect());
34
- });
35
- });