@pooder/kit 4.3.1 → 5.0.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 (60) hide show
  1. package/.test-dist/src/CanvasService.js +249 -0
  2. package/.test-dist/src/ViewportSystem.js +75 -0
  3. package/.test-dist/src/background.js +203 -0
  4. package/.test-dist/src/bridgeSelection.js +20 -0
  5. package/.test-dist/src/constraints.js +237 -0
  6. package/.test-dist/src/coordinate.js +74 -0
  7. package/.test-dist/src/dieline.js +723 -0
  8. package/.test-dist/src/edgeScale.js +12 -0
  9. package/.test-dist/src/feature.js +752 -0
  10. package/.test-dist/src/featureComplete.js +32 -0
  11. package/.test-dist/src/film.js +167 -0
  12. package/.test-dist/src/geometry.js +506 -0
  13. package/.test-dist/src/image.js +1234 -0
  14. package/.test-dist/src/index.js +35 -0
  15. package/.test-dist/src/maskOps.js +270 -0
  16. package/.test-dist/src/mirror.js +104 -0
  17. package/.test-dist/src/renderSpec.js +2 -0
  18. package/.test-dist/src/ruler.js +343 -0
  19. package/.test-dist/src/sceneLayout.js +99 -0
  20. package/.test-dist/src/sceneLayoutModel.js +196 -0
  21. package/.test-dist/src/sceneView.js +40 -0
  22. package/.test-dist/src/sceneVisibility.js +42 -0
  23. package/.test-dist/src/size.js +332 -0
  24. package/.test-dist/src/tracer.js +544 -0
  25. package/.test-dist/src/units.js +30 -0
  26. package/.test-dist/src/white-ink.js +829 -0
  27. package/.test-dist/src/wrappedOffsets.js +33 -0
  28. package/.test-dist/tests/run.js +94 -0
  29. package/CHANGELOG.md +11 -0
  30. package/dist/index.d.mts +339 -36
  31. package/dist/index.d.ts +339 -36
  32. package/dist/index.js +3572 -850
  33. package/dist/index.mjs +3565 -852
  34. package/package.json +2 -2
  35. package/src/CanvasService.ts +300 -96
  36. package/src/ViewportSystem.ts +92 -92
  37. package/src/background.ts +230 -230
  38. package/src/bridgeSelection.ts +17 -0
  39. package/src/coordinate.ts +106 -106
  40. package/src/dieline.ts +897 -973
  41. package/src/edgeScale.ts +19 -0
  42. package/src/feature.ts +83 -30
  43. package/src/film.ts +194 -194
  44. package/src/geometry.ts +242 -84
  45. package/src/image.ts +1582 -512
  46. package/src/index.ts +14 -10
  47. package/src/maskOps.ts +326 -0
  48. package/src/mirror.ts +128 -128
  49. package/src/renderSpec.ts +18 -0
  50. package/src/ruler.ts +449 -508
  51. package/src/sceneLayout.ts +121 -0
  52. package/src/sceneLayoutModel.ts +335 -0
  53. package/src/sceneVisibility.ts +49 -0
  54. package/src/size.ts +379 -0
  55. package/src/tracer.ts +719 -570
  56. package/src/units.ts +27 -27
  57. package/src/white-ink.ts +1018 -373
  58. package/src/wrappedOffsets.ts +33 -0
  59. package/tests/run.ts +118 -0
  60. package/tsconfig.test.json +15 -15
package/src/background.ts CHANGED
@@ -1,230 +1,230 @@
1
- import {
2
- Extension,
3
- ExtensionContext,
4
- ContributionPointIds,
5
- CommandContribution,
6
- ConfigurationContribution,
7
- } from "@pooder/core";
8
- import { Rect, FabricImage as Image } from "fabric";
9
- import CanvasService from "./CanvasService";
10
-
11
- export class BackgroundTool implements Extension {
12
- id = "pooder.kit.background";
13
- public metadata = {
14
- name: "BackgroundTool",
15
- };
16
-
17
- private color: string = "";
18
- private url: string = "";
19
-
20
- private canvasService?: CanvasService;
21
-
22
- constructor(
23
- options?: Partial<{
24
- color: string;
25
- url: string;
26
- }>,
27
- ) {
28
- if (options) {
29
- Object.assign(this, options);
30
- }
31
- }
32
-
33
- activate(context: ExtensionContext) {
34
- this.canvasService = context.services.get<CanvasService>("CanvasService");
35
- if (!this.canvasService) {
36
- console.warn("CanvasService not found for BackgroundTool");
37
- return;
38
- }
39
-
40
- const configService = context.services.get<any>("ConfigurationService");
41
- if (configService) {
42
- // Load initial config
43
- this.color = configService.get("background.color", this.color);
44
- this.url = configService.get("background.url", this.url);
45
-
46
- // Listen for changes
47
- configService.onAnyChange((e: { key: string; value: any }) => {
48
- if (e.key.startsWith("background.")) {
49
- const prop = e.key.split(".")[1];
50
- console.log(
51
- `[BackgroundTool] Config change detected: ${e.key} -> ${e.value}, prop: ${prop}`,
52
- );
53
- if (prop && prop in this) {
54
- console.log(
55
- `[BackgroundTool] Updating option ${prop} to ${e.value}`,
56
- );
57
- (this as any)[prop] = e.value;
58
- this.updateBackground();
59
- } else {
60
- console.warn(
61
- `[BackgroundTool] Property ${prop} not found in options`,
62
- );
63
- }
64
- }
65
- });
66
- }
67
-
68
- this.initLayer();
69
- this.updateBackground();
70
- }
71
-
72
- deactivate(context: ExtensionContext) {
73
- if (this.canvasService) {
74
- const layer = this.canvasService.getLayer("background");
75
- if (layer) {
76
- this.canvasService.canvas.remove(layer);
77
- }
78
- this.canvasService = undefined;
79
- }
80
- }
81
-
82
- contribute() {
83
- return {
84
- [ContributionPointIds.CONFIGURATIONS]: [
85
- {
86
- id: "background.color",
87
- type: "color",
88
- label: "Background Color",
89
- default: "",
90
- },
91
- {
92
- id: "background.url",
93
- type: "string",
94
- label: "Image URL",
95
- default: "",
96
- },
97
- ] as ConfigurationContribution[],
98
- [ContributionPointIds.COMMANDS]: [
99
- {
100
- command: "reset",
101
- title: "Reset Background",
102
- handler: () => {
103
- this.updateBackground();
104
- return true;
105
- },
106
- },
107
- {
108
- command: "clear",
109
- title: "Clear Background",
110
- handler: () => {
111
- this.color = "transparent";
112
- this.url = "";
113
- this.updateBackground();
114
- return true;
115
- },
116
- },
117
- {
118
- command: "setBackgroundColor",
119
- title: "Set Background Color",
120
- handler: (color: string) => {
121
- if (this.color === color) return true;
122
- this.color = color;
123
- this.updateBackground();
124
- return true;
125
- },
126
- },
127
- {
128
- command: "setBackgroundImage",
129
- title: "Set Background Image",
130
- handler: (url: string) => {
131
- if (this.url === url) return true;
132
- this.url = url;
133
- this.updateBackground();
134
- return true;
135
- },
136
- },
137
- ] as CommandContribution[],
138
- };
139
- }
140
-
141
- private initLayer() {
142
- if (!this.canvasService) return;
143
- let backgroundLayer = this.canvasService.getLayer("background");
144
- if (!backgroundLayer) {
145
- backgroundLayer = this.canvasService.createLayer("background", {
146
- width: this.canvasService.canvas.width,
147
- height: this.canvasService.canvas.height,
148
- selectable: false,
149
- evented: false,
150
- });
151
- this.canvasService.canvas.sendObjectToBack(backgroundLayer);
152
- }
153
- }
154
-
155
- private async updateBackground() {
156
- if (!this.canvasService) return;
157
- const layer = this.canvasService.getLayer("background");
158
- if (!layer) {
159
- console.warn("[BackgroundTool] Background layer not found");
160
- return;
161
- }
162
-
163
- const { color, url } = this;
164
-
165
- const width = this.canvasService.canvas.width || 800;
166
- const height = this.canvasService.canvas.height || 600;
167
-
168
- let rect = this.canvasService.getObject(
169
- "background-color-rect",
170
- "background",
171
- ) as Rect;
172
- if (rect) {
173
- rect.set({
174
- fill: color,
175
- });
176
- } else {
177
- rect = new Rect({
178
- width,
179
- height,
180
- fill: color,
181
- selectable: false,
182
- evented: false,
183
- data: {
184
- id: "background-color-rect",
185
- },
186
- });
187
- layer.add(rect);
188
- layer.sendObjectToBack(rect);
189
- }
190
-
191
- let img = this.canvasService.getObject(
192
- "background-image",
193
- "background",
194
- ) as Image;
195
- try {
196
- if (img) {
197
- if (img.getSrc() !== url) {
198
- if (url) {
199
- await img.setSrc(url);
200
- } else {
201
- layer.remove(img);
202
- }
203
- }
204
- } else {
205
- if (url) {
206
- img = await Image.fromURL(url, { crossOrigin: "anonymous" });
207
- img.set({
208
- originX: "left",
209
- originY: "top",
210
- left: 0,
211
- top: 0,
212
- selectable: false,
213
- evented: false,
214
- data: {
215
- id: "background-image",
216
- },
217
- });
218
- img.scaleToWidth(width);
219
- if (img.getScaledHeight() < height) img.scaleToHeight(height);
220
- layer.add(img);
221
- }
222
- }
223
- this.canvasService.requestRenderAll();
224
- } catch (e) {
225
- console.error("[BackgroundTool] Failed to load image", e);
226
- }
227
- layer.dirty = true;
228
- this.canvasService.requestRenderAll();
229
- }
230
- }
1
+ import {
2
+ Extension,
3
+ ExtensionContext,
4
+ ContributionPointIds,
5
+ CommandContribution,
6
+ ConfigurationContribution,
7
+ } from "@pooder/core";
8
+ import { Rect, FabricImage as Image } from "fabric";
9
+ import CanvasService from "./CanvasService";
10
+
11
+ export class BackgroundTool implements Extension {
12
+ id = "pooder.kit.background";
13
+ public metadata = {
14
+ name: "BackgroundTool",
15
+ };
16
+
17
+ private color: string = "";
18
+ private url: string = "";
19
+
20
+ private canvasService?: CanvasService;
21
+
22
+ constructor(
23
+ options?: Partial<{
24
+ color: string;
25
+ url: string;
26
+ }>,
27
+ ) {
28
+ if (options) {
29
+ Object.assign(this, options);
30
+ }
31
+ }
32
+
33
+ activate(context: ExtensionContext) {
34
+ this.canvasService = context.services.get<CanvasService>("CanvasService");
35
+ if (!this.canvasService) {
36
+ console.warn("CanvasService not found for BackgroundTool");
37
+ return;
38
+ }
39
+
40
+ const configService = context.services.get<any>("ConfigurationService");
41
+ if (configService) {
42
+ // Load initial config
43
+ this.color = configService.get("background.color", this.color);
44
+ this.url = configService.get("background.url", this.url);
45
+
46
+ // Listen for changes
47
+ configService.onAnyChange((e: { key: string; value: any }) => {
48
+ if (e.key.startsWith("background.")) {
49
+ const prop = e.key.split(".")[1];
50
+ console.log(
51
+ `[BackgroundTool] Config change detected: ${e.key} -> ${e.value}, prop: ${prop}`,
52
+ );
53
+ if (prop && prop in this) {
54
+ console.log(
55
+ `[BackgroundTool] Updating option ${prop} to ${e.value}`,
56
+ );
57
+ (this as any)[prop] = e.value;
58
+ this.updateBackground();
59
+ } else {
60
+ console.warn(
61
+ `[BackgroundTool] Property ${prop} not found in options`,
62
+ );
63
+ }
64
+ }
65
+ });
66
+ }
67
+
68
+ this.initLayer();
69
+ this.updateBackground();
70
+ }
71
+
72
+ deactivate(context: ExtensionContext) {
73
+ if (this.canvasService) {
74
+ const layer = this.canvasService.getLayer("background");
75
+ if (layer) {
76
+ this.canvasService.canvas.remove(layer);
77
+ }
78
+ this.canvasService = undefined;
79
+ }
80
+ }
81
+
82
+ contribute() {
83
+ return {
84
+ [ContributionPointIds.CONFIGURATIONS]: [
85
+ {
86
+ id: "background.color",
87
+ type: "color",
88
+ label: "Background Color",
89
+ default: "",
90
+ },
91
+ {
92
+ id: "background.url",
93
+ type: "string",
94
+ label: "Image URL",
95
+ default: "",
96
+ },
97
+ ] as ConfigurationContribution[],
98
+ [ContributionPointIds.COMMANDS]: [
99
+ {
100
+ command: "reset",
101
+ title: "Reset Background",
102
+ handler: () => {
103
+ this.updateBackground();
104
+ return true;
105
+ },
106
+ },
107
+ {
108
+ command: "clear",
109
+ title: "Clear Background",
110
+ handler: () => {
111
+ this.color = "transparent";
112
+ this.url = "";
113
+ this.updateBackground();
114
+ return true;
115
+ },
116
+ },
117
+ {
118
+ command: "setBackgroundColor",
119
+ title: "Set Background Color",
120
+ handler: (color: string) => {
121
+ if (this.color === color) return true;
122
+ this.color = color;
123
+ this.updateBackground();
124
+ return true;
125
+ },
126
+ },
127
+ {
128
+ command: "setBackgroundImage",
129
+ title: "Set Background Image",
130
+ handler: (url: string) => {
131
+ if (this.url === url) return true;
132
+ this.url = url;
133
+ this.updateBackground();
134
+ return true;
135
+ },
136
+ },
137
+ ] as CommandContribution[],
138
+ };
139
+ }
140
+
141
+ private initLayer() {
142
+ if (!this.canvasService) return;
143
+ let backgroundLayer = this.canvasService.getLayer("background");
144
+ if (!backgroundLayer) {
145
+ backgroundLayer = this.canvasService.createLayer("background", {
146
+ width: this.canvasService.canvas.width,
147
+ height: this.canvasService.canvas.height,
148
+ selectable: false,
149
+ evented: false,
150
+ });
151
+ this.canvasService.canvas.sendObjectToBack(backgroundLayer);
152
+ }
153
+ }
154
+
155
+ private async updateBackground() {
156
+ if (!this.canvasService) return;
157
+ const layer = this.canvasService.getLayer("background");
158
+ if (!layer) {
159
+ console.warn("[BackgroundTool] Background layer not found");
160
+ return;
161
+ }
162
+
163
+ const { color, url } = this;
164
+
165
+ const width = this.canvasService.canvas.width || 800;
166
+ const height = this.canvasService.canvas.height || 600;
167
+
168
+ let rect = this.canvasService.getObject(
169
+ "background-color-rect",
170
+ "background",
171
+ ) as Rect;
172
+ if (rect) {
173
+ rect.set({
174
+ fill: color,
175
+ });
176
+ } else {
177
+ rect = new Rect({
178
+ width,
179
+ height,
180
+ fill: color,
181
+ selectable: false,
182
+ evented: false,
183
+ data: {
184
+ id: "background-color-rect",
185
+ },
186
+ });
187
+ layer.add(rect);
188
+ layer.sendObjectToBack(rect);
189
+ }
190
+
191
+ let img = this.canvasService.getObject(
192
+ "background-image",
193
+ "background",
194
+ ) as Image;
195
+ try {
196
+ if (img) {
197
+ if (img.getSrc() !== url) {
198
+ if (url) {
199
+ await img.setSrc(url);
200
+ } else {
201
+ layer.remove(img);
202
+ }
203
+ }
204
+ } else {
205
+ if (url) {
206
+ img = await Image.fromURL(url, { crossOrigin: "anonymous" });
207
+ img.set({
208
+ originX: "left",
209
+ originY: "top",
210
+ left: 0,
211
+ top: 0,
212
+ selectable: false,
213
+ evented: false,
214
+ data: {
215
+ id: "background-image",
216
+ },
217
+ });
218
+ img.scaleToWidth(width);
219
+ if (img.getScaledHeight() < height) img.scaleToHeight(height);
220
+ layer.add(img);
221
+ }
222
+ }
223
+ this.canvasService.requestRenderAll();
224
+ } catch (e) {
225
+ console.error("[BackgroundTool] Failed to load image", e);
226
+ }
227
+ layer.dirty = true;
228
+ this.canvasService.requestRenderAll();
229
+ }
230
+ }
@@ -0,0 +1,17 @@
1
+ export function pickExitIndex(
2
+ hits: Array<{ insideAbove: boolean; insideBelow: boolean }>,
3
+ ): number {
4
+ for (let i = 0; i < hits.length; i++) {
5
+ const h = hits[i];
6
+ if (h.insideBelow && !h.insideAbove) return i;
7
+ }
8
+ return -1;
9
+ }
10
+
11
+ export function scoreOutsideAbove(samples: Array<{ outsideAbove: boolean }>): number {
12
+ let score = 0;
13
+ for (const s of samples) {
14
+ if (s.outsideAbove) score++;
15
+ }
16
+ return score;
17
+ }