@fcannizzaro/streamdeck-react 0.1.10 → 0.1.12

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 (76) hide show
  1. package/LICENSE +190 -21
  2. package/README.md +2 -0
  3. package/dist/action.d.ts +2 -2
  4. package/dist/action.js +1 -2
  5. package/dist/adapter/index.d.ts +2 -0
  6. package/dist/adapter/physical-device.d.ts +2 -0
  7. package/dist/adapter/physical-device.js +153 -0
  8. package/dist/adapter/types.d.ts +127 -0
  9. package/dist/bundler-shared.d.ts +11 -0
  10. package/dist/bundler-shared.js +11 -0
  11. package/dist/context/event-bus.d.ts +1 -1
  12. package/dist/context/event-bus.js +1 -1
  13. package/dist/context/touchstrip-context.d.ts +2 -0
  14. package/dist/context/touchstrip-context.js +5 -0
  15. package/dist/devtools/bridge.d.ts +35 -7
  16. package/dist/devtools/bridge.js +152 -46
  17. package/dist/devtools/highlight.d.ts +5 -0
  18. package/dist/devtools/highlight.js +107 -57
  19. package/dist/devtools/index.js +6 -0
  20. package/dist/devtools/observers/lifecycle.d.ts +4 -4
  21. package/dist/devtools/server.d.ts +6 -1
  22. package/dist/devtools/server.js +6 -1
  23. package/dist/devtools/types.d.ts +50 -6
  24. package/dist/font-inline.d.ts +5 -1
  25. package/dist/font-inline.js +8 -3
  26. package/dist/hooks/animation.d.ts +154 -0
  27. package/dist/hooks/animation.js +381 -0
  28. package/dist/hooks/events.js +2 -6
  29. package/dist/hooks/sdk.js +11 -11
  30. package/dist/hooks/touchstrip.d.ts +6 -0
  31. package/dist/hooks/touchstrip.js +37 -0
  32. package/dist/hooks/utility.js +3 -2
  33. package/dist/index.d.ts +9 -2
  34. package/dist/index.js +4 -2
  35. package/dist/manifest-codegen.d.ts +38 -0
  36. package/dist/manifest-codegen.js +110 -0
  37. package/dist/plugin.js +86 -106
  38. package/dist/reconciler/host-config.js +19 -1
  39. package/dist/reconciler/vnode.d.ts +26 -2
  40. package/dist/reconciler/vnode.js +40 -10
  41. package/dist/render/buffer-pool.d.ts +19 -0
  42. package/dist/render/buffer-pool.js +51 -0
  43. package/dist/render/cache.d.ts +29 -0
  44. package/dist/render/cache.js +137 -5
  45. package/dist/render/image-cache.d.ts +54 -0
  46. package/dist/render/image-cache.js +144 -0
  47. package/dist/render/metrics.d.ts +57 -0
  48. package/dist/render/metrics.js +98 -0
  49. package/dist/render/pipeline.d.ts +36 -1
  50. package/dist/render/pipeline.js +304 -34
  51. package/dist/render/png.d.ts +1 -1
  52. package/dist/render/png.js +26 -11
  53. package/dist/render/render-pool.d.ts +24 -0
  54. package/dist/render/render-pool.js +130 -0
  55. package/dist/render/svg.d.ts +7 -0
  56. package/dist/render/svg.js +139 -0
  57. package/dist/render/worker.d.ts +1 -0
  58. package/dist/rollup.d.ts +23 -9
  59. package/dist/rollup.js +24 -9
  60. package/dist/roots/registry.d.ts +9 -11
  61. package/dist/roots/registry.js +39 -42
  62. package/dist/roots/root.d.ts +9 -6
  63. package/dist/roots/root.js +52 -29
  64. package/dist/roots/settings-equality.d.ts +5 -0
  65. package/dist/roots/settings-equality.js +24 -0
  66. package/dist/roots/{touchbar-root.d.ts → touchstrip-root.d.ts} +30 -8
  67. package/dist/roots/touchstrip-root.js +263 -0
  68. package/dist/types.d.ts +73 -23
  69. package/dist/vite.d.ts +22 -8
  70. package/dist/vite.js +24 -8
  71. package/package.json +7 -4
  72. package/dist/context/touchbar-context.d.ts +0 -2
  73. package/dist/context/touchbar-context.js +0 -5
  74. package/dist/hooks/touchbar.d.ts +0 -6
  75. package/dist/hooks/touchbar.js +0 -37
  76. package/dist/roots/touchbar-root.js +0 -175
@@ -1,62 +1,329 @@
1
- import { vnodeToElement } from "../reconciler/vnode.js";
2
- import { fnv1a } from "./cache.js";
1
+ import { isContainerDirty } from "../reconciler/vnode.js";
2
+ import { computeCacheKey, computeTreeHash, fnv1a } from "./cache.js";
3
+ import { getBufferPool } from "./buffer-pool.js";
3
4
  import { encodePng } from "./png.js";
4
- import { createElement } from "react";
5
- import { fromJsx } from "@takumi-rs/helpers/jsx";
5
+ import { getImageCache, getTouchStripCache } from "./image-cache.js";
6
+ import { metrics } from "./metrics.js";
7
+ import { serializeSvgTree } from "./svg.js";
6
8
  //#region src/render/pipeline.ts
9
+ var ROOT_STYLE = {
10
+ display: "flex",
11
+ width: "100%",
12
+ height: "100%"
13
+ };
14
+ /** Warn threshold for consecutive identical renders in debug mode. */
15
+ var DUP_RENDER_WARN_THRESHOLD = 3;
16
+ /** Maximum recommended VNode tree depth. Warn in debug mode when exceeded. */
17
+ var MAX_DEPTH_WARN = 25;
18
+ /** Whether we've already warned about tree depth (avoid log spam). */
19
+ var depthWarned = false;
20
+ var SKIP_PROPS = new Set([
21
+ "children",
22
+ "className",
23
+ "src",
24
+ "tw"
25
+ ]);
26
+ function copyPropsToNode(target, props) {
27
+ for (const key of Object.keys(props)) if (!SKIP_PROPS.has(key)) target[key] = props[key];
28
+ }
29
+ function vnodeToTakumiNode(node, depth = 0) {
30
+ if (!depthWarned && depth > MAX_DEPTH_WARN) {
31
+ depthWarned = true;
32
+ console.warn(`[@fcannizzaro/streamdeck-react] VNode tree depth ${depth} exceeds recommended limit ${MAX_DEPTH_WARN}. Deep nesting increases render cost.`);
33
+ }
34
+ if (node.type === "#text") return {
35
+ type: "text",
36
+ text: node.text ?? ""
37
+ };
38
+ const props = node.props;
39
+ const rawTw = typeof props.tw === "string" ? props.tw : void 0;
40
+ const className = props.className;
41
+ let tw = rawTw;
42
+ if (typeof className === "string" && className.length > 0) tw = tw ? tw + " " + className : className;
43
+ if (node.type === "img" && typeof props.src === "string") {
44
+ const result = {
45
+ type: "image",
46
+ src: props.src
47
+ };
48
+ if (tw) result.tw = tw;
49
+ copyPropsToNode(result, props);
50
+ return result;
51
+ }
52
+ if (node.type === "svg") {
53
+ const result = {
54
+ type: "image",
55
+ src: serializeSvgTree(node),
56
+ tagName: "svg"
57
+ };
58
+ const width = typeof props.width === "number" ? props.width : void 0;
59
+ const height = typeof props.height === "number" ? props.height : void 0;
60
+ if (width != null) result.width = width;
61
+ if (height != null) result.height = height;
62
+ if (tw) result.tw = tw;
63
+ if (props.style) result.style = props.style;
64
+ return result;
65
+ }
66
+ const result = { type: "container" };
67
+ if (tw) result.tw = tw;
68
+ copyPropsToNode(result, props);
69
+ if (node.children.length > 0) result.children = node.children.map((child) => vnodeToTakumiNode(child, depth + 1));
70
+ return result;
71
+ }
72
+ /** Build the root Takumi container wrapping the VNode children. */
73
+ function buildTakumiRoot(container) {
74
+ return {
75
+ type: "container",
76
+ style: ROOT_STYLE,
77
+ children: container.children.map(vnodeToTakumiNode)
78
+ };
79
+ }
80
+ function measureTree(nodes) {
81
+ let maxDepth = 0;
82
+ let count = 0;
83
+ function walk(children, depth) {
84
+ for (const child of children) {
85
+ count++;
86
+ if (depth > maxDepth) maxDepth = depth;
87
+ walk(child.children, depth + 1);
88
+ }
89
+ }
90
+ walk(nodes, 1);
91
+ return {
92
+ depth: maxDepth,
93
+ count
94
+ };
95
+ }
7
96
  function bufferToDataUri(buffer, format) {
8
97
  return `data:image/${format};base64,${(Buffer.isBuffer(buffer) ? buffer : Buffer.from(buffer)).toString("base64")}`;
9
98
  }
99
+ function emitProfile(config, times, opts) {
100
+ const stats = measureTree(opts.container.children);
101
+ const cache = config.imageCacheMaxBytes > 0 ? getImageCache(config.imageCacheMaxBytes) : null;
102
+ config.onProfile({
103
+ vnodeConversionMs: times.t1 - times.t0,
104
+ takumiRenderMs: times.t2 - times.t1,
105
+ hashMs: times.t3 - times.t2,
106
+ base64Ms: 0,
107
+ totalMs: times.t3 - times.t0,
108
+ skipped: opts.skipped,
109
+ cacheHit: opts.cacheHit,
110
+ treeDepth: stats.depth,
111
+ nodeCount: stats.count,
112
+ cacheStats: cache?.stats ?? null
113
+ });
114
+ }
10
115
  async function renderToDataUri(container, width, height, config) {
11
116
  if (container.children.length === 0) return null;
12
- const { node, stylesheets } = await fromJsx(createElement("div", { style: {
13
- display: "flex",
14
- width: "100%",
15
- height: "100%"
16
- } }, ...container.children.map(vnodeToElement)));
17
- const buffer = await config.renderer.render(node, {
18
- width,
19
- height,
20
- format: config.imageFormat,
21
- stylesheets,
22
- devicePixelRatio: config.devicePixelRatio
23
- });
117
+ metrics.recordFlush();
118
+ if (config.caching && !isContainerDirty(container)) {
119
+ metrics.recordDirtySkip();
120
+ return null;
121
+ }
122
+ const profiling = config.onProfile != null;
123
+ const t0 = profiling ? performance.now() : 0;
124
+ let t1 = t0;
125
+ let t2 = t0;
126
+ let treeHash;
127
+ let cacheKey;
128
+ if (config.caching && config.imageCacheMaxBytes > 0) {
129
+ treeHash = computeTreeHash(container);
130
+ cacheKey = computeCacheKey(treeHash, width, height, config.devicePixelRatio, config.imageFormat);
131
+ const cached = getImageCache(config.imageCacheMaxBytes).get(cacheKey);
132
+ if (cached !== void 0) {
133
+ metrics.recordCacheHit();
134
+ if (profiling) {
135
+ const tNow = performance.now();
136
+ emitProfile(config, {
137
+ t0,
138
+ t1: tNow,
139
+ t2: tNow,
140
+ t3: tNow
141
+ }, {
142
+ skipped: false,
143
+ cacheHit: true,
144
+ container
145
+ });
146
+ }
147
+ container._dupCount = 0;
148
+ config.onRender?.(container, cached);
149
+ return cached;
150
+ }
151
+ }
152
+ let buffer;
153
+ if (config.renderPool?.isAvailable) {
154
+ buffer = await config.renderPool.render(container.children, width, height, config.imageFormat, config.devicePixelRatio);
155
+ t2 = profiling ? performance.now() : 0;
156
+ t1 = t0;
157
+ } else {
158
+ const rootNode = buildTakumiRoot(container);
159
+ t1 = profiling ? performance.now() : 0;
160
+ buffer = await config.renderer.render(rootNode, {
161
+ width,
162
+ height,
163
+ format: config.imageFormat,
164
+ devicePixelRatio: config.devicePixelRatio
165
+ });
166
+ t2 = profiling ? performance.now() : 0;
167
+ }
24
168
  if (config.caching) {
25
169
  const hash = fnv1a(buffer);
26
- if (hash === container.lastSvgHash) return null;
170
+ if (hash === container.lastSvgHash) {
171
+ metrics.recordHashDedup();
172
+ if (config.debug) {
173
+ container._dupCount++;
174
+ if (container._dupCount > DUP_RENDER_WARN_THRESHOLD) console.warn(`[@fcannizzaro/streamdeck-react] ${container._dupCount} consecutive identical renders — component likely re-rendering without visual change`);
175
+ }
176
+ if (profiling) {
177
+ const tEnd = performance.now();
178
+ emitProfile(config, {
179
+ t0,
180
+ t1,
181
+ t2,
182
+ t3: tEnd
183
+ }, {
184
+ skipped: true,
185
+ cacheHit: false,
186
+ container
187
+ });
188
+ }
189
+ return null;
190
+ }
27
191
  container.lastSvgHash = hash;
192
+ container._dupCount = 0;
28
193
  }
29
194
  const dataUri = bufferToDataUri(buffer, config.imageFormat);
195
+ if (config.caching && config.imageCacheMaxBytes > 0) {
196
+ if (treeHash === void 0 || cacheKey === void 0) {
197
+ treeHash = computeTreeHash(container);
198
+ cacheKey = computeCacheKey(treeHash, width, height, config.devicePixelRatio, config.imageFormat);
199
+ }
200
+ getImageCache(config.imageCacheMaxBytes).set(cacheKey, dataUri, dataUri.length * 2 + 64);
201
+ }
202
+ metrics.recordRender(t2 - t0);
203
+ if (profiling) {
204
+ const tEnd = performance.now();
205
+ emitProfile(config, {
206
+ t0,
207
+ t1,
208
+ t2,
209
+ t3: tEnd
210
+ }, {
211
+ skipped: false,
212
+ cacheHit: false,
213
+ container
214
+ });
215
+ }
30
216
  config.onRender?.(container, dataUri);
31
217
  return dataUri;
32
218
  }
33
219
  async function renderToRaw(container, width, height, config) {
34
220
  if (container.children.length === 0) return null;
35
- const { node, stylesheets } = await fromJsx(createElement("div", { style: {
36
- display: "flex",
37
- width: "100%",
38
- height: "100%"
39
- } }, ...container.children.map(vnodeToElement)));
40
- const buffer = await config.renderer.render(node, {
41
- width,
42
- height,
43
- format: "raw",
44
- stylesheets,
45
- devicePixelRatio: config.devicePixelRatio
46
- });
221
+ metrics.recordFlush();
222
+ if (config.caching && !isContainerDirty(container)) {
223
+ metrics.recordDirtySkip();
224
+ return null;
225
+ }
226
+ const profiling = config.onProfile != null;
227
+ const t0 = profiling ? performance.now() : 0;
228
+ let t1 = t0;
229
+ let t2 = t0;
230
+ let treeHash;
231
+ let cacheKey;
232
+ if (config.caching && config.touchStripCacheMaxBytes > 0) {
233
+ treeHash = computeTreeHash(container);
234
+ cacheKey = computeCacheKey(treeHash, width, height, config.devicePixelRatio, "raw");
235
+ const cached = getTouchStripCache(config.touchStripCacheMaxBytes).get(cacheKey);
236
+ if (cached !== void 0) {
237
+ metrics.recordCacheHit();
238
+ if (profiling) {
239
+ const tNow = performance.now();
240
+ emitProfile(config, {
241
+ t0,
242
+ t1: tNow,
243
+ t2: tNow,
244
+ t3: tNow
245
+ }, {
246
+ skipped: false,
247
+ cacheHit: true,
248
+ container
249
+ });
250
+ }
251
+ return {
252
+ buffer: cached,
253
+ width,
254
+ height
255
+ };
256
+ }
257
+ }
258
+ let buffer;
259
+ if (config.renderPool?.isAvailable) {
260
+ buffer = await config.renderPool.render(container.children, width, height, "raw", config.devicePixelRatio);
261
+ t2 = profiling ? performance.now() : 0;
262
+ t1 = t0;
263
+ } else {
264
+ const rootNode = buildTakumiRoot(container);
265
+ t1 = profiling ? performance.now() : 0;
266
+ buffer = await config.renderer.render(rootNode, {
267
+ width,
268
+ height,
269
+ format: "raw",
270
+ devicePixelRatio: config.devicePixelRatio
271
+ });
272
+ t2 = profiling ? performance.now() : 0;
273
+ }
47
274
  if (config.caching) {
48
275
  const hash = fnv1a(buffer);
49
- if (hash === container.lastSvgHash) return null;
276
+ if (hash === container.lastSvgHash) {
277
+ metrics.recordHashDedup();
278
+ if (profiling) {
279
+ const tEnd = performance.now();
280
+ emitProfile(config, {
281
+ t0,
282
+ t1,
283
+ t2,
284
+ t3: tEnd
285
+ }, {
286
+ skipped: true,
287
+ cacheHit: false,
288
+ container
289
+ });
290
+ }
291
+ return null;
292
+ }
50
293
  container.lastSvgHash = hash;
51
294
  }
295
+ const buf = Buffer.isBuffer(buffer) ? buffer : Buffer.from(buffer);
296
+ if (config.caching && config.touchStripCacheMaxBytes > 0) {
297
+ if (treeHash === void 0 || cacheKey === void 0) {
298
+ treeHash = computeTreeHash(container);
299
+ cacheKey = computeCacheKey(treeHash, width, height, config.devicePixelRatio, "raw");
300
+ }
301
+ getTouchStripCache(config.touchStripCacheMaxBytes).set(cacheKey, buf, buf.byteLength + 64);
302
+ }
303
+ metrics.recordRender(t2 - t0);
304
+ if (profiling) {
305
+ const tEnd = performance.now();
306
+ emitProfile(config, {
307
+ t0,
308
+ t1,
309
+ t2,
310
+ t3: tEnd
311
+ }, {
312
+ skipped: false,
313
+ cacheHit: false,
314
+ container
315
+ });
316
+ }
52
317
  return {
53
- buffer: Buffer.isBuffer(buffer) ? buffer : Buffer.from(buffer),
318
+ buffer: buf,
54
319
  width,
55
320
  height
56
321
  };
57
322
  }
58
323
  function cropSlice(raw, fullWidth, column, segmentWidth, segmentHeight) {
59
- const slice = Buffer.alloc(segmentWidth * segmentHeight * 4);
324
+ const pool = getBufferPool();
325
+ const sliceSize = segmentWidth * segmentHeight * 4;
326
+ const slice = pool.acquire(sliceSize);
60
327
  const srcRowBytes = fullWidth * 4;
61
328
  const dstRowBytes = segmentWidth * 4;
62
329
  const xOffset = column * segmentWidth * 4;
@@ -68,7 +335,10 @@ function cropSlice(raw, fullWidth, column, segmentWidth, segmentHeight) {
68
335
  return slice;
69
336
  }
70
337
  function sliceToDataUri(raw, fullWidth, fullHeight, column, segmentWidth, segmentHeight) {
71
- return bufferToDataUri(encodePng(segmentWidth, segmentHeight, cropSlice(raw, fullWidth, column, segmentWidth, segmentHeight)), "png");
338
+ const cropped = cropSlice(raw, fullWidth, column, segmentWidth, segmentHeight);
339
+ const png = encodePng(segmentWidth, segmentHeight, cropped);
340
+ getBufferPool().release(cropped);
341
+ return bufferToDataUri(png, "png");
72
342
  }
73
343
  //#endregion
74
- export { bufferToDataUri, renderToDataUri, renderToRaw, sliceToDataUri };
344
+ export { bufferToDataUri, buildTakumiRoot, measureTree, renderToDataUri, renderToRaw, sliceToDataUri };
@@ -1,5 +1,5 @@
1
1
  /**
2
- * Encode raw RGBA pixels into a PNG buffer.
2
+ * Encode raw RGBA pixels into a PNG buffer (synchronous).
3
3
  *
4
4
  * @param width Image width in pixels.
5
5
  * @param height Image height in pixels.
@@ -1,3 +1,4 @@
1
+ import { getBufferPool } from "./buffer-pool.js";
1
2
  import { deflateSync } from "node:zlib";
2
3
  //#region src/render/png.ts
3
4
  var crcTable = new Uint32Array(256);
@@ -35,14 +36,7 @@ var PNG_SIGNATURE = Buffer.from([
35
36
  26,
36
37
  10
37
38
  ]);
38
- /**
39
- * Encode raw RGBA pixels into a PNG buffer.
40
- *
41
- * @param width Image width in pixels.
42
- * @param height Image height in pixels.
43
- * @param rgba Raw RGBA pixel data (width × height × 4 bytes, row-major).
44
- */
45
- function encodePng(width, height, rgba) {
39
+ function buildIhdr(width, height) {
46
40
  const ihdr = Buffer.alloc(13);
47
41
  ihdr.writeUInt32BE(width, 0);
48
42
  ihdr.writeUInt32BE(height, 4);
@@ -51,15 +45,22 @@ function encodePng(width, height, rgba) {
51
45
  ihdr[10] = 0;
52
46
  ihdr[11] = 0;
53
47
  ihdr[12] = 0;
48
+ return ihdr;
49
+ }
50
+ function buildFilteredScanlines(width, height, rgba) {
54
51
  const rowBytes = width * 4;
55
- const filtered = Buffer.alloc(height * (1 + rowBytes));
52
+ const filteredSize = height * (1 + rowBytes);
53
+ const filtered = getBufferPool().acquire(filteredSize);
56
54
  for (let y = 0; y < height; y++) {
57
55
  const dstOff = y * (1 + rowBytes);
58
56
  filtered[dstOff] = 0;
59
57
  const srcOff = y * rowBytes;
60
- for (let i = 0; i < rowBytes; i++) filtered[dstOff + 1 + i] = rgba[srcOff + i];
58
+ if (Buffer.isBuffer(rgba)) rgba.copy(filtered, dstOff + 1, srcOff, srcOff + rowBytes);
59
+ else filtered.set(rgba.subarray(srcOff, srcOff + rowBytes), dstOff + 1);
61
60
  }
62
- const compressed = deflateSync(filtered);
61
+ return filtered;
62
+ }
63
+ function assemblePng(ihdr, compressed) {
63
64
  return Buffer.concat([
64
65
  PNG_SIGNATURE,
65
66
  pngChunk("IHDR", ihdr),
@@ -67,5 +68,19 @@ function encodePng(width, height, rgba) {
67
68
  pngChunk("IEND", Buffer.alloc(0))
68
69
  ]);
69
70
  }
71
+ /**
72
+ * Encode raw RGBA pixels into a PNG buffer (synchronous).
73
+ *
74
+ * @param width Image width in pixels.
75
+ * @param height Image height in pixels.
76
+ * @param rgba Raw RGBA pixel data (width × height × 4 bytes, row-major).
77
+ */
78
+ function encodePng(width, height, rgba) {
79
+ const ihdr = buildIhdr(width, height);
80
+ const filtered = buildFilteredScanlines(width, height, rgba);
81
+ const compressed = deflateSync(filtered);
82
+ getBufferPool().release(filtered);
83
+ return assemblePng(ihdr, compressed);
84
+ }
70
85
  //#endregion
71
86
  export { encodePng };
@@ -0,0 +1,24 @@
1
+ import { FontConfig } from '../types';
2
+ import { VNode } from '../reconciler/vnode';
3
+ export declare class RenderPool {
4
+ private worker;
5
+ private ready;
6
+ private failed;
7
+ private nextId;
8
+ private pending;
9
+ private initPromise;
10
+ private fonts;
11
+ constructor(fonts: FontConfig[]);
12
+ /** Start the worker. Call once during plugin initialization. */
13
+ initialize(): Promise<boolean>;
14
+ private doInitialize;
15
+ /** Whether the worker is available for offloaded rendering. */
16
+ get isAvailable(): boolean;
17
+ /**
18
+ * Render VNode children in the worker thread.
19
+ * Returns the raw raster buffer.
20
+ */
21
+ render(vnodes: VNode[], width: number, height: number, format: string, dpr: number): Promise<Buffer>;
22
+ private handleResponse;
23
+ private handleWorkerDeath;
24
+ }
@@ -0,0 +1,130 @@
1
+ import { Worker } from "node:worker_threads";
2
+ //#region src/render/render-pool.ts
3
+ /** Strip internal fields and function props from a VNode tree for worker transfer. */
4
+ function serializeVNode(node) {
5
+ const cleanProps = {};
6
+ for (const [key, value] of Object.entries(node.props)) if (typeof value !== "function" && typeof value !== "symbol") cleanProps[key] = value;
7
+ return {
8
+ type: node.type,
9
+ props: cleanProps,
10
+ children: node.children.map(serializeVNode),
11
+ text: node.text
12
+ };
13
+ }
14
+ var RenderPool = class {
15
+ worker = null;
16
+ ready = false;
17
+ failed = false;
18
+ nextId = 0;
19
+ pending = /* @__PURE__ */ new Map();
20
+ initPromise = null;
21
+ fonts;
22
+ constructor(fonts) {
23
+ this.fonts = fonts;
24
+ }
25
+ /** Start the worker. Call once during plugin initialization. */
26
+ async initialize() {
27
+ if (this.ready) return true;
28
+ if (this.failed) return false;
29
+ if (this.initPromise != null) {
30
+ await this.initPromise;
31
+ return this.ready;
32
+ }
33
+ this.initPromise = this.doInitialize();
34
+ await this.initPromise;
35
+ return this.ready;
36
+ }
37
+ async doInitialize() {
38
+ try {
39
+ this.worker = new Worker(new URL("data:video/mp2t;base64,// ── Render Worker ────────────────────────────────────────────────────
//
// Runs in a separate thread via Node.js worker_threads.  Handles the
// full render pipeline: serialized VNode data → Takumi nodes → raster.
//
// Uses the direct VNode → Takumi node bypass (same as main thread's
// vnodeToTakumiNode in pipeline.ts), skipping vnodeToElement() and
// fromJsx() entirely.
//
// This unblocks the main thread during the expensive Takumi
// rasterization step (~5–30ms per frame).
//
// Why code is duplicated (inlined):
//   Worker threads can't import from the main bundle — they load
//   the compiled worker.js file independently.  SVG serialization
//   and VNode→Takumi conversion must be self-contained here.
//   Both mirror the logic in svg.ts and pipeline.ts respectively.
//
// Zero-copy return:
//   The rendered buffer is transferred (not copied) back to the main
//   thread via postMessage's transfer list.  This avoids copying
//   potentially large raster buffers (e.g. 800×100×4 = 320KB for
//   TouchStrip) across the thread boundary.

import { parentPort, workerData } from "node:worker_threads";

// ── Types ───────────────────────────────────────────────────────────

interface SerializedVNode {
  type: string;
  props: Record<string, unknown>;
  children: SerializedVNode[];
  text?: string;
}

/** Matches @takumi-rs/helpers Node union. Plain object accepted by Renderer.render(). */
interface TakumiNode {
  type: string;
  [key: string]: unknown;
}

interface InitMessage {
  type: "init";
  fonts: Array<{
    name: string;
    data: ArrayBuffer | Buffer;
    weight: number;
    style: string;
  }>;
}

interface RenderMessage {
  type: "render";
  id: number;
  vnodes: SerializedVNode[];
  width: number;
  height: number;
  format: string;
  dpr: number;
}

interface ShutdownMessage {
  type: "shutdown";
}

type WorkerMessage = InitMessage | RenderMessage | ShutdownMessage;

// ── SVG Serialization (inlined for worker context) ──────────────────
// Mirrors the serializeSvgTree() from svg.ts. Inlined to avoid
// cross-module import issues in the worker thread.

const SVG_CAMEL_ATTRS: ReadonlySet<string> = new Set([
  "accentHeight",
  "alignmentBaseline",
  "arabicForm",
  "baselineShift",
  "capHeight",
  "clipPath",
  "clipPathUnits",
  "clipRule",
  "colorInterpolation",
  "colorInterpolationFilters",
  "colorProfile",
  "colorRendering",
  "enableBackground",
  "fillOpacity",
  "fillRule",
  "floodColor",
  "floodOpacity",
  "fontFamily",
  "fontSize",
  "fontSizeAdjust",
  "fontStretch",
  "fontStyle",
  "fontVariant",
  "fontWeight",
  "glyphName",
  "glyphOrientationHorizontal",
  "glyphOrientationVertical",
  "horizAdvX",
  "horizOriginX",
  "imageRendering",
  "letterSpacing",
  "lightingColor",
  "markerEnd",
  "markerMid",
  "markerStart",
  "overlinePosition",
  "overlineThickness",
  "paintOrder",
  "pointerEvents",
  "preserveAspectRatio",
  "shapeRendering",
  "stopColor",
  "stopOpacity",
  "strokeDasharray",
  "strokeDashoffset",
  "strokeLinecap",
  "strokeLinejoin",
  "strokeMiterlimit",
  "strokeOpacity",
  "strokeWidth",
  "textAnchor",
  "textDecoration",
  "textRendering",
  "transformOrigin",
  "underlinePosition",
  "underlineThickness",
  "unicodeBidi",
  "unicodeRange",
  "unitsPerEm",
  "vAlphabetic",
  "vHanging",
  "vIdeographic",
  "vMathematical",
  "vectorEffect",
  "vertAdvY",
  "vertOriginX",
  "vertOriginY",
  "wordSpacing",
  "writingMode",
]);

const SVG_SKIP_PROPS: ReadonlySet<string> = new Set([
  "children",
  "key",
  "ref",
  "__self",
  "__source",
]);

function camelToKebab(str: string): string {
  return str.replace(/[A-Z]/g, (ch) => `-${ch.toLowerCase()}`);
}

function escapeAttr(value: string): string {
  return value
    .replace(/&/g, "&amp;")
    .replace(/"/g, "&quot;")
    .replace(/</g, "&lt;")
    .replace(/>/g, "&gt;");
}

function serializeSvgStyle(style: Record<string, unknown>): string {
  const parts: string[] = [];
  for (const key of Object.keys(style)) {
    const value = style[key];
    if (value == null) continue;
    parts.push(`${camelToKebab(key)}:${String(value).trim()}`);
  }
  return parts.join(";");
}

function serializeSvgAttr(key: string, value: unknown): string | null {
  if (SVG_SKIP_PROPS.has(key) || value == null) return null;
  let attrName: string;
  if (key === "className") attrName = "class";
  else if (SVG_CAMEL_ATTRS.has(key)) attrName = camelToKebab(key);
  else attrName = key;
  if (key === "style" && typeof value === "object") {
    const css = serializeSvgStyle(value as Record<string, unknown>);
    if (!css) return null;
    return `${attrName}="${escapeAttr(css)}"`;
  }
  if (typeof value === "boolean") return `${attrName}="${String(value)}"`;
  return `${attrName}="${escapeAttr(String(value))}"`;
}

function serializeSvgVNode(node: SerializedVNode): string {
  if (node.type === "#text") return node.text ?? "";
  const attrs: string[] = [];
  for (const [key, value] of Object.entries(node.props)) {
    const attr = serializeSvgAttr(key, value);
    if (attr != null) attrs.push(attr);
  }
  const childMarkup = node.children.map(serializeSvgVNode).join("");
  const attrStr = attrs.length > 0 ? ` ${attrs.join(" ")}` : "";
  return `<${node.type}${attrStr}>${childMarkup}</${node.type}>`;
}

function serializeSvgTree(svgNode: SerializedVNode): string {
  if (!("xmlns" in svgNode.props)) {
    const augmented = {
      ...svgNode,
      props: { ...svgNode.props, xmlns: "http://www.w3.org/2000/svg" },
    };
    return serializeSvgVNode(augmented);
  }
  return serializeSvgVNode(svgNode);
}

// ── Direct VNode → Takumi Node Conversion ───────────────────────────
// Mirrors the main-thread vnodeToTakumiNode() from pipeline.ts.
// Inlined to avoid cross-module import issues in worker context.

function vnodeToTakumiNode(node: SerializedVNode): TakumiNode {
  // Text nodes → Takumi TextNode
  if (node.type === "#text") {
    return { type: "text", text: node.text ?? "" };
  }

  const { children: _children, className, src, ...restProps } = node.props;

  // Map className → tw (same logic as main thread)
  let tw: string | undefined = typeof restProps.tw === "string" ? restProps.tw : undefined;
  if (typeof className === "string" && className.length > 0) {
    tw = tw ? tw + " " + className : className;
  }

  // Image nodes → Takumi ImageNode
  if (node.type === "img" && typeof src === "string") {
    return {
      type: "image",
      src: src as string,
      ...(tw ? { tw } : {}),
      ...restProps,
    };
  }

  // SVG nodes → Takumi ImageNode (serialize subtree to SVG markup)
  if (node.type === "svg") {
    const svgMarkup = serializeSvgTree(node);
    const width = typeof node.props.width === "number" ? node.props.width : undefined;
    const height = typeof node.props.height === "number" ? node.props.height : undefined;
    return {
      type: "image",
      src: svgMarkup,
      ...(width != null ? { width } : {}),
      ...(height != null ? { height } : {}),
      ...(tw ? { tw } : {}),
      ...(node.props.style ? { style: node.props.style } : {}),
      tagName: "svg",
    };
  }

  // All other nodes → Takumi ContainerNode
  const takumiChildren =
    node.children.length > 0 ? node.children.map(vnodeToTakumiNode) : undefined;

  return {
    type: "container",
    ...(tw ? { tw } : {}),
    ...restProps,
    ...(takumiChildren ? { children: takumiChildren } : {}),
  };
}

// ── Root style constant ─────────────────────────────────────────────

const ROOT_STYLE = { display: "flex", width: "100%", height: "100%" } as const;

// ── Worker State ────────────────────────────────────────────────────

let renderer: import("@takumi-rs/core").Renderer | null = null;

// ── Message Handler ─────────────────────────────────────────────────

async function handleMessage(msg: WorkerMessage): Promise<void> {
  switch (msg.type) {
    case "init": {
      try {
        // Dynamic import — may fail if the native addon can't load in a worker
        const core = await import("@takumi-rs/core");

        renderer = new core.Renderer({
          fonts: msg.fonts.map((f) => ({
            name: f.name,
            data: f.data,
            weight: f.weight as 100 | 200 | 300 | 400 | 500 | 600 | 700 | 800 | 900,
            style: f.style as "normal" | "italic",
          })),
        });

        parentPort!.postMessage({ type: "ready" });
      } catch (err) {
        parentPort!.postMessage({
          type: "error",
          id: -1,
          error: `Worker init failed: ${err instanceof Error ? err.message : String(err)}`,
        });
      }
      break;
    }

    case "render": {
      if (renderer == null) {
        parentPort!.postMessage({
          type: "error",
          id: msg.id,
          error: "Worker not initialized",
        });
        return;
      }

      try {
        // 1. Convert serialized VNode data → Takumi nodes directly (bypass fromJsx)
        const children = msg.vnodes.map(vnodeToTakumiNode);
        const rootNode: TakumiNode = {
          type: "container",
          style: ROOT_STYLE,
          children,
        };

        // 2. Render to raster image
        const buffer = await renderer.render(rootNode, {
          width: msg.width,
          height: msg.height,
          format: msg.format as import("@takumi-rs/core").OutputFormat,
          devicePixelRatio: msg.dpr,
        });

        // Transfer the buffer (zero-copy) back to the main thread
        const ab =
          buffer instanceof ArrayBuffer
            ? buffer
            : buffer.buffer.slice(buffer.byteOffset, buffer.byteOffset + buffer.byteLength);

        parentPort!.postMessage(
          { type: "result", id: msg.id, buffer: ab },
          { transfer: [ab as ArrayBuffer] },
        );
      } catch (err) {
        parentPort!.postMessage({
          type: "error",
          id: msg.id,
          error: err instanceof Error ? err.message : String(err),
        });
      }
      break;
    }

    case "shutdown": {
      process.exit(0);
    }
  }
}

// ── Auto-init if fonts provided via workerData ──────────────────────

if (workerData?.fonts) {
  handleMessage({ type: "init", fonts: workerData.fonts });
}

parentPort!.on("message", (msg: WorkerMessage) => {
  handleMessage(msg);
});
", "" + import.meta.url), { workerData: { fonts: this.fonts.map((f) => ({
40
+ name: f.name,
41
+ data: f.data,
42
+ weight: f.weight,
43
+ style: f.style
44
+ })) } });
45
+ await new Promise((resolve, reject) => {
46
+ const timeout = setTimeout(() => {
47
+ reject(/* @__PURE__ */ new Error("Worker initialization timed out (5s)"));
48
+ }, 5e3);
49
+ const onMessage = (msg) => {
50
+ if (msg.type === "ready") {
51
+ clearTimeout(timeout);
52
+ this.worker.off("message", onMessage);
53
+ this.worker.off("error", onError);
54
+ resolve();
55
+ } else if (msg.type === "error" && msg.id === -1) {
56
+ clearTimeout(timeout);
57
+ this.worker.off("message", onMessage);
58
+ this.worker.off("error", onError);
59
+ reject(new Error(msg.error ?? "Unknown worker init error"));
60
+ }
61
+ };
62
+ const onError = (err) => {
63
+ clearTimeout(timeout);
64
+ reject(err);
65
+ };
66
+ this.worker.on("message", onMessage);
67
+ this.worker.on("error", onError);
68
+ });
69
+ this.worker.on("message", this.handleResponse.bind(this));
70
+ this.worker.on("exit", (code) => {
71
+ if (code !== 0) console.warn(`[@fcannizzaro/streamdeck-react] Render worker exited with code ${code}`);
72
+ this.handleWorkerDeath();
73
+ });
74
+ this.worker.on("error", (err) => {
75
+ console.error("[@fcannizzaro/streamdeck-react] Render worker error:", err);
76
+ this.handleWorkerDeath();
77
+ });
78
+ this.ready = true;
79
+ } catch (err) {
80
+ console.warn("[@fcannizzaro/streamdeck-react] Worker initialization failed, falling back to main-thread rendering:", err instanceof Error ? err.message : err);
81
+ this.failed = true;
82
+ this.worker?.terminate().catch(() => {});
83
+ this.worker = null;
84
+ }
85
+ }
86
+ /** Whether the worker is available for offloaded rendering. */
87
+ get isAvailable() {
88
+ return this.ready && this.worker != null;
89
+ }
90
+ /**
91
+ * Render VNode children in the worker thread.
92
+ * Returns the raw raster buffer.
93
+ */
94
+ async render(vnodes, width, height, format, dpr) {
95
+ if (!this.isAvailable) throw new Error("Worker not available");
96
+ const id = this.nextId++;
97
+ const serialized = vnodes.map(serializeVNode);
98
+ return new Promise((resolve, reject) => {
99
+ this.pending.set(id, {
100
+ resolve,
101
+ reject
102
+ });
103
+ this.worker.postMessage({
104
+ type: "render",
105
+ id,
106
+ vnodes: serialized,
107
+ width,
108
+ height,
109
+ format,
110
+ dpr
111
+ });
112
+ });
113
+ }
114
+ handleResponse(msg) {
115
+ if (msg.type === "ready") return;
116
+ const req = this.pending.get(msg.id);
117
+ if (req == null) return;
118
+ this.pending.delete(msg.id);
119
+ if (msg.type === "result" && msg.buffer != null) req.resolve(Buffer.from(msg.buffer));
120
+ else if (msg.type === "error") req.reject(new Error(msg.error ?? "Unknown worker render error"));
121
+ }
122
+ handleWorkerDeath() {
123
+ this.ready = false;
124
+ this.worker = null;
125
+ for (const [_, req] of this.pending) req.reject(/* @__PURE__ */ new Error("Worker died unexpectedly"));
126
+ this.pending.clear();
127
+ }
128
+ };
129
+ //#endregion
130
+ export { RenderPool };
@@ -0,0 +1,7 @@
1
+ import { VNode } from '../reconciler/vnode';
2
+ /**
3
+ * Serialize an `<svg>` VNode (and its entire subtree) to an SVG markup string.
4
+ * Auto-injects `xmlns="http://www.w3.org/2000/svg"` if not present.
5
+ * The returned string can be used as the `src` of a Takumi ImageNode.
6
+ */
7
+ export declare function serializeSvgTree(svgNode: VNode): string;