js-draw 1.0.1 → 1.1.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 (197) hide show
  1. package/LICENSE +21 -0
  2. package/dist/Editor.css +1 -0
  3. package/dist/bundle.js +1 -1
  4. package/dist/bundledStyles.js +1 -1
  5. package/dist/cjs/toolbar/AbstractToolbar.d.ts +9 -13
  6. package/dist/cjs/toolbar/AbstractToolbar.js +14 -19
  7. package/dist/cjs/toolbar/widgets/SaveActionWidget.d.ts +10 -0
  8. package/dist/cjs/toolbar/widgets/SaveActionWidget.js +26 -0
  9. package/dist/cjs/toolbar/widgets/keybindings.d.ts +1 -0
  10. package/dist/cjs/toolbar/widgets/keybindings.js +4 -1
  11. package/dist/cjs/version.js +1 -1
  12. package/dist/mjs/toolbar/AbstractToolbar.d.ts +9 -13
  13. package/dist/mjs/toolbar/AbstractToolbar.mjs +14 -19
  14. package/dist/mjs/toolbar/widgets/SaveActionWidget.d.ts +10 -0
  15. package/dist/mjs/toolbar/widgets/SaveActionWidget.mjs +21 -0
  16. package/dist/mjs/toolbar/widgets/keybindings.d.ts +1 -0
  17. package/dist/mjs/toolbar/widgets/keybindings.mjs +3 -0
  18. package/dist/mjs/version.mjs +1 -1
  19. package/docs/img/readme-images/js-draw.jpg +0 -0
  20. package/docs/img/readme-images/unsupported-elements--in-editor.png +0 -0
  21. package/package.json +5 -4
  22. package/src/toolbar/EdgeToolbar.scss +1 -0
  23. package/dist-test/test_imports/package-lock.json +0 -13
  24. package/dist-test/test_imports/package.json +0 -12
  25. package/dist-test/test_imports/test-imports.js +0 -11
  26. package/dist-test/test_imports/test-require.cjs +0 -14
  27. package/src/Editor.loadFrom.test.ts +0 -24
  28. package/src/Editor.test.ts +0 -107
  29. package/src/Editor.toSVG.test.ts +0 -294
  30. package/src/Editor.ts +0 -1443
  31. package/src/EditorImage.test.ts +0 -117
  32. package/src/EditorImage.ts +0 -609
  33. package/src/EventDispatcher.test.ts +0 -123
  34. package/src/EventDispatcher.ts +0 -72
  35. package/src/Pointer.ts +0 -183
  36. package/src/SVGLoader.test.ts +0 -114
  37. package/src/SVGLoader.ts +0 -672
  38. package/src/UndoRedoHistory.test.ts +0 -34
  39. package/src/UndoRedoHistory.ts +0 -102
  40. package/src/Viewport.ts +0 -322
  41. package/src/bundle/bundled.ts +0 -7
  42. package/src/commands/Command.ts +0 -45
  43. package/src/commands/Duplicate.ts +0 -75
  44. package/src/commands/Erase.ts +0 -95
  45. package/src/commands/SerializableCommand.ts +0 -49
  46. package/src/commands/UnresolvedCommand.ts +0 -37
  47. package/src/commands/invertCommand.ts +0 -58
  48. package/src/commands/lib.ts +0 -16
  49. package/src/commands/localization.ts +0 -47
  50. package/src/commands/uniteCommands.test.ts +0 -23
  51. package/src/commands/uniteCommands.ts +0 -140
  52. package/src/components/AbstractComponent.transformBy.test.ts +0 -23
  53. package/src/components/AbstractComponent.ts +0 -383
  54. package/src/components/BackgroundComponent.test.ts +0 -44
  55. package/src/components/BackgroundComponent.ts +0 -348
  56. package/src/components/ImageComponent.ts +0 -176
  57. package/src/components/RestylableComponent.ts +0 -161
  58. package/src/components/SVGGlobalAttributesObject.ts +0 -79
  59. package/src/components/Stroke.test.ts +0 -137
  60. package/src/components/Stroke.ts +0 -294
  61. package/src/components/TextComponent.test.ts +0 -202
  62. package/src/components/TextComponent.ts +0 -429
  63. package/src/components/UnknownSVGObject.test.ts +0 -10
  64. package/src/components/UnknownSVGObject.ts +0 -60
  65. package/src/components/builders/ArrowBuilder.ts +0 -106
  66. package/src/components/builders/CircleBuilder.ts +0 -100
  67. package/src/components/builders/FreehandLineBuilder.test.ts +0 -24
  68. package/src/components/builders/FreehandLineBuilder.ts +0 -210
  69. package/src/components/builders/LineBuilder.ts +0 -77
  70. package/src/components/builders/PressureSensitiveFreehandLineBuilder.ts +0 -453
  71. package/src/components/builders/RectangleBuilder.ts +0 -73
  72. package/src/components/builders/types.ts +0 -15
  73. package/src/components/lib.ts +0 -31
  74. package/src/components/localization.ts +0 -24
  75. package/src/components/util/StrokeSmoother.ts +0 -302
  76. package/src/components/util/describeComponentList.ts +0 -18
  77. package/src/dialogs/makeAboutDialog.ts +0 -82
  78. package/src/inputEvents.ts +0 -143
  79. package/src/lib.ts +0 -91
  80. package/src/localization.ts +0 -34
  81. package/src/localizations/de.ts +0 -146
  82. package/src/localizations/en.ts +0 -8
  83. package/src/localizations/es.ts +0 -74
  84. package/src/localizations/getLocalizationTable.test.ts +0 -27
  85. package/src/localizations/getLocalizationTable.ts +0 -74
  86. package/src/rendering/Display.ts +0 -247
  87. package/src/rendering/RenderablePathSpec.ts +0 -88
  88. package/src/rendering/RenderingStyle.test.ts +0 -68
  89. package/src/rendering/RenderingStyle.ts +0 -55
  90. package/src/rendering/TextRenderingStyle.ts +0 -55
  91. package/src/rendering/caching/CacheRecord.test.ts +0 -48
  92. package/src/rendering/caching/CacheRecord.ts +0 -76
  93. package/src/rendering/caching/CacheRecordManager.ts +0 -71
  94. package/src/rendering/caching/RenderingCache.test.ts +0 -43
  95. package/src/rendering/caching/RenderingCache.ts +0 -66
  96. package/src/rendering/caching/RenderingCacheNode.ts +0 -404
  97. package/src/rendering/caching/testUtils.ts +0 -35
  98. package/src/rendering/caching/types.ts +0 -34
  99. package/src/rendering/lib.ts +0 -8
  100. package/src/rendering/localization.ts +0 -20
  101. package/src/rendering/renderers/AbstractRenderer.ts +0 -232
  102. package/src/rendering/renderers/CanvasRenderer.ts +0 -312
  103. package/src/rendering/renderers/DummyRenderer.test.ts +0 -41
  104. package/src/rendering/renderers/DummyRenderer.ts +0 -142
  105. package/src/rendering/renderers/SVGRenderer.ts +0 -434
  106. package/src/rendering/renderers/TextOnlyRenderer.test.ts +0 -34
  107. package/src/rendering/renderers/TextOnlyRenderer.ts +0 -68
  108. package/src/shortcuts/KeyBinding.test.ts +0 -61
  109. package/src/shortcuts/KeyBinding.ts +0 -257
  110. package/src/shortcuts/KeyboardShortcutManager.test.ts +0 -95
  111. package/src/shortcuts/KeyboardShortcutManager.ts +0 -163
  112. package/src/shortcuts/lib.ts +0 -3
  113. package/src/testing/createEditor.ts +0 -11
  114. package/src/testing/getUniquePointerId.ts +0 -18
  115. package/src/testing/lib.ts +0 -3
  116. package/src/testing/sendPenEvent.ts +0 -36
  117. package/src/testing/sendTouchEvent.ts +0 -71
  118. package/src/toolbar/AbstractToolbar.ts +0 -542
  119. package/src/toolbar/DropdownToolbar.ts +0 -220
  120. package/src/toolbar/EdgeToolbar.test.ts +0 -54
  121. package/src/toolbar/EdgeToolbar.ts +0 -543
  122. package/src/toolbar/IconProvider.ts +0 -861
  123. package/src/toolbar/constants.ts +0 -1
  124. package/src/toolbar/lib.ts +0 -6
  125. package/src/toolbar/localization.ts +0 -136
  126. package/src/toolbar/types.ts +0 -13
  127. package/src/toolbar/widgets/ActionButtonWidget.ts +0 -39
  128. package/src/toolbar/widgets/BaseToolWidget.ts +0 -81
  129. package/src/toolbar/widgets/BaseWidget.ts +0 -495
  130. package/src/toolbar/widgets/DocumentPropertiesWidget.ts +0 -250
  131. package/src/toolbar/widgets/EraserToolWidget.ts +0 -84
  132. package/src/toolbar/widgets/HandToolWidget.ts +0 -239
  133. package/src/toolbar/widgets/InsertImageWidget.ts +0 -248
  134. package/src/toolbar/widgets/OverflowWidget.ts +0 -92
  135. package/src/toolbar/widgets/PenToolWidget.ts +0 -369
  136. package/src/toolbar/widgets/SelectionToolWidget.ts +0 -195
  137. package/src/toolbar/widgets/TextToolWidget.ts +0 -149
  138. package/src/toolbar/widgets/components/makeColorInput.ts +0 -184
  139. package/src/toolbar/widgets/components/makeFileInput.ts +0 -128
  140. package/src/toolbar/widgets/components/makeGridSelector.ts +0 -179
  141. package/src/toolbar/widgets/components/makeSeparator.ts +0 -17
  142. package/src/toolbar/widgets/components/makeThicknessSlider.ts +0 -62
  143. package/src/toolbar/widgets/keybindings.ts +0 -19
  144. package/src/toolbar/widgets/layout/DropdownLayoutManager.ts +0 -262
  145. package/src/toolbar/widgets/layout/EdgeToolbarLayoutManager.ts +0 -71
  146. package/src/toolbar/widgets/layout/types.ts +0 -74
  147. package/src/toolbar/widgets/lib.ts +0 -13
  148. package/src/tools/BaseTool.ts +0 -169
  149. package/src/tools/Eraser.test.ts +0 -103
  150. package/src/tools/Eraser.ts +0 -173
  151. package/src/tools/FindTool.test.ts +0 -67
  152. package/src/tools/FindTool.ts +0 -153
  153. package/src/tools/InputFilter/FunctionMapper.ts +0 -17
  154. package/src/tools/InputFilter/InputMapper.ts +0 -41
  155. package/src/tools/InputFilter/InputPipeline.test.ts +0 -41
  156. package/src/tools/InputFilter/InputPipeline.ts +0 -34
  157. package/src/tools/InputFilter/InputStabilizer.ts +0 -254
  158. package/src/tools/InputFilter/StrokeKeyboardControl.ts +0 -104
  159. package/src/tools/PanZoom.test.ts +0 -339
  160. package/src/tools/PanZoom.ts +0 -525
  161. package/src/tools/PasteHandler.ts +0 -94
  162. package/src/tools/Pen.test.ts +0 -260
  163. package/src/tools/Pen.ts +0 -284
  164. package/src/tools/PipetteTool.ts +0 -84
  165. package/src/tools/SelectionTool/SelectAllShortcutHandler.ts +0 -29
  166. package/src/tools/SelectionTool/Selection.ts +0 -647
  167. package/src/tools/SelectionTool/SelectionHandle.ts +0 -142
  168. package/src/tools/SelectionTool/SelectionTool.test.ts +0 -370
  169. package/src/tools/SelectionTool/SelectionTool.ts +0 -510
  170. package/src/tools/SelectionTool/TransformMode.ts +0 -112
  171. package/src/tools/SelectionTool/types.ts +0 -11
  172. package/src/tools/SoundUITool.ts +0 -221
  173. package/src/tools/TextTool.ts +0 -339
  174. package/src/tools/ToolController.ts +0 -224
  175. package/src/tools/ToolEnabledGroup.ts +0 -14
  176. package/src/tools/ToolSwitcherShortcut.ts +0 -39
  177. package/src/tools/ToolbarShortcutHandler.ts +0 -39
  178. package/src/tools/UndoRedoShortcut.test.ts +0 -62
  179. package/src/tools/UndoRedoShortcut.ts +0 -24
  180. package/src/tools/keybindings.ts +0 -85
  181. package/src/tools/lib.ts +0 -22
  182. package/src/tools/localization.ts +0 -76
  183. package/src/types.ts +0 -151
  184. package/src/util/ReactiveValue.test.ts +0 -168
  185. package/src/util/ReactiveValue.ts +0 -241
  186. package/src/util/assertions.ts +0 -55
  187. package/src/util/fileToBase64.ts +0 -18
  188. package/src/util/guessKeyCodeFromKey.ts +0 -36
  189. package/src/util/listPrefixMatch.ts +0 -19
  190. package/src/util/stopPropagationOfScrollingWheelEvents.ts +0 -20
  191. package/src/util/untilNextAnimationFrame.ts +0 -9
  192. package/src/util/waitForAll.ts +0 -18
  193. package/src/util/waitForTimeout.ts +0 -9
  194. package/src/version.test.ts +0 -12
  195. package/src/version.ts +0 -3
  196. package/tools/allLocales.js +0 -4
  197. package/tools/copyREADME.ts +0 -62
@@ -1,404 +0,0 @@
1
-
2
- // A cache record with sub-nodes.
3
-
4
- import { ImageNode, sortLeavesByZIndex } from '../../EditorImage';
5
- import { Rect2, Color4 } from '@js-draw/math';
6
- import Viewport from '../../Viewport';
7
- import AbstractRenderer from '../renderers/AbstractRenderer';
8
- import CacheRecord from './CacheRecord';
9
- import { CacheState } from './types';
10
-
11
- // 3x3 divisions for each node.
12
- const cacheDivisionSize = 3;
13
-
14
- // True: Show rendering updates.
15
- const debugMode = false;
16
-
17
- export default class RenderingCacheNode {
18
- // invariant: instantiatedChildren.length === 9
19
- private instantiatedChildren: RenderingCacheNode[] = [];
20
- private parent: RenderingCacheNode|null = null;
21
-
22
- private cachedRenderer: CacheRecord|null = null;
23
- // invariant: sortedInAscendingOrder(renderedIds)
24
- private renderedIds: Array<number> = [];
25
- private renderedMaxZIndex: number|null = null;
26
-
27
- public constructor(
28
- public readonly region: Rect2, private readonly cacheState: CacheState
29
- ) {
30
- }
31
-
32
- // Creates a previous layer of the cache tree and adds this as a child near the
33
- // center of the previous layer's children.
34
- // Returns this' parent if it already exists.
35
- public generateParent(): RenderingCacheNode {
36
- if (this.parent) {
37
- return this.parent;
38
- }
39
-
40
- const parentRegion = Rect2.fromCorners(
41
- this.region.topLeft.minus(this.region.size),
42
- this.region.bottomRight.plus(this.region.size)
43
- );
44
- const parent = new RenderingCacheNode(parentRegion, this.cacheState);
45
- parent.generateChildren();
46
-
47
- // Ensure the new node is matches the middle child's region.
48
- const checkTolerance = this.region.maxDimension / 100;
49
- const middleChildIdx = (parent.instantiatedChildren.length - 1) / 2;
50
- if (!parent.instantiatedChildren[middleChildIdx].region.eq(this.region, checkTolerance)) {
51
- console.error(parent.instantiatedChildren[middleChildIdx].region, '≠', this.region);
52
- throw new Error('Logic error: [this] is not contained within its parent\'s center child');
53
- }
54
-
55
- // Replace the middle child
56
- parent.instantiatedChildren[middleChildIdx] = this;
57
- this.parent = parent;
58
-
59
- return parent;
60
- }
61
-
62
- // Generates children, if missing.
63
- private generateChildren() {
64
- if (this.instantiatedChildren.length === 0) {
65
- const childRects = this.region.divideIntoGrid(cacheDivisionSize, cacheDivisionSize);
66
-
67
- if (this.region.size.x === 0 || this.region.size.y === 0) {
68
- console.warn('Cache element has zero size! Not generating children.');
69
- return;
70
- }
71
-
72
- for (const rect of childRects) {
73
- const child = new RenderingCacheNode(rect, this.cacheState);
74
- child.parent = this;
75
- this.instantiatedChildren.push(child);
76
- }
77
- }
78
- this.checkRep();
79
- }
80
-
81
- // Returns CacheNodes directly contained within this.
82
- private getChildren(): RenderingCacheNode[] {
83
- this.checkRep();
84
- this.generateChildren();
85
-
86
- return this.instantiatedChildren;
87
- }
88
-
89
- public smallestChildContaining(rect: Rect2): RenderingCacheNode|null {
90
- const largerThanChildren = rect.maxDimension > this.region.maxDimension / cacheDivisionSize;
91
- if (!this.region.containsRect(rect) || largerThanChildren) {
92
- return null;
93
- }
94
-
95
- for (const child of this.getChildren()) {
96
- if (child.region.containsRect(rect)) {
97
- return child.smallestChildContaining(rect) ?? child;
98
- }
99
- }
100
-
101
- return null;
102
- }
103
-
104
- // => [true] iff [this] can be rendered without too much scaling
105
- private renderingWouldBeHighEnoughResolution(viewport: Viewport) {
106
- // Determine how 1px in this corresponds to 1px on the canvas.
107
- // this.region.w is in canvas units. Thus,
108
- const sizeOfThisPixelOnCanvas = this.region.w / this.cacheState.props.blockResolution.x;
109
- const sizeOfThisPixelOnScreen = viewport.getScaleFactor() * sizeOfThisPixelOnCanvas;
110
-
111
- if (sizeOfThisPixelOnScreen > this.cacheState.props.maxScale) {
112
- return false;
113
- }
114
- return true;
115
- }
116
-
117
- // => [true] if all children of this can be rendered from their caches.
118
- private allChildrenCanRender(viewport: Viewport, leavesSortedById: ImageNode[]) {
119
- if (this.instantiatedChildren.length === 0) {
120
- return false;
121
- }
122
-
123
- for (const child of this.instantiatedChildren) {
124
- if (!child.region.intersects(viewport.visibleRect)) {
125
- continue;
126
- }
127
-
128
- if (!child.renderingIsUpToDate(this.idsOfIntersecting(leavesSortedById))) {
129
- return false;
130
- }
131
- }
132
-
133
- return true;
134
- }
135
-
136
- private computeSortedByLeafIds(leaves: ImageNode[]) {
137
- const ids = leaves.slice();
138
- ids.sort((a, b) => a.getId() - b.getId());
139
- return ids;
140
- }
141
-
142
- // Returns a list of the ids of the nodes intersecting this
143
- private idsOfIntersecting(nodes: ImageNode[]) {
144
- const result = [];
145
- for (const node of nodes) {
146
- if (node.getBBox().intersects(this.region)) {
147
- result.push(node.getId());
148
- }
149
- }
150
- return result;
151
- }
152
-
153
- // Returns true iff all elems of this.renderedIds are in sortedIds.
154
- // sortedIds should be sorted by z-index (or some other order, so long as they are
155
- // sorted by the same thing as this.renderedIds.)
156
- private allRenderedIdsIn(sortedIds: number[]) {
157
- if (this.renderedIds.length > sortedIds.length) {
158
- return false;
159
- }
160
-
161
- for (let i = 0; i < this.renderedIds.length; i++) {
162
- if (sortedIds[i] !== this.renderedIds[i]) {
163
- return false;
164
- }
165
- }
166
-
167
- return true;
168
- }
169
-
170
- private renderingIsUpToDate(sortedIds: number[]) {
171
- if (this.cachedRenderer === null || sortedIds.length !== this.renderedIds.length) {
172
- return false;
173
- }
174
-
175
- return this.allRenderedIdsIn(sortedIds);
176
- }
177
-
178
- // Render all [items] within [viewport]
179
- public renderItems(screenRenderer: AbstractRenderer, items: ImageNode[], viewport: Viewport) {
180
- if (
181
- !viewport.visibleRect.intersects(this.region)
182
- || items.length === 0
183
- ) {
184
- return;
185
- }
186
-
187
- const newItems = [];
188
- // Divide [items] until nodes are leaves or smaller than this
189
- for (const item of items) {
190
- const bbox = item.getBBox();
191
- if (!bbox.intersects(this.region)) {
192
- continue;
193
- }
194
-
195
- if (bbox.maxDimension >= this.region.maxDimension) {
196
- newItems.push(...item.getChildrenOrSelfIntersectingRegion(this.region));
197
- } else {
198
- newItems.push(item);
199
- }
200
- }
201
- items = newItems;
202
-
203
- // Can we cache at all?
204
- if (!this.cacheState.props.isOfCorrectType(screenRenderer)) {
205
- items.forEach(item => item.render(screenRenderer, viewport.visibleRect));
206
- return;
207
- }
208
-
209
- if (debugMode) {
210
- screenRenderer.drawRect(this.region, 0.5 * viewport.getSizeOfPixelOnCanvas(), { fill: Color4.yellow });
211
- }
212
-
213
- // Could we render direclty from [this] or do we need to recurse?
214
- const couldRender = this.renderingWouldBeHighEnoughResolution(viewport);
215
- if (!couldRender) {
216
- for (const child of this.getChildren()) {
217
- child.renderItems(screenRenderer, items.filter(item => {
218
- return item.getBBox().intersects(child.region);
219
- }), viewport);
220
- }
221
- } else {
222
- // Determine whether we already have rendered the items
223
- const tooSmallToRender = (rect: Rect2) => rect.w / this.region.w < 1 / this.cacheState.props.blockResolution.x;
224
-
225
- const leaves = [];
226
- for (const item of items) {
227
- leaves.push(
228
- ...item.getLeavesIntersectingRegion(
229
- this.region, tooSmallToRender,
230
- )
231
- );
232
- }
233
- sortLeavesByZIndex(leaves);
234
- const leavesByIds = this.computeSortedByLeafIds(leaves);
235
-
236
- // No intersecting leaves? No need to render
237
- if (leavesByIds.length === 0) {
238
- return;
239
- }
240
-
241
- const leafIds = leavesByIds.map(leaf => leaf.getId());
242
-
243
- let thisRenderer;
244
- if (!this.renderingIsUpToDate(leafIds)) {
245
- if (this.allChildrenCanRender(viewport, leavesByIds)) {
246
- for (const child of this.getChildren()) {
247
- child.renderItems(screenRenderer, items, viewport);
248
- }
249
- return;
250
- }
251
-
252
- let leafApproxRenderTime = 0;
253
- for (const leaf of leavesByIds) {
254
- leafApproxRenderTime += leaf.getContent()!.getProportionalRenderingTime();
255
- }
256
-
257
- // Is it worth it to render the items?
258
- if (leafApproxRenderTime > this.cacheState.props.minProportionalRenderTimePerCache) {
259
- let fullRerenderNeeded = true;
260
- if (!this.cachedRenderer) {
261
- this.cachedRenderer = this.cacheState.recordManager.allocCanvas(
262
- this.region,
263
- () => this.onRegionDealloc()
264
- );
265
- } else if (leavesByIds.length > this.renderedIds.length && this.allRenderedIdsIn(leafIds) && this.renderedMaxZIndex !== null) {
266
- // We often don't need to do a full re-render even if something's changed.
267
- // Check whether we can just draw on top of the existing cache.
268
- const newLeaves = [];
269
-
270
- let minNewZIndex: number|null = null;
271
-
272
- for (let i = 0; i < leavesByIds.length; i++) {
273
- const leaf = leavesByIds[i];
274
- const content = leaf.getContent()!;
275
-
276
- const zIndex = content.getZIndex();
277
- if (i >= this.renderedIds.length || leaf.getId() !== this.renderedIds[i]) {
278
- newLeaves.push(leaf);
279
-
280
- if (minNewZIndex === null || zIndex < minNewZIndex) {
281
- minNewZIndex = zIndex;
282
- }
283
- }
284
- }
285
-
286
- if (minNewZIndex !== null && minNewZIndex > this.renderedMaxZIndex!) {
287
- fullRerenderNeeded = false;
288
- thisRenderer = this.cachedRenderer.startRender();
289
-
290
- // Looping is faster than re-sorting.
291
- for (let i = 0; i < leaves.length; i++) {
292
- const leaf = leaves[i];
293
- const zIndex = leaf.getContent()!.getZIndex();
294
-
295
- if (zIndex > this.renderedMaxZIndex) {
296
- leaf.render(thisRenderer, this.region);
297
- this.renderedMaxZIndex = zIndex;
298
- }
299
- }
300
-
301
- if (debugMode) {
302
- screenRenderer.drawRect(this.region, viewport.getSizeOfPixelOnCanvas(), { fill: Color4.clay });
303
- }
304
- }
305
- } else if (debugMode) {
306
- console.log('Decided on a full re-render. Reason: At least one of the following is false:',
307
- '\n leafIds.length > this.renderedIds.length: ', leafIds.length > this.renderedIds.length,
308
- '\n this.allRenderedIdsIn(leafIds): ', this.allRenderedIdsIn(leafIds),
309
- '\n this.renderedMaxZIndex !== null: ', this.renderedMaxZIndex !== null,
310
- '\n\nthis.rerenderedIds: ', this.renderedIds, ', leafIds: ', leafIds);
311
- }
312
-
313
- if (fullRerenderNeeded) {
314
- thisRenderer = this.cachedRenderer.startRender();
315
- thisRenderer.clear();
316
-
317
- this.renderedMaxZIndex = null;
318
- for (const leaf of leaves) {
319
- const content = leaf.getContent()!;
320
- this.renderedMaxZIndex ??= content.getZIndex();
321
- this.renderedMaxZIndex = Math.max(this.renderedMaxZIndex, content.getZIndex());
322
-
323
- leaf.render(thisRenderer, this.region);
324
- }
325
-
326
- if (debugMode) {
327
- screenRenderer.drawRect(this.region, viewport.getSizeOfPixelOnCanvas(), { fill: Color4.red });
328
- }
329
- }
330
- this.renderedIds = leafIds;
331
- } else {
332
- this.cachedRenderer?.dealloc();
333
-
334
- // Slightly increase the clip region to prevent seams.
335
- // Divide by two because grownBy expands the rectangle on all sides.
336
- const pixelSize = viewport.getSizeOfPixelOnCanvas();
337
- const expandedRegion = new Rect2(
338
- this.region.x, this.region.y,
339
- this.region.w + pixelSize, this.region.h + pixelSize
340
- );
341
-
342
- const clip = true;
343
- screenRenderer.startObject(expandedRegion, clip);
344
- for (const leaf of leaves) {
345
- leaf.render(screenRenderer, this.region.intersection(viewport.visibleRect)!);
346
- }
347
-
348
- screenRenderer.endObject();
349
- }
350
- } else {
351
- thisRenderer = this.cachedRenderer!.startRender();
352
- }
353
-
354
- if (thisRenderer) {
355
- const transformMat = this.cachedRenderer!.getTransform(this.region).inverse();
356
- screenRenderer.renderFromOtherOfSameType(transformMat, thisRenderer);
357
- }
358
-
359
- // Can we clean up this' children? (Are they unused?)
360
- if (this.instantiatedChildren.every(child => child.isEmpty())) {
361
- this.instantiatedChildren = [];
362
- }
363
- }
364
-
365
- this.checkRep();
366
- }
367
-
368
- // Returns true iff this/its children have no cached state.
369
- private isEmpty(): boolean {
370
- if (this.cachedRenderer !== null) {
371
- return false;
372
- }
373
-
374
- return this.instantiatedChildren.every(child => child.isEmpty());
375
- }
376
-
377
- private onRegionDealloc() {
378
- this.cachedRenderer = null;
379
- if (this.isEmpty()) {
380
- this.instantiatedChildren = [];
381
- }
382
- }
383
-
384
- private checkRep() {
385
- if (this.instantiatedChildren.length !== cacheDivisionSize * cacheDivisionSize && this.instantiatedChildren.length !== 0) {
386
- throw new Error(`Repcheck: Wrong number of children. Got ${this.instantiatedChildren.length}`);
387
- }
388
-
389
- if (this.renderedIds[1] !== undefined && this.renderedIds[0] >= this.renderedIds[1]) {
390
- console.error(this.renderedIds);
391
- throw new Error('Repcheck: First two ids are not in ascending order!');
392
- }
393
-
394
- for (const child of this.instantiatedChildren) {
395
- if (child.parent !== this) {
396
- throw new Error('Children should be linked to their parents!');
397
- }
398
- }
399
-
400
- if (this.cachedRenderer && !this.cachedRenderer.isAllocd()) {
401
- throw new Error('this\' cachedRenderer != null, but is dealloc\'d');
402
- }
403
- }
404
- }
@@ -1,35 +0,0 @@
1
- import { Vec2 } from '@js-draw/math';
2
- import DummyRenderer from '../renderers/DummyRenderer';
3
- import createEditor from '../../testing/createEditor';
4
- import AbstractRenderer from '../renderers/AbstractRenderer';
5
- import RenderingCache from './RenderingCache';
6
- import { CacheProps } from './types';
7
-
8
- type RenderAllocCallback = (renderer: DummyRenderer)=> void;
9
-
10
- // Override any default test options with [cacheOptions]
11
- export const createCache = (onRenderAlloc?: RenderAllocCallback, cacheOptions?: Partial<CacheProps>) => {
12
- const editor = createEditor();
13
-
14
- const cache = new RenderingCache({
15
- createRenderer() {
16
- const renderer = new DummyRenderer(editor.viewport);
17
- onRenderAlloc?.(renderer);
18
- return renderer;
19
- },
20
- isOfCorrectType(renderer: AbstractRenderer) {
21
- return renderer instanceof DummyRenderer;
22
- },
23
- blockResolution: Vec2.of(500, 500),
24
- cacheSize: 500 * 10 * 4,
25
- maxScale: 2,
26
- minProportionalRenderTimePerCache: 0,
27
- minProportionalRenderTimeToUseCache: 0,
28
- ...cacheOptions
29
- });
30
-
31
- return {
32
- cache,
33
- editor
34
- };
35
- };
@@ -1,34 +0,0 @@
1
- import type { Vec2 } from '@js-draw/math';
2
- import type AbstractRenderer from '../renderers/AbstractRenderer';
3
- import type { CacheRecordManager } from './CacheRecordManager';
4
-
5
-
6
- export type CacheAddress = number;
7
- export type BeforeDeallocCallback = ()=>void;
8
-
9
-
10
- export interface CacheProps {
11
- createRenderer(): AbstractRenderer;
12
- // Returns whether the cache can be rendered onto [renderer].
13
- isOfCorrectType(renderer: AbstractRenderer): boolean;
14
-
15
- blockResolution: Vec2;
16
- cacheSize: number;
17
-
18
- // Maximum amount a cached image can be scaled without a re-render
19
- // (larger numbers = blurrier, but faster)
20
- maxScale: number;
21
-
22
- // Minimum component count to cache, rather than just re-render each time.
23
- minProportionalRenderTimePerCache: number;
24
-
25
- // Minimum number of strokes/etc. to use the cache to render, isntead of
26
- // rendering directly.
27
- minProportionalRenderTimeToUseCache: number;
28
- }
29
-
30
- export interface CacheState {
31
- currentRenderingCycle: number;
32
- props: CacheProps;
33
- recordManager: CacheRecordManager;
34
- }
@@ -1,8 +0,0 @@
1
-
2
- export { default as AbstractRenderer } from './renderers/AbstractRenderer';
3
- export { default as DummyRenderer } from './renderers/DummyRenderer';
4
- export { default as SVGRenderer } from './renderers/SVGRenderer';
5
- export { default as CanvasRenderer } from './renderers/CanvasRenderer';
6
- export { default as Display, RenderingMode } from './Display';
7
- export { default as TextRenderingStyle } from './TextRenderingStyle';
8
- export { default as RenderingStyle } from './RenderingStyle';
@@ -1,20 +0,0 @@
1
-
2
- export interface TextRendererLocalization {
3
- pathNodeCount(pathCount: number): string;
4
- textNodeCount(nodeCount: number): string;
5
- imageNodeCount(nodeCount: number): string;
6
- textNode(content: string): string;
7
- unlabeledImageNode: string;
8
- imageNode(label: string): string;
9
- rerenderAsText: string;
10
- }
11
-
12
- export const defaultTextRendererLocalization: TextRendererLocalization = {
13
- pathNodeCount: (count: number) => `There are ${count} visible path objects.`,
14
- textNodeCount: (count: number) => `There are ${count} visible text nodes.`,
15
- imageNodeCount: (nodeCount: number) => `There are ${nodeCount} visible image nodes.`,
16
- textNode: (content: string) => `Text: ${content}`,
17
- imageNode: (label: string) => `Image: ${label}`,
18
- unlabeledImageNode: 'Unlabeled image',
19
- rerenderAsText: 'Re-render as text',
20
- };