js-draw 0.0.10 → 0.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 (95) hide show
  1. package/CHANGELOG.md +4 -0
  2. package/dist/bundle.js +1 -1
  3. package/dist/src/Editor.d.ts +2 -2
  4. package/dist/src/Editor.js +13 -7
  5. package/dist/src/EditorImage.d.ts +15 -7
  6. package/dist/src/EditorImage.js +41 -35
  7. package/dist/src/SVGLoader.d.ts +3 -2
  8. package/dist/src/SVGLoader.js +9 -7
  9. package/dist/src/Viewport.d.ts +4 -0
  10. package/dist/src/Viewport.js +41 -0
  11. package/dist/src/components/AbstractComponent.d.ts +3 -2
  12. package/dist/src/components/AbstractComponent.js +3 -0
  13. package/dist/src/components/SVGGlobalAttributesObject.d.ts +1 -1
  14. package/dist/src/components/SVGGlobalAttributesObject.js +1 -1
  15. package/dist/src/components/Stroke.d.ts +1 -1
  16. package/dist/src/components/UnknownSVGObject.d.ts +1 -1
  17. package/dist/src/components/UnknownSVGObject.js +1 -1
  18. package/dist/src/components/builders/ArrowBuilder.d.ts +1 -1
  19. package/dist/src/components/builders/FreehandLineBuilder.d.ts +1 -1
  20. package/dist/src/components/builders/LineBuilder.d.ts +1 -1
  21. package/dist/src/components/builders/RectangleBuilder.d.ts +1 -1
  22. package/dist/src/components/builders/types.d.ts +1 -1
  23. package/dist/src/geometry/Mat33.js +3 -0
  24. package/dist/src/geometry/Path.d.ts +1 -1
  25. package/dist/src/geometry/Path.js +5 -3
  26. package/dist/src/geometry/Rect2.d.ts +1 -0
  27. package/dist/src/geometry/Rect2.js +28 -1
  28. package/dist/src/{Display.d.ts → rendering/Display.d.ts} +5 -2
  29. package/dist/src/{Display.js → rendering/Display.js} +33 -4
  30. package/dist/src/rendering/caching/CacheRecord.d.ts +19 -0
  31. package/dist/src/rendering/caching/CacheRecord.js +51 -0
  32. package/dist/src/rendering/caching/CacheRecordManager.d.ts +11 -0
  33. package/dist/src/rendering/caching/CacheRecordManager.js +39 -0
  34. package/dist/src/rendering/caching/RenderingCache.d.ts +12 -0
  35. package/dist/src/rendering/caching/RenderingCache.js +36 -0
  36. package/dist/src/rendering/caching/RenderingCacheNode.d.ts +28 -0
  37. package/dist/src/rendering/caching/RenderingCacheNode.js +294 -0
  38. package/dist/src/rendering/caching/testUtils.d.ts +9 -0
  39. package/dist/src/rendering/caching/testUtils.js +20 -0
  40. package/dist/src/rendering/caching/types.d.ts +20 -0
  41. package/dist/src/rendering/caching/types.js +1 -0
  42. package/dist/src/rendering/{AbstractRenderer.d.ts → renderers/AbstractRenderer.d.ts} +18 -8
  43. package/dist/src/rendering/{AbstractRenderer.js → renderers/AbstractRenderer.js} +36 -2
  44. package/dist/src/rendering/{CanvasRenderer.d.ts → renderers/CanvasRenderer.d.ts} +10 -5
  45. package/dist/src/rendering/{CanvasRenderer.js → renderers/CanvasRenderer.js} +60 -20
  46. package/dist/src/rendering/{DummyRenderer.d.ts → renderers/DummyRenderer.d.ts} +9 -5
  47. package/dist/src/rendering/{DummyRenderer.js → renderers/DummyRenderer.js} +35 -4
  48. package/dist/src/rendering/{SVGRenderer.d.ts → renderers/SVGRenderer.d.ts} +4 -3
  49. package/dist/src/rendering/{SVGRenderer.js → renderers/SVGRenderer.js} +14 -11
  50. package/dist/src/testing/createEditor.js +1 -1
  51. package/dist/src/toolbar/HTMLToolbar.js +1 -1
  52. package/dist/src/tools/SelectionTool.js +9 -24
  53. package/dist/src/types.d.ts +2 -1
  54. package/package.json +1 -1
  55. package/src/Editor.ts +15 -8
  56. package/src/EditorImage.test.ts +2 -2
  57. package/src/EditorImage.ts +53 -41
  58. package/src/SVGLoader.ts +11 -8
  59. package/src/Viewport.ts +56 -0
  60. package/src/components/AbstractComponent.ts +6 -2
  61. package/src/components/SVGGlobalAttributesObject.ts +2 -2
  62. package/src/components/Stroke.ts +1 -1
  63. package/src/components/UnknownSVGObject.ts +2 -2
  64. package/src/components/builders/ArrowBuilder.ts +1 -1
  65. package/src/components/builders/FreehandLineBuilder.ts +1 -1
  66. package/src/components/builders/LineBuilder.ts +1 -1
  67. package/src/components/builders/RectangleBuilder.ts +1 -1
  68. package/src/components/builders/types.ts +1 -1
  69. package/src/geometry/Mat33.ts +3 -0
  70. package/src/geometry/Path.toString.test.ts +12 -2
  71. package/src/geometry/Path.ts +8 -4
  72. package/src/geometry/Rect2.test.ts +38 -8
  73. package/src/geometry/Rect2.ts +32 -1
  74. package/src/{Display.ts → rendering/Display.ts} +38 -6
  75. package/src/rendering/caching/CacheRecord.test.ts +49 -0
  76. package/src/rendering/caching/CacheRecord.ts +72 -0
  77. package/src/rendering/caching/CacheRecordManager.ts +55 -0
  78. package/src/rendering/caching/RenderingCache.test.ts +44 -0
  79. package/src/rendering/caching/RenderingCache.ts +56 -0
  80. package/src/rendering/caching/RenderingCacheNode.ts +365 -0
  81. package/src/rendering/caching/testUtils.ts +34 -0
  82. package/src/rendering/caching/types.ts +35 -0
  83. package/src/rendering/{AbstractRenderer.ts → renderers/AbstractRenderer.ts} +55 -8
  84. package/src/rendering/{CanvasRenderer.ts → renderers/CanvasRenderer.ts} +74 -25
  85. package/src/rendering/renderers/DummyRenderer.test.ts +43 -0
  86. package/src/rendering/{DummyRenderer.ts → renderers/DummyRenderer.ts} +50 -7
  87. package/src/rendering/{SVGRenderer.ts → renderers/SVGRenderer.ts} +17 -13
  88. package/src/testing/createEditor.ts +1 -1
  89. package/src/toolbar/HTMLToolbar.ts +1 -1
  90. package/src/tools/SelectionTool.test.ts +1 -1
  91. package/src/tools/SelectionTool.ts +12 -33
  92. package/src/types.ts +10 -3
  93. package/tsconfig.json +1 -0
  94. package/dist/__mocks__/coloris.d.ts +0 -2
  95. package/dist/__mocks__/coloris.js +0 -5
@@ -0,0 +1 @@
1
+ export {};
@@ -1,8 +1,9 @@
1
- import Color4 from '../Color4';
2
- import { PathCommand } from '../geometry/Path';
3
- import Rect2 from '../geometry/Rect2';
4
- import { Point2, Vec2 } from '../geometry/Vec2';
5
- import Viewport from '../Viewport';
1
+ import Color4 from '../../Color4';
2
+ import Mat33 from '../../geometry/Mat33';
3
+ import { PathCommand } from '../../geometry/Path';
4
+ import Rect2 from '../../geometry/Rect2';
5
+ import { Point2, Vec2 } from '../../geometry/Vec2';
6
+ import Viewport from '../../Viewport';
6
7
  export interface RenderingStyle {
7
8
  fill: Color4;
8
9
  stroke?: {
@@ -16,8 +17,10 @@ export interface RenderablePathSpec {
16
17
  style: RenderingStyle;
17
18
  }
18
19
  export default abstract class AbstractRenderer {
19
- protected viewport: Viewport;
20
+ private viewport;
21
+ private selfTransform;
20
22
  protected constructor(viewport: Viewport);
23
+ protected getViewport(): Viewport;
21
24
  abstract displaySize(): Vec2;
22
25
  abstract clear(): void;
23
26
  protected abstract beginPath(startPoint: Point2): void;
@@ -26,14 +29,21 @@ export default abstract class AbstractRenderer {
26
29
  protected abstract moveTo(point: Point2): void;
27
30
  protected abstract traceCubicBezierCurve(p1: Point2, p2: Point2, p3: Point2): void;
28
31
  protected abstract traceQuadraticBezierCurve(controlPoint: Point2, endPoint: Point2): void;
32
+ abstract isTooSmallToRender(rect: Rect2): boolean;
29
33
  setDraftMode(_draftMode: boolean): void;
30
- private objectLevel;
34
+ protected objectLevel: number;
31
35
  private currentPaths;
32
36
  private flushPath;
33
37
  drawPath(path: RenderablePathSpec): void;
34
38
  drawRect(rect: Rect2, lineWidth: number, lineFill: RenderingStyle): void;
35
- startObject(_boundingBox: Rect2): void;
39
+ startObject(_boundingBox: Rect2, _clip?: boolean): void;
36
40
  endObject(): void;
37
41
  protected getNestingLevel(): number;
38
42
  abstract drawPoints(...points: Point2[]): void;
43
+ canRenderFromWithoutDataLoss(_other: AbstractRenderer): boolean;
44
+ renderFromOtherOfSameType(_renderTo: Mat33, other: AbstractRenderer): void;
45
+ setTransform(transform: Mat33 | null): void;
46
+ getCanvasToScreenTransform(): Mat33;
47
+ canvasToScreen(vec: Vec2): Vec2;
48
+ getSizeOfCanvasPixelOnScreen(): number;
39
49
  }
@@ -1,4 +1,5 @@
1
- import Path, { PathCommandType } from '../geometry/Path';
1
+ import Path, { PathCommandType } from '../../geometry/Path';
2
+ import { Vec2 } from '../../geometry/Vec2';
2
3
  const stylesEqual = (a, b) => {
3
4
  var _a, _b, _c, _d, _e;
4
5
  return a === b || (a.fill.eq(b.fill)
@@ -8,9 +9,14 @@ const stylesEqual = (a, b) => {
8
9
  export default class AbstractRenderer {
9
10
  constructor(viewport) {
10
11
  this.viewport = viewport;
12
+ // If null, this' transformation is linked to the Viewport
13
+ this.selfTransform = null;
11
14
  this.objectLevel = 0;
12
15
  this.currentPaths = null;
13
16
  }
17
+ // this.canvasToScreen, etc. should be used instead of the corresponding
18
+ // methods on Viewport.
19
+ getViewport() { return this.viewport; }
14
20
  setDraftMode(_draftMode) { }
15
21
  flushPath() {
16
22
  if (!this.currentPaths) {
@@ -68,7 +74,8 @@ export default class AbstractRenderer {
68
74
  this.drawPath(path.toRenderable(lineFill));
69
75
  }
70
76
  // Note the start/end of an object with the given bounding box.
71
- startObject(_boundingBox) {
77
+ // Renderers are not required to support [clip]
78
+ startObject(_boundingBox, _clip) {
72
79
  this.currentPaths = [];
73
80
  this.objectLevel++;
74
81
  }
@@ -84,4 +91,31 @@ export default class AbstractRenderer {
84
91
  getNestingLevel() {
85
92
  return this.objectLevel;
86
93
  }
94
+ // Returns true iff other can be rendered onto this without data loss.
95
+ canRenderFromWithoutDataLoss(_other) {
96
+ return false;
97
+ }
98
+ // MUST throw if other and this are not of the same base class.
99
+ renderFromOtherOfSameType(_renderTo, other) {
100
+ throw new Error(`Unable to render from ${other}: Not implemented`);
101
+ }
102
+ // Set a transformation to apply to things before rendering,
103
+ // replacing the viewport's transform.
104
+ setTransform(transform) {
105
+ this.selfTransform = transform;
106
+ }
107
+ // Get the matrix that transforms a vector on the canvas to a vector on this'
108
+ // rendering target.
109
+ getCanvasToScreenTransform() {
110
+ if (this.selfTransform) {
111
+ return this.selfTransform;
112
+ }
113
+ return this.viewport.canvasToScreenTransform;
114
+ }
115
+ canvasToScreen(vec) {
116
+ return this.getCanvasToScreenTransform().transformVec2(vec);
117
+ }
118
+ getSizeOfCanvasPixelOnScreen() {
119
+ return this.getCanvasToScreenTransform().transformVec3(Vec2.unitX).length();
120
+ }
87
121
  }
@@ -1,7 +1,8 @@
1
- import Rect2 from '../geometry/Rect2';
2
- import { Point2, Vec2 } from '../geometry/Vec2';
3
- import Vec3 from '../geometry/Vec3';
4
- import Viewport from '../Viewport';
1
+ import Mat33 from '../../geometry/Mat33';
2
+ import Rect2 from '../../geometry/Rect2';
3
+ import { Point2, Vec2 } from '../../geometry/Vec2';
4
+ import Vec3 from '../../geometry/Vec3';
5
+ import Viewport from '../../Viewport';
5
6
  import AbstractRenderer, { RenderablePathSpec, RenderingStyle } from './AbstractRenderer';
6
7
  export default class CanvasRenderer extends AbstractRenderer {
7
8
  private ctx;
@@ -11,6 +12,8 @@ export default class CanvasRenderer extends AbstractRenderer {
11
12
  private minRenderSizeAnyDimen;
12
13
  private minRenderSizeBothDimens;
13
14
  constructor(ctx: CanvasRenderingContext2D, viewport: Viewport);
15
+ canRenderFromWithoutDataLoss(other: AbstractRenderer): boolean;
16
+ renderFromOtherOfSameType(transformBy: Mat33, other: AbstractRenderer): void;
14
17
  setDraftMode(draftMode: boolean): void;
15
18
  displaySize(): Vec2;
16
19
  clear(): void;
@@ -21,7 +24,9 @@ export default class CanvasRenderer extends AbstractRenderer {
21
24
  protected traceCubicBezierCurve(p1: Point2, p2: Point2, p3: Point2): void;
22
25
  protected traceQuadraticBezierCurve(controlPoint: Vec3, endPoint: Vec3): void;
23
26
  drawPath(path: RenderablePathSpec): void;
24
- startObject(boundingBox: Rect2): void;
27
+ private clipLevels;
28
+ startObject(boundingBox: Rect2, clip: boolean): void;
25
29
  endObject(): void;
26
30
  drawPoints(...points: Point2[]): void;
31
+ isTooSmallToRender(rect: Rect2): boolean;
27
32
  }
@@ -1,5 +1,5 @@
1
- import Color4 from '../Color4';
2
- import { Vec2 } from '../geometry/Vec2';
1
+ import Color4 from '../../Color4';
2
+ import { Vec2 } from '../../geometry/Vec2';
3
3
  import AbstractRenderer from './AbstractRenderer';
4
4
  export default class CanvasRenderer extends AbstractRenderer {
5
5
  constructor(ctx, viewport) {
@@ -7,8 +7,29 @@ export default class CanvasRenderer extends AbstractRenderer {
7
7
  this.ctx = ctx;
8
8
  this.ignoreObjectsAboveLevel = null;
9
9
  this.ignoringObject = false;
10
+ this.clipLevels = [];
10
11
  this.setDraftMode(false);
11
12
  }
13
+ canRenderFromWithoutDataLoss(other) {
14
+ return other instanceof CanvasRenderer;
15
+ }
16
+ renderFromOtherOfSameType(transformBy, other) {
17
+ if (!(other instanceof CanvasRenderer)) {
18
+ throw new Error(`${other} cannot be rendered onto ${this}`);
19
+ }
20
+ transformBy = this.getCanvasToScreenTransform().rightMul(transformBy);
21
+ this.ctx.save();
22
+ // From MDN, transform(a,b,c,d,e,f)
23
+ // takes input such that
24
+ // ⎡ a c e ⎤
25
+ // ⎢ b d f ⎥ transforms content drawn to [ctx].
26
+ // ⎣ 0 0 1 ⎦
27
+ this.ctx.transform(transformBy.a1, transformBy.b1, // a, b
28
+ transformBy.a2, transformBy.b2, // c, d
29
+ transformBy.a3, transformBy.b3);
30
+ this.ctx.drawImage(other.ctx.canvas, 0, 0);
31
+ this.ctx.restore();
32
+ }
12
33
  // Set parameters for lower/higher quality rendering
13
34
  setDraftMode(draftMode) {
14
35
  if (draftMode) {
@@ -29,7 +50,7 @@ export default class CanvasRenderer extends AbstractRenderer {
29
50
  this.ctx.clearRect(0, 0, this.ctx.canvas.width, this.ctx.canvas.height);
30
51
  }
31
52
  beginPath(startPoint) {
32
- startPoint = this.viewport.canvasToScreen(startPoint);
53
+ startPoint = this.canvasToScreen(startPoint);
33
54
  this.ctx.beginPath();
34
55
  this.ctx.moveTo(startPoint.x, startPoint.y);
35
56
  }
@@ -38,23 +59,23 @@ export default class CanvasRenderer extends AbstractRenderer {
38
59
  this.ctx.fill();
39
60
  if (style.stroke) {
40
61
  this.ctx.strokeStyle = style.stroke.color.toHexString();
41
- this.ctx.lineWidth = this.viewport.getScaleFactor() * style.stroke.width;
62
+ this.ctx.lineWidth = this.getSizeOfCanvasPixelOnScreen() * style.stroke.width;
42
63
  this.ctx.stroke();
43
64
  }
44
65
  this.ctx.closePath();
45
66
  }
46
67
  lineTo(point) {
47
- point = this.viewport.canvasToScreen(point);
68
+ point = this.canvasToScreen(point);
48
69
  this.ctx.lineTo(point.x, point.y);
49
70
  }
50
71
  moveTo(point) {
51
- point = this.viewport.canvasToScreen(point);
72
+ point = this.canvasToScreen(point);
52
73
  this.ctx.moveTo(point.x, point.y);
53
74
  }
54
75
  traceCubicBezierCurve(p1, p2, p3) {
55
- p1 = this.viewport.canvasToScreen(p1);
56
- p2 = this.viewport.canvasToScreen(p2);
57
- p3 = this.viewport.canvasToScreen(p3);
76
+ p1 = this.canvasToScreen(p1);
77
+ p2 = this.canvasToScreen(p2);
78
+ p3 = this.canvasToScreen(p3);
58
79
  // Approximate the curve if small enough.
59
80
  const delta1 = p2.minus(p1);
60
81
  const delta2 = p3.minus(p2);
@@ -67,8 +88,8 @@ export default class CanvasRenderer extends AbstractRenderer {
67
88
  }
68
89
  }
69
90
  traceQuadraticBezierCurve(controlPoint, endPoint) {
70
- controlPoint = this.viewport.canvasToScreen(controlPoint);
71
- endPoint = this.viewport.canvasToScreen(endPoint);
91
+ controlPoint = this.canvasToScreen(controlPoint);
92
+ endPoint = this.canvasToScreen(endPoint);
72
93
  // Approximate the curve with a line if small enough
73
94
  const delta = controlPoint.minus(endPoint);
74
95
  if (delta.magnitudeSquared() < this.minSquareCurveApproxDist) {
@@ -84,20 +105,30 @@ export default class CanvasRenderer extends AbstractRenderer {
84
105
  }
85
106
  super.drawPath(path);
86
107
  }
87
- startObject(boundingBox) {
88
- // Should we ignore all objects within this object's bbox?
89
- const diagonal = this.viewport.canvasToScreenTransform.transformVec3(boundingBox.size);
90
- const bothDimenMinSize = this.minRenderSizeBothDimens;
91
- const bothTooSmall = Math.abs(diagonal.x) < bothDimenMinSize && Math.abs(diagonal.y) < bothDimenMinSize;
92
- const anyDimenMinSize = this.minRenderSizeAnyDimen;
93
- const anyTooSmall = Math.abs(diagonal.x) < anyDimenMinSize || Math.abs(diagonal.y) < anyDimenMinSize;
94
- if (bothTooSmall || anyTooSmall) {
108
+ startObject(boundingBox, clip) {
109
+ if (this.isTooSmallToRender(boundingBox)) {
95
110
  this.ignoreObjectsAboveLevel = this.getNestingLevel();
96
111
  this.ignoringObject = true;
97
112
  }
98
113
  super.startObject(boundingBox);
114
+ if (!this.ignoringObject && clip) {
115
+ this.clipLevels.push(this.objectLevel);
116
+ this.ctx.save();
117
+ this.ctx.beginPath();
118
+ for (const corner of boundingBox.corners) {
119
+ const screenCorner = this.canvasToScreen(corner);
120
+ this.ctx.lineTo(screenCorner.x, screenCorner.y);
121
+ }
122
+ this.ctx.clip();
123
+ }
99
124
  }
100
125
  endObject() {
126
+ if (!this.ignoringObject && this.clipLevels.length > 0) {
127
+ if (this.clipLevels[this.clipLevels.length - 1] === this.objectLevel) {
128
+ this.ctx.restore();
129
+ this.clipLevels.pop();
130
+ }
131
+ }
101
132
  super.endObject();
102
133
  // If exiting an object with a too-small-to-draw bounding box,
103
134
  if (this.ignoreObjectsAboveLevel !== null && this.getNestingLevel() <= this.ignoreObjectsAboveLevel) {
@@ -108,7 +139,7 @@ export default class CanvasRenderer extends AbstractRenderer {
108
139
  drawPoints(...points) {
109
140
  const pointRadius = 10;
110
141
  for (let i = 0; i < points.length; i++) {
111
- const point = this.viewport.canvasToScreen(points[i]);
142
+ const point = this.canvasToScreen(points[i]);
112
143
  this.ctx.beginPath();
113
144
  this.ctx.arc(point.x, point.y, pointRadius, 0, Math.PI * 2);
114
145
  this.ctx.fillStyle = Color4.ofRGBA(0.5 + Math.sin(i) / 2, 1.0, 0.5 + Math.cos(i * 0.2) / 4, 0.5).toHexString();
@@ -121,4 +152,13 @@ export default class CanvasRenderer extends AbstractRenderer {
121
152
  this.ctx.fillText(`${i}`, point.x, point.y, pointRadius * 2);
122
153
  }
123
154
  }
155
+ isTooSmallToRender(rect) {
156
+ // Should we ignore all objects within this object's bbox?
157
+ const diagonal = this.getCanvasToScreenTransform().transformVec3(rect.size);
158
+ const bothDimenMinSize = this.minRenderSizeBothDimens;
159
+ const bothTooSmall = Math.abs(diagonal.x) < bothDimenMinSize && Math.abs(diagonal.y) < bothDimenMinSize;
160
+ const anyDimenMinSize = this.minRenderSizeAnyDimen;
161
+ const anyTooSmall = Math.abs(diagonal.x) < anyDimenMinSize || Math.abs(diagonal.y) < anyDimenMinSize;
162
+ return bothTooSmall || anyTooSmall;
163
+ }
124
164
  }
@@ -1,7 +1,8 @@
1
- import Rect2 from '../geometry/Rect2';
2
- import { Point2, Vec2 } from '../geometry/Vec2';
3
- import Vec3 from '../geometry/Vec3';
4
- import Viewport from '../Viewport';
1
+ import Mat33 from '../../geometry/Mat33';
2
+ import Rect2 from '../../geometry/Rect2';
3
+ import { Point2, Vec2 } from '../../geometry/Vec2';
4
+ import Vec3 from '../../geometry/Vec3';
5
+ import Viewport from '../../Viewport';
5
6
  import AbstractRenderer, { RenderingStyle } from './AbstractRenderer';
6
7
  export default class DummyRenderer extends AbstractRenderer {
7
8
  clearedCount: number;
@@ -20,6 +21,9 @@ export default class DummyRenderer extends AbstractRenderer {
20
21
  protected traceCubicBezierCurve(p1: Vec3, p2: Vec3, p3: Vec3): void;
21
22
  protected traceQuadraticBezierCurve(controlPoint: Vec3, endPoint: Vec3): void;
22
23
  drawPoints(..._points: Vec3[]): void;
23
- startObject(boundingBox: Rect2): void;
24
+ startObject(boundingBox: Rect2, _clip: boolean): void;
24
25
  endObject(): void;
26
+ isTooSmallToRender(_rect: Rect2): boolean;
27
+ canRenderFromWithoutDataLoss(other: AbstractRenderer): boolean;
28
+ renderFromOtherOfSameType(transform: Mat33, other: AbstractRenderer): void;
25
29
  }
@@ -1,5 +1,5 @@
1
1
  // Renderer that outputs nothing. Useful for automated tests.
2
- import { Vec2 } from '../geometry/Vec2';
2
+ import { Vec2 } from '../../geometry/Vec2';
3
3
  import AbstractRenderer from './AbstractRenderer';
4
4
  export default class DummyRenderer extends AbstractRenderer {
5
5
  constructor(viewport) {
@@ -14,8 +14,15 @@ export default class DummyRenderer extends AbstractRenderer {
14
14
  this.pointBuffer = [];
15
15
  }
16
16
  displaySize() {
17
- // Return a dummy
18
- return Vec2.of(640, 480);
17
+ // Do we have a stored viewport size?
18
+ const viewportSize = this.getViewport().getResolution();
19
+ // Don't use a 0x0 viewport — DummyRenderer is often used
20
+ // for tests that run without a display, so pretend we have a
21
+ // reasonable-sized display.
22
+ if (viewportSize.x === 0 || viewportSize.y === 0) {
23
+ return Vec2.of(640, 480);
24
+ }
25
+ return viewportSize;
19
26
  }
20
27
  clear() {
21
28
  this.clearedCount++;
@@ -35,18 +42,25 @@ export default class DummyRenderer extends AbstractRenderer {
35
42
  this.lastFillStyle = style;
36
43
  }
37
44
  lineTo(point) {
45
+ point = this.canvasToScreen(point);
38
46
  this.lastPoint = point;
39
47
  this.pointBuffer.push(point);
40
48
  }
41
49
  moveTo(point) {
50
+ point = this.canvasToScreen(point);
42
51
  this.lastPoint = point;
43
52
  this.pointBuffer.push(point);
44
53
  }
45
54
  traceCubicBezierCurve(p1, p2, p3) {
55
+ p1 = this.canvasToScreen(p1);
56
+ p2 = this.canvasToScreen(p2);
57
+ p3 = this.canvasToScreen(p3);
46
58
  this.lastPoint = p3;
47
59
  this.pointBuffer.push(p1, p2, p3);
48
60
  }
49
61
  traceQuadraticBezierCurve(controlPoint, endPoint) {
62
+ controlPoint = this.canvasToScreen(controlPoint);
63
+ endPoint = this.canvasToScreen(endPoint);
50
64
  this.lastPoint = endPoint;
51
65
  this.pointBuffer.push(controlPoint, endPoint);
52
66
  }
@@ -54,7 +68,7 @@ export default class DummyRenderer extends AbstractRenderer {
54
68
  // drawPoints is intended for debugging.
55
69
  // As such, it is unlikely to be the target of automated tests.
56
70
  }
57
- startObject(boundingBox) {
71
+ startObject(boundingBox, _clip) {
58
72
  super.startObject(boundingBox);
59
73
  this.objectNestingLevel += 1;
60
74
  }
@@ -62,4 +76,21 @@ export default class DummyRenderer extends AbstractRenderer {
62
76
  super.endObject();
63
77
  this.objectNestingLevel -= 1;
64
78
  }
79
+ isTooSmallToRender(_rect) {
80
+ return false;
81
+ }
82
+ canRenderFromWithoutDataLoss(other) {
83
+ return other instanceof DummyRenderer;
84
+ }
85
+ renderFromOtherOfSameType(transform, other) {
86
+ if (!(other instanceof DummyRenderer)) {
87
+ throw new Error(`${other} cannot be rendered onto ${this}`);
88
+ }
89
+ this.renderedPathCount += other.renderedPathCount;
90
+ this.lastFillStyle = other.lastFillStyle;
91
+ this.lastPoint = other.lastPoint;
92
+ this.pointBuffer.push(...other.pointBuffer.map(point => {
93
+ return transform.transformVec2(point);
94
+ }));
95
+ }
65
96
  }
@@ -1,6 +1,6 @@
1
- import Rect2 from '../geometry/Rect2';
2
- import { Point2, Vec2 } from '../geometry/Vec2';
3
- import Viewport from '../Viewport';
1
+ import Rect2 from '../../geometry/Rect2';
2
+ import { Point2, Vec2 } from '../../geometry/Vec2';
3
+ import Viewport from '../../Viewport';
4
4
  import AbstractRenderer, { RenderingStyle } from './AbstractRenderer';
5
5
  export default class SVGRenderer extends AbstractRenderer {
6
6
  private elem;
@@ -26,4 +26,5 @@ export default class SVGRenderer extends AbstractRenderer {
26
26
  protected traceQuadraticBezierCurve(controlPoint: Point2, endPoint: Point2): void;
27
27
  drawPoints(...points: Point2[]): void;
28
28
  drawSVGElem(elem: SVGElement): void;
29
+ isTooSmallToRender(_rect: Rect2): boolean;
29
30
  }
@@ -1,5 +1,5 @@
1
- import Path, { PathCommandType } from '../geometry/Path';
2
- import { Vec2 } from '../geometry/Vec2';
1
+ import Path, { PathCommandType } from '../../geometry/Path';
2
+ import { Vec2 } from '../../geometry/Vec2';
3
3
  import AbstractRenderer from './AbstractRenderer';
4
4
  const svgNameSpace = 'http://www.w3.org/2000/svg';
5
5
  export default class SVGRenderer extends AbstractRenderer {
@@ -44,7 +44,7 @@ export default class SVGRenderer extends AbstractRenderer {
44
44
  beginPath(startPoint) {
45
45
  var _a;
46
46
  this.currentPath = [];
47
- this.pathStart = this.viewport.canvasToScreen(startPoint);
47
+ this.pathStart = this.canvasToScreen(startPoint);
48
48
  (_a = this.lastPathStart) !== null && _a !== void 0 ? _a : (this.lastPathStart = this.pathStart);
49
49
  }
50
50
  endPath(style) {
@@ -98,23 +98,23 @@ export default class SVGRenderer extends AbstractRenderer {
98
98
  this.addPathToSVG();
99
99
  }
100
100
  lineTo(point) {
101
- point = this.viewport.canvasToScreen(point);
101
+ point = this.canvasToScreen(point);
102
102
  this.currentPath.push({
103
103
  kind: PathCommandType.LineTo,
104
104
  point,
105
105
  });
106
106
  }
107
107
  moveTo(point) {
108
- point = this.viewport.canvasToScreen(point);
108
+ point = this.canvasToScreen(point);
109
109
  this.currentPath.push({
110
110
  kind: PathCommandType.MoveTo,
111
111
  point,
112
112
  });
113
113
  }
114
114
  traceCubicBezierCurve(controlPoint1, controlPoint2, endPoint) {
115
- controlPoint1 = this.viewport.canvasToScreen(controlPoint1);
116
- controlPoint2 = this.viewport.canvasToScreen(controlPoint2);
117
- endPoint = this.viewport.canvasToScreen(endPoint);
115
+ controlPoint1 = this.canvasToScreen(controlPoint1);
116
+ controlPoint2 = this.canvasToScreen(controlPoint2);
117
+ endPoint = this.canvasToScreen(endPoint);
118
118
  this.currentPath.push({
119
119
  kind: PathCommandType.CubicBezierTo,
120
120
  controlPoint1,
@@ -123,8 +123,8 @@ export default class SVGRenderer extends AbstractRenderer {
123
123
  });
124
124
  }
125
125
  traceQuadraticBezierCurve(controlPoint, endPoint) {
126
- controlPoint = this.viewport.canvasToScreen(controlPoint);
127
- endPoint = this.viewport.canvasToScreen(endPoint);
126
+ controlPoint = this.canvasToScreen(controlPoint);
127
+ endPoint = this.canvasToScreen(endPoint);
128
128
  this.currentPath.push({
129
129
  kind: PathCommandType.QuadraticBezierTo,
130
130
  controlPoint,
@@ -140,8 +140,11 @@ export default class SVGRenderer extends AbstractRenderer {
140
140
  this.mainGroup.appendChild(elem);
141
141
  });
142
142
  }
143
- // Renders a copy of the given element.
143
+ // Renders a **copy** of the given element.
144
144
  drawSVGElem(elem) {
145
145
  this.elem.appendChild(elem.cloneNode(true));
146
146
  }
147
+ isTooSmallToRender(_rect) {
148
+ return false;
149
+ }
147
150
  }
@@ -1,3 +1,3 @@
1
- import { RenderingMode } from '../Display';
1
+ import { RenderingMode } from '../rendering/Display';
2
2
  import Editor from '../Editor';
3
3
  export default () => new Editor(document.body, { renderingMode: RenderingMode.DummyRenderer });
@@ -7,7 +7,7 @@ import Eraser from '../tools/Eraser';
7
7
  import SelectionTool from '../tools/SelectionTool';
8
8
  import { makeFreehandLineBuilder } from '../components/builders/FreehandLineBuilder';
9
9
  import { Vec2 } from '../geometry/Vec2';
10
- import SVGRenderer from '../rendering/SVGRenderer';
10
+ import SVGRenderer from '../rendering/renderers/SVGRenderer';
11
11
  import Viewport from '../Viewport';
12
12
  import EventDispatcher from '../EventDispatcher';
13
13
  import { makeArrowBuilder } from '../components/builders/ArrowBuilder';
@@ -13,7 +13,6 @@ import Mat33 from '../geometry/Mat33';
13
13
  import Rect2 from '../geometry/Rect2';
14
14
  import { Vec2 } from '../geometry/Vec2';
15
15
  import { EditorEventType } from '../types';
16
- import Viewport from '../Viewport';
17
16
  import BaseTool from './BaseTool';
18
17
  import { ToolType } from './ToolController';
19
18
  const handleScreenSize = 30;
@@ -115,7 +114,7 @@ const makeDraggable = (element, onDrag, onDragEnd) => {
115
114
  element.addEventListener('pointercancel', onPointerEnd);
116
115
  };
117
116
  // Maximum number of strokes to transform without a re-render.
118
- const updateChunkSize = 50;
117
+ const updateChunkSize = 100;
119
118
  class Selection {
120
119
  constructor(startPoint, editor) {
121
120
  this.startPoint = startPoint;
@@ -285,10 +284,13 @@ class Selection {
285
284
  if (this.region.containsRect(elem.getBBox())) {
286
285
  return true;
287
286
  }
288
- else if (this.region.getEdges().some(edge => elem.intersects(edge))) {
289
- return true;
287
+ // Calculated bounding boxes can be slightly larger than their actual contents' bounding box.
288
+ // As such, test with more lines than just this' edges.
289
+ const testLines = [];
290
+ for (const subregion of this.region.divideIntoGrid(2, 2)) {
291
+ testLines.push(...subregion.getEdges());
290
292
  }
291
- return false;
293
+ return testLines.some(edge => elem.intersects(edge));
292
294
  });
293
295
  // Find the bounding box of all selected elements.
294
296
  if (!this.recomputeRegion()) {
@@ -392,26 +394,9 @@ export default class SelectionTool extends BaseTool {
392
394
  tool: this,
393
395
  });
394
396
  if (hasSelection) {
395
- const visibleRect = this.editor.viewport.visibleRect;
396
- const selectionRect = this.selectionBox.region;
397
397
  this.editor.announceForAccessibility(this.editor.localization.selectedElements(this.selectionBox.getSelectedItemCount()));
398
- // Try to move the selection within the center 2/3rds of the viewport.
399
- const targetRect = visibleRect.transformedBoundingBox(Mat33.scaling2D(2 / 3, visibleRect.center));
400
- // Ensure that the selection fits within the target
401
- if (targetRect.w < selectionRect.w || targetRect.h < selectionRect.h) {
402
- const multiplier = Math.max(selectionRect.w / targetRect.w, selectionRect.h / targetRect.h);
403
- const visibleRectTransform = Mat33.scaling2D(multiplier, targetRect.topLeft);
404
- const viewportContentTransform = visibleRectTransform.inverse();
405
- (new Viewport.ViewportTransform(viewportContentTransform)).apply(this.editor);
406
- }
407
- // Ensure that the top left is visible
408
- if (!targetRect.containsRect(selectionRect)) {
409
- // target position - current position
410
- const translation = selectionRect.center.minus(targetRect.center);
411
- const visibleRectTransform = Mat33.translation(translation);
412
- const viewportContentTransform = visibleRectTransform.inverse();
413
- (new Viewport.ViewportTransform(viewportContentTransform)).apply(this.editor);
414
- }
398
+ const selectionRect = this.selectionBox.region;
399
+ this.editor.viewport.zoomTo(selectionRect).apply(this.editor);
415
400
  }
416
401
  }
417
402
  onPointerUp(event) {
@@ -89,8 +89,9 @@ export interface ColorPickerToggled {
89
89
  export declare type EditorEventDataType = EditorToolEvent | EditorObjectEvent | EditorViewportChangedEvent | DisplayResizedEvent | EditorUndoStackUpdated | ColorPickerToggled;
90
90
  export declare type OnProgressListener = (amountProcessed: number, totalToProcess: number) => Promise<void> | null;
91
91
  export declare type ComponentAddedListener = (component: AbstractComponent) => void;
92
+ export declare type OnDetermineExportRectListener = (exportRect: Rect2) => void;
92
93
  export interface ImageLoader {
93
- start(onAddComponent: ComponentAddedListener, onProgressListener: OnProgressListener): Promise<Rect2>;
94
+ start(onAddComponent: ComponentAddedListener, onProgressListener: OnProgressListener, onDetermineExportRect?: OnDetermineExportRectListener): Promise<void>;
94
95
  }
95
96
  export interface StrokeDataPoint {
96
97
  pos: Point2;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "js-draw",
3
- "version": "0.0.10",
3
+ "version": "0.1.0",
4
4
  "description": "Draw pictures using a pen, touchscreen, or mouse! JS-draw is a drawing library for JavaScript and TypeScript. ",
5
5
  "main": "dist/src/Editor.js",
6
6
  "types": "dist/src/Editor.d.ts",
package/src/Editor.ts CHANGED
@@ -9,9 +9,9 @@ import EventDispatcher from './EventDispatcher';
9
9
  import { Point2, Vec2 } from './geometry/Vec2';
10
10
  import Vec3 from './geometry/Vec3';
11
11
  import HTMLToolbar from './toolbar/HTMLToolbar';
12
- import { RenderablePathSpec } from './rendering/AbstractRenderer';
13
- import Display, { RenderingMode } from './Display';
14
- import SVGRenderer from './rendering/SVGRenderer';
12
+ import { RenderablePathSpec } from './rendering/renderers/AbstractRenderer';
13
+ import Display, { RenderingMode } from './rendering/Display';
14
+ import SVGRenderer from './rendering/renderers/SVGRenderer';
15
15
  import Color4 from './Color4';
16
16
  import SVGLoader from './SVGLoader';
17
17
  import Pointer from './Pointer';
@@ -380,7 +380,8 @@ export class Editor {
380
380
  );
381
381
  }
382
382
 
383
- this.image.render(renderer, this.viewport);
383
+ //this.image.render(renderer, this.viewport);
384
+ this.image.renderWithCache(renderer, this.display.getCache(), this.viewport);
384
385
  this.rerenderQueued = false;
385
386
  }
386
387
 
@@ -463,22 +464,28 @@ export class Editor {
463
464
 
464
465
  public async loadFrom(loader: ImageLoader) {
465
466
  this.showLoadingWarning(0);
466
- const imageRect = await loader.start((component) => {
467
+ this.display.setDraftMode(true);
468
+
469
+ await loader.start((component) => {
467
470
  (new EditorImage.AddElementCommand(component)).apply(this);
468
471
  }, (countProcessed: number, totalToProcess: number) => {
469
- if (countProcessed % 100 === 0) {
472
+ if (countProcessed % 500 === 0) {
470
473
  this.showLoadingWarning(countProcessed / totalToProcess);
471
- this.rerender(false);
474
+ this.rerender();
472
475
  return new Promise(resolve => {
473
476
  requestAnimationFrame(() => resolve());
474
477
  });
475
478
  }
476
479
 
477
480
  return null;
481
+ }, (importExportRect: Rect2) => {
482
+ this.setImportExportRect(importExportRect).apply(this);
483
+ this.viewport.zoomTo(importExportRect).apply(this);
478
484
  });
479
485
  this.hideLoadingWarning();
480
486
 
481
- this.setImportExportRect(imageRect).apply(this);
487
+ this.display.setDraftMode(false);
488
+ this.queueRerender();
482
489
  }
483
490
 
484
491
  // Returns the size of the visible region of the output SVG
@@ -5,8 +5,8 @@ import Stroke from './components/Stroke';
5
5
  import { Vec2 } from './geometry/Vec2';
6
6
  import Path, { PathCommandType } from './geometry/Path';
7
7
  import Color4 from './Color4';
8
- import DummyRenderer from './rendering/DummyRenderer';
9
- import { RenderingStyle } from './rendering/AbstractRenderer';
8
+ import DummyRenderer from './rendering/renderers/DummyRenderer';
9
+ import { RenderingStyle } from './rendering/renderers/AbstractRenderer';
10
10
  import createEditor from './testing/createEditor';
11
11
 
12
12
  describe('EditorImage', () => {