@fcannizzaro/streamdeck-react 0.1.11 → 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 (43) hide show
  1. package/dist/action.js +0 -1
  2. package/dist/adapter/index.d.ts +2 -0
  3. package/dist/adapter/physical-device.d.ts +2 -0
  4. package/dist/adapter/physical-device.js +153 -0
  5. package/dist/adapter/types.d.ts +127 -0
  6. package/dist/devtools/bridge.d.ts +2 -2
  7. package/dist/devtools/bridge.js +7 -8
  8. package/dist/devtools/highlight.d.ts +1 -2
  9. package/dist/devtools/highlight.js +4 -3
  10. package/dist/devtools/types.d.ts +5 -5
  11. package/dist/hooks/animation.d.ts +1 -1
  12. package/dist/hooks/animation.js +2 -2
  13. package/dist/hooks/events.js +1 -1
  14. package/dist/hooks/sdk.js +11 -11
  15. package/dist/hooks/utility.js +3 -2
  16. package/dist/index.d.ts +3 -1
  17. package/dist/index.js +2 -1
  18. package/dist/plugin.js +69 -100
  19. package/dist/reconciler/vnode.d.ts +0 -2
  20. package/dist/reconciler/vnode.js +0 -1
  21. package/dist/render/cache.d.ts +5 -17
  22. package/dist/render/cache.js +7 -29
  23. package/dist/render/image-cache.d.ts +8 -7
  24. package/dist/render/image-cache.js +33 -17
  25. package/dist/render/metrics.d.ts +9 -10
  26. package/dist/render/metrics.js +36 -39
  27. package/dist/render/pipeline.d.ts +4 -14
  28. package/dist/render/pipeline.js +47 -111
  29. package/dist/render/png.d.ts +0 -9
  30. package/dist/render/png.js +5 -8
  31. package/dist/render/render-pool.d.ts +0 -2
  32. package/dist/render/render-pool.js +1 -12
  33. package/dist/roots/registry.d.ts +5 -9
  34. package/dist/roots/registry.js +10 -27
  35. package/dist/roots/root.d.ts +7 -34
  36. package/dist/roots/root.js +23 -90
  37. package/dist/roots/touchstrip-root.d.ts +6 -32
  38. package/dist/roots/touchstrip-root.js +61 -181
  39. package/dist/types.d.ts +23 -19
  40. package/package.json +6 -4
  41. package/dist/node_modules/.bun/xxhash-wasm@1.1.0/node_modules/xxhash-wasm/esm/xxhash-wasm.js +0 -3157
  42. package/dist/roots/flush-coordinator.d.ts +0 -18
  43. package/dist/roots/flush-coordinator.js +0 -38
@@ -5,8 +5,8 @@ import { CacheStats } from './image-cache';
5
5
  import { RenderPool } from './render-pool';
6
6
  /** Per-render timing and diagnostic data exposed via `RenderConfig.onProfile`. */
7
7
  export interface RenderProfile {
8
- vnodeToElementMs: number;
9
- fromJsxMs: number;
8
+ /** Time to convert VNode tree to Takumi node tree (ms). */
9
+ vnodeConversionMs: number;
10
10
  takumiRenderMs: number;
11
11
  hashMs: number;
12
12
  base64Ms: number;
@@ -28,12 +28,10 @@ export interface RenderConfig {
28
28
  debug: boolean;
29
29
  /** Maximum image cache size in bytes. Set to 0 to disable. @default 16777216 (16 MB) */
30
30
  imageCacheMaxBytes: number;
31
- /** Maximum touchstrip cache size in bytes. Set to 0 to disable. @default 8388608 (8 MB) */
32
- touchstripCacheMaxBytes: number;
31
+ /** Maximum TouchStrip cache size in bytes. Set to 0 to disable. @default 8388608 (8 MB) */
32
+ touchStripCacheMaxBytes: number;
33
33
  /** Worker thread pool for offloading Takumi renders. null = main-thread rendering. */
34
34
  renderPool: RenderPool | null;
35
- /** Image format for touchstrip segment encoding. @default "webp" */
36
- touchstripImageFormat: OutputFormat;
37
35
  /** DevTools callback. Called after a non-null render with the container and data URI. */
38
36
  onRender?: (container: VContainer, dataUri: string) => void;
39
37
  /** Profiling callback. Called after every renderToDataUri / renderToRaw attempt. */
@@ -41,12 +39,6 @@ export interface RenderConfig {
41
39
  }
42
40
  /** Build the root Takumi container wrapping the VNode children. */
43
41
  export declare function buildTakumiRoot(container: VContainer): TakumiNode;
44
- /**
45
- * Convert a container's VNode children to Takumi nodes.
46
- * Used by the touchstrip native-format path to build the Takumi node tree
47
- * once and share it across all N segment renders in a single flush.
48
- */
49
- export declare function buildTakumiChildren(container: VContainer): TakumiNode[];
50
42
  export declare function measureTree(nodes: VNode[]): {
51
43
  depth: number;
52
44
  count: number;
@@ -61,5 +53,3 @@ export interface RawRenderResult {
61
53
  export declare function renderToRaw(container: VContainer, width: number, height: number, config: RenderConfig): Promise<RawRenderResult | null>;
62
54
  export declare function cropSlice(raw: Buffer, fullWidth: number, column: number, segmentWidth: number, segmentHeight: number): Buffer;
63
55
  export declare function sliceToDataUri(raw: Buffer, fullWidth: number, fullHeight: number, column: number, segmentWidth: number, segmentHeight: number): string;
64
- export declare function sliceToDataUriAsync(raw: Buffer, fullWidth: number, fullHeight: number, column: number, segmentWidth: number, segmentHeight: number): Promise<string>;
65
- export declare function renderSegmentToDataUri(container: VContainer, fullWidth: number, segmentHeight: number, column: number, segmentWidth: number, format: OutputFormat, config: RenderConfig, prebuiltTakumiChildren?: TakumiNode[]): Promise<string | null>;
@@ -1,8 +1,8 @@
1
- import { clearDirtyFlags, isContainerDirty } from "../reconciler/vnode.js";
1
+ import { isContainerDirty } from "../reconciler/vnode.js";
2
2
  import { computeCacheKey, computeTreeHash, fnv1a } from "./cache.js";
3
3
  import { getBufferPool } from "./buffer-pool.js";
4
- import { encodePngAsync } from "./png.js";
5
- import { getImageCache, getTouchstripCache } from "./image-cache.js";
4
+ import { encodePng } from "./png.js";
5
+ import { getImageCache, getTouchStripCache } from "./image-cache.js";
6
6
  import { metrics } from "./metrics.js";
7
7
  import { serializeSvgTree } from "./svg.js";
8
8
  //#region src/render/pipeline.ts
@@ -20,17 +20,11 @@ var depthWarned = false;
20
20
  var SKIP_PROPS = new Set([
21
21
  "children",
22
22
  "className",
23
- "src"
23
+ "src",
24
+ "tw"
24
25
  ]);
25
26
  function copyPropsToNode(target, props) {
26
- const keys = Object.keys(props);
27
- for (let i = 0; i < keys.length; i++) {
28
- const key = keys[i];
29
- if (SKIP_PROPS.has(key)) continue;
30
- const value = props[key];
31
- if (key === "tw") continue;
32
- target[key] = value;
33
- }
27
+ for (const key of Object.keys(props)) if (!SKIP_PROPS.has(key)) target[key] = props[key];
34
28
  }
35
29
  function vnodeToTakumiNode(node, depth = 0) {
36
30
  if (!depthWarned && depth > MAX_DEPTH_WARN) {
@@ -83,14 +77,6 @@ function buildTakumiRoot(container) {
83
77
  children: container.children.map(vnodeToTakumiNode)
84
78
  };
85
79
  }
86
- /**
87
- * Convert a container's VNode children to Takumi nodes.
88
- * Used by the touchstrip native-format path to build the Takumi node tree
89
- * once and share it across all N segment renders in a single flush.
90
- */
91
- function buildTakumiChildren(container) {
92
- return container.children.map(vnodeToTakumiNode);
93
- }
94
80
  function measureTree(nodes) {
95
81
  let maxDepth = 0;
96
82
  let count = 0;
@@ -114,12 +100,11 @@ function emitProfile(config, times, opts) {
114
100
  const stats = measureTree(opts.container.children);
115
101
  const cache = config.imageCacheMaxBytes > 0 ? getImageCache(config.imageCacheMaxBytes) : null;
116
102
  config.onProfile({
117
- vnodeToElementMs: times.t1 - times.t0,
118
- fromJsxMs: times.t2 - times.t1,
119
- takumiRenderMs: times.t3 - times.t2,
120
- hashMs: times.t4 - times.t3,
121
- base64Ms: times.t5 - times.t4,
122
- totalMs: times.t5 - times.t0,
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,
123
108
  skipped: opts.skipped,
124
109
  cacheHit: opts.cacheHit,
125
110
  treeDepth: stats.depth,
@@ -138,7 +123,6 @@ async function renderToDataUri(container, width, height, config) {
138
123
  const t0 = profiling ? performance.now() : 0;
139
124
  let t1 = t0;
140
125
  let t2 = t0;
141
- let t3 = t0;
142
126
  let treeHash;
143
127
  let cacheKey;
144
128
  if (config.caching && config.imageCacheMaxBytes > 0) {
@@ -153,9 +137,7 @@ async function renderToDataUri(container, width, height, config) {
153
137
  t0,
154
138
  t1: tNow,
155
139
  t2: tNow,
156
- t3: tNow,
157
- t4: tNow,
158
- t5: tNow
140
+ t3: tNow
159
141
  }, {
160
142
  skipped: false,
161
143
  cacheHit: true,
@@ -164,27 +146,24 @@ async function renderToDataUri(container, width, height, config) {
164
146
  }
165
147
  container._dupCount = 0;
166
148
  config.onRender?.(container, cached);
167
- clearDirtyFlags(container);
168
149
  return cached;
169
150
  }
170
151
  }
171
152
  let buffer;
172
153
  if (config.renderPool?.isAvailable) {
173
154
  buffer = await config.renderPool.render(container.children, width, height, config.imageFormat, config.devicePixelRatio);
174
- t3 = profiling ? performance.now() : 0;
155
+ t2 = profiling ? performance.now() : 0;
175
156
  t1 = t0;
176
- t2 = t0;
177
157
  } else {
178
158
  const rootNode = buildTakumiRoot(container);
179
159
  t1 = profiling ? performance.now() : 0;
180
- t2 = t1;
181
160
  buffer = await config.renderer.render(rootNode, {
182
161
  width,
183
162
  height,
184
163
  format: config.imageFormat,
185
164
  devicePixelRatio: config.devicePixelRatio
186
165
  });
187
- t3 = profiling ? performance.now() : 0;
166
+ t2 = profiling ? performance.now() : 0;
188
167
  }
189
168
  if (config.caching) {
190
169
  const hash = fnv1a(buffer);
@@ -195,29 +174,24 @@ async function renderToDataUri(container, width, height, config) {
195
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`);
196
175
  }
197
176
  if (profiling) {
198
- const t4 = performance.now();
177
+ const tEnd = performance.now();
199
178
  emitProfile(config, {
200
179
  t0,
201
180
  t1,
202
181
  t2,
203
- t3,
204
- t4,
205
- t5: t4
182
+ t3: tEnd
206
183
  }, {
207
184
  skipped: true,
208
185
  cacheHit: false,
209
186
  container
210
187
  });
211
188
  }
212
- clearDirtyFlags(container);
213
189
  return null;
214
190
  }
215
191
  container.lastSvgHash = hash;
216
192
  container._dupCount = 0;
217
193
  }
218
- const t4 = profiling ? performance.now() : 0;
219
194
  const dataUri = bufferToDataUri(buffer, config.imageFormat);
220
- const t5 = profiling ? performance.now() : 0;
221
195
  if (config.caching && config.imageCacheMaxBytes > 0) {
222
196
  if (treeHash === void 0 || cacheKey === void 0) {
223
197
  treeHash = computeTreeHash(container);
@@ -225,21 +199,21 @@ async function renderToDataUri(container, width, height, config) {
225
199
  }
226
200
  getImageCache(config.imageCacheMaxBytes).set(cacheKey, dataUri, dataUri.length * 2 + 64);
227
201
  }
228
- metrics.recordRender(t3 - t0);
229
- if (profiling) emitProfile(config, {
230
- t0,
231
- t1,
232
- t2,
233
- t3,
234
- t4,
235
- t5
236
- }, {
237
- skipped: false,
238
- cacheHit: false,
239
- container
240
- });
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
+ }
241
216
  config.onRender?.(container, dataUri);
242
- clearDirtyFlags(container);
243
217
  return dataUri;
244
218
  }
245
219
  async function renderToRaw(container, width, height, config) {
@@ -252,13 +226,13 @@ async function renderToRaw(container, width, height, config) {
252
226
  const profiling = config.onProfile != null;
253
227
  const t0 = profiling ? performance.now() : 0;
254
228
  let t1 = t0;
255
- let t3 = t0;
229
+ let t2 = t0;
256
230
  let treeHash;
257
231
  let cacheKey;
258
- if (config.caching && config.touchstripCacheMaxBytes > 0) {
232
+ if (config.caching && config.touchStripCacheMaxBytes > 0) {
259
233
  treeHash = computeTreeHash(container);
260
234
  cacheKey = computeCacheKey(treeHash, width, height, config.devicePixelRatio, "raw");
261
- const cached = getTouchstripCache(config.touchstripCacheMaxBytes).get(cacheKey);
235
+ const cached = getTouchStripCache(config.touchStripCacheMaxBytes).get(cacheKey);
262
236
  if (cached !== void 0) {
263
237
  metrics.recordCacheHit();
264
238
  if (profiling) {
@@ -267,16 +241,13 @@ async function renderToRaw(container, width, height, config) {
267
241
  t0,
268
242
  t1: tNow,
269
243
  t2: tNow,
270
- t3: tNow,
271
- t4: tNow,
272
- t5: tNow
244
+ t3: tNow
273
245
  }, {
274
246
  skipped: false,
275
247
  cacheHit: true,
276
248
  container
277
249
  });
278
250
  }
279
- clearDirtyFlags(container);
280
251
  return {
281
252
  buffer: cached,
282
253
  width,
@@ -287,7 +258,7 @@ async function renderToRaw(container, width, height, config) {
287
258
  let buffer;
288
259
  if (config.renderPool?.isAvailable) {
289
260
  buffer = await config.renderPool.render(container.children, width, height, "raw", config.devicePixelRatio);
290
- t3 = profiling ? performance.now() : 0;
261
+ t2 = profiling ? performance.now() : 0;
291
262
  t1 = t0;
292
263
  } else {
293
264
  const rootNode = buildTakumiRoot(container);
@@ -298,57 +269,51 @@ async function renderToRaw(container, width, height, config) {
298
269
  format: "raw",
299
270
  devicePixelRatio: config.devicePixelRatio
300
271
  });
301
- t3 = profiling ? performance.now() : 0;
272
+ t2 = profiling ? performance.now() : 0;
302
273
  }
303
274
  if (config.caching) {
304
275
  const hash = fnv1a(buffer);
305
276
  if (hash === container.lastSvgHash) {
306
277
  metrics.recordHashDedup();
307
278
  if (profiling) {
308
- const t4 = performance.now();
279
+ const tEnd = performance.now();
309
280
  emitProfile(config, {
310
281
  t0,
311
282
  t1,
312
- t2: t1,
313
- t3,
314
- t4,
315
- t5: t4
283
+ t2,
284
+ t3: tEnd
316
285
  }, {
317
286
  skipped: true,
318
287
  cacheHit: false,
319
288
  container
320
289
  });
321
290
  }
322
- clearDirtyFlags(container);
323
291
  return null;
324
292
  }
325
293
  container.lastSvgHash = hash;
326
294
  }
327
295
  const buf = Buffer.isBuffer(buffer) ? buffer : Buffer.from(buffer);
328
- if (config.caching && config.touchstripCacheMaxBytes > 0) {
296
+ if (config.caching && config.touchStripCacheMaxBytes > 0) {
329
297
  if (treeHash === void 0 || cacheKey === void 0) {
330
298
  treeHash = computeTreeHash(container);
331
299
  cacheKey = computeCacheKey(treeHash, width, height, config.devicePixelRatio, "raw");
332
300
  }
333
- getTouchstripCache(config.touchstripCacheMaxBytes).set(cacheKey, buf, buf.byteLength + 64);
301
+ getTouchStripCache(config.touchStripCacheMaxBytes).set(cacheKey, buf, buf.byteLength + 64);
334
302
  }
335
- metrics.recordRender(t3 - t0);
303
+ metrics.recordRender(t2 - t0);
336
304
  if (profiling) {
337
305
  const tEnd = performance.now();
338
306
  emitProfile(config, {
339
307
  t0,
340
308
  t1,
341
- t2: t1,
342
- t3,
343
- t4: tEnd,
344
- t5: tEnd
309
+ t2,
310
+ t3: tEnd
345
311
  }, {
346
312
  skipped: false,
347
313
  cacheHit: false,
348
314
  container
349
315
  });
350
316
  }
351
- clearDirtyFlags(container);
352
317
  return {
353
318
  buffer: buf,
354
319
  width,
@@ -369,40 +334,11 @@ function cropSlice(raw, fullWidth, column, segmentWidth, segmentHeight) {
369
334
  }
370
335
  return slice;
371
336
  }
372
- async function sliceToDataUriAsync(raw, fullWidth, fullHeight, column, segmentWidth, segmentHeight) {
337
+ function sliceToDataUri(raw, fullWidth, fullHeight, column, segmentWidth, segmentHeight) {
373
338
  const cropped = cropSlice(raw, fullWidth, column, segmentWidth, segmentHeight);
374
- const png = await encodePngAsync(segmentWidth, segmentHeight, cropped);
339
+ const png = encodePng(segmentWidth, segmentHeight, cropped);
375
340
  getBufferPool().release(cropped);
376
341
  return bufferToDataUri(png, "png");
377
342
  }
378
- async function renderSegmentToDataUri(container, fullWidth, segmentHeight, column, segmentWidth, format, config, prebuiltTakumiChildren) {
379
- if (container.children.length === 0) return null;
380
- const children = prebuiltTakumiChildren ?? container.children.map(vnodeToTakumiNode);
381
- const innerNode = {
382
- type: "container",
383
- style: {
384
- ...ROOT_STYLE,
385
- width: fullWidth,
386
- height: segmentHeight,
387
- marginLeft: -(column * segmentWidth)
388
- },
389
- children
390
- };
391
- const clipNode = {
392
- type: "container",
393
- style: {
394
- width: segmentWidth,
395
- height: segmentHeight,
396
- overflow: "hidden"
397
- },
398
- children: [innerNode]
399
- };
400
- return bufferToDataUri(await config.renderer.render(clipNode, {
401
- width: segmentWidth,
402
- height: segmentHeight,
403
- format,
404
- devicePixelRatio: config.devicePixelRatio
405
- }), format);
406
- }
407
343
  //#endregion
408
- export { bufferToDataUri, buildTakumiChildren, buildTakumiRoot, measureTree, renderSegmentToDataUri, renderToDataUri, renderToRaw, sliceToDataUriAsync };
344
+ export { bufferToDataUri, buildTakumiRoot, measureTree, renderToDataUri, renderToRaw, sliceToDataUri };
@@ -6,12 +6,3 @@
6
6
  * @param rgba Raw RGBA pixel data (width × height × 4 bytes, row-major).
7
7
  */
8
8
  export declare function encodePng(width: number, height: number, rgba: Buffer | Uint8Array): Buffer;
9
- /**
10
- * Encode raw RGBA pixels into a PNG buffer (async).
11
- * Uses libuv thread pool for deflate compression, avoiding main-thread blocking.
12
- *
13
- * @param width Image width in pixels.
14
- * @param height Image height in pixels.
15
- * @param rgba Raw RGBA pixel data (width × height × 4 bytes, row-major).
16
- */
17
- export declare function encodePngAsync(width: number, height: number, rgba: Buffer | Uint8Array): Promise<Buffer>;
@@ -1,6 +1,5 @@
1
1
  import { getBufferPool } from "./buffer-pool.js";
2
- import { deflate } from "node:zlib";
3
- import { promisify } from "node:util";
2
+ import { deflateSync } from "node:zlib";
4
3
  //#region src/render/png.ts
5
4
  var crcTable = new Uint32Array(256);
6
5
  for (let n = 0; n < 256; n++) {
@@ -37,7 +36,6 @@ var PNG_SIGNATURE = Buffer.from([
37
36
  26,
38
37
  10
39
38
  ]);
40
- var deflateAsync = promisify(deflate);
41
39
  function buildIhdr(width, height) {
42
40
  const ihdr = Buffer.alloc(13);
43
41
  ihdr.writeUInt32BE(width, 0);
@@ -71,19 +69,18 @@ function assemblePng(ihdr, compressed) {
71
69
  ]);
72
70
  }
73
71
  /**
74
- * Encode raw RGBA pixels into a PNG buffer (async).
75
- * Uses libuv thread pool for deflate compression, avoiding main-thread blocking.
72
+ * Encode raw RGBA pixels into a PNG buffer (synchronous).
76
73
  *
77
74
  * @param width Image width in pixels.
78
75
  * @param height Image height in pixels.
79
76
  * @param rgba Raw RGBA pixel data (width × height × 4 bytes, row-major).
80
77
  */
81
- async function encodePngAsync(width, height, rgba) {
78
+ function encodePng(width, height, rgba) {
82
79
  const ihdr = buildIhdr(width, height);
83
80
  const filtered = buildFilteredScanlines(width, height, rgba);
84
- const compressed = await deflateAsync(filtered);
81
+ const compressed = deflateSync(filtered);
85
82
  getBufferPool().release(filtered);
86
83
  return assemblePng(ihdr, compressed);
87
84
  }
88
85
  //#endregion
89
- export { encodePngAsync };
86
+ export { encodePng };
@@ -19,8 +19,6 @@ export declare class RenderPool {
19
19
  * Returns the raw raster buffer.
20
20
  */
21
21
  render(vnodes: VNode[], width: number, height: number, format: string, dpr: number): Promise<Buffer>;
22
- /** Gracefully shut down the worker. */
23
- shutdown(): Promise<void>;
24
22
  private handleResponse;
25
23
  private handleWorkerDeath;
26
24
  }
@@ -36,7 +36,7 @@ var RenderPool = class {
36
36
  }
37
37
  async doInitialize() {
38
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) => ({
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
40
  name: f.name,
41
41
  data: f.data,
42
42
  weight: f.weight,
@@ -111,17 +111,6 @@ var RenderPool = class {
111
111
  });
112
112
  });
113
113
  }
114
- /** Gracefully shut down the worker. */
115
- async shutdown() {
116
- if (this.worker != null) {
117
- for (const [_, req] of this.pending) req.reject(/* @__PURE__ */ new Error("Worker shutting down"));
118
- this.pending.clear();
119
- this.worker.postMessage({ type: "shutdown" });
120
- await this.worker.terminate();
121
- this.worker = null;
122
- }
123
- this.ready = false;
124
- }
125
114
  handleResponse(msg) {
126
115
  if (msg.type === "ready") return;
127
116
  const req = this.pending.get(msg.id);
@@ -1,7 +1,7 @@
1
1
  import { ComponentType } from 'react';
2
- import { WillAppearEvent } from '@elgato/streamdeck';
3
2
  import { JsonObject } from '@elgato/utils';
4
- import { ActionDefinition, EventMap, StreamDeckAccess, WrapperComponent } from '../types';
3
+ import { AdapterWillAppearEvent, StreamDeckAdapter } from '../adapter/types';
4
+ import { ActionDefinition, EventMap, WrapperComponent } from '../types';
5
5
  import { RenderConfig } from '../render/pipeline';
6
6
  import { RegistryObserver } from '../devtools/observers/lifecycle';
7
7
  export declare class RootRegistry {
@@ -9,21 +9,17 @@ export declare class RootRegistry {
9
9
  private touchStripRoots;
10
10
  private touchStripActions;
11
11
  private renderConfig;
12
- private renderDebounceMs;
13
- private sdkInstance;
12
+ private adapter;
14
13
  private globalSettings;
15
14
  private onGlobalSettingsChange;
16
15
  private wrapper?;
17
- private flushCoordinator;
18
16
  /** DevTools observer. Set externally by startDevtoolsServer(). null when devtools is off. */
19
17
  observer: RegistryObserver | null;
20
- constructor(renderConfig: RenderConfig, renderDebounceMs: number, sdkInstance: StreamDeckAccess["sdk"], onGlobalSettingsChange: (settings: JsonObject) => Promise<void>, wrapper?: WrapperComponent);
18
+ constructor(renderConfig: RenderConfig, adapter: StreamDeckAdapter, onGlobalSettingsChange: (settings: JsonObject) => Promise<void>, wrapper?: WrapperComponent);
21
19
  setGlobalSettings(settings: JsonObject): void;
22
- create(ev: WillAppearEvent<JsonObject>, component: ComponentType, definition: ActionDefinition): void;
20
+ create(ev: AdapterWillAppearEvent, component: ComponentType, definition: ActionDefinition): void;
23
21
  private registerTouchStripColumn;
24
- private getEncoderColumn;
25
22
  destroy(contextId: string): void;
26
- private static readonly INTERACTION_EVENTS;
27
23
  dispatch<K extends keyof EventMap>(contextId: string, event: K, payload: EventMap[K]): void;
28
24
  private dispatchToTouchStrip;
29
25
  updateSettings(contextId: string, settings: JsonObject): void;