@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.
- package/dist/action.js +0 -1
- package/dist/adapter/index.d.ts +2 -0
- package/dist/adapter/physical-device.d.ts +2 -0
- package/dist/adapter/physical-device.js +153 -0
- package/dist/adapter/types.d.ts +127 -0
- package/dist/devtools/bridge.d.ts +2 -2
- package/dist/devtools/bridge.js +7 -8
- package/dist/devtools/highlight.d.ts +1 -2
- package/dist/devtools/highlight.js +4 -3
- package/dist/devtools/types.d.ts +5 -5
- package/dist/hooks/animation.d.ts +1 -1
- package/dist/hooks/animation.js +2 -2
- package/dist/hooks/events.js +1 -1
- package/dist/hooks/sdk.js +11 -11
- package/dist/hooks/utility.js +3 -2
- package/dist/index.d.ts +3 -1
- package/dist/index.js +2 -1
- package/dist/plugin.js +69 -100
- package/dist/reconciler/vnode.d.ts +0 -2
- package/dist/reconciler/vnode.js +0 -1
- package/dist/render/cache.d.ts +5 -17
- package/dist/render/cache.js +7 -29
- package/dist/render/image-cache.d.ts +8 -7
- package/dist/render/image-cache.js +33 -17
- package/dist/render/metrics.d.ts +9 -10
- package/dist/render/metrics.js +36 -39
- package/dist/render/pipeline.d.ts +4 -14
- package/dist/render/pipeline.js +47 -111
- package/dist/render/png.d.ts +0 -9
- package/dist/render/png.js +5 -8
- package/dist/render/render-pool.d.ts +0 -2
- package/dist/render/render-pool.js +1 -12
- package/dist/roots/registry.d.ts +5 -9
- package/dist/roots/registry.js +10 -27
- package/dist/roots/root.d.ts +7 -34
- package/dist/roots/root.js +23 -90
- package/dist/roots/touchstrip-root.d.ts +6 -32
- package/dist/roots/touchstrip-root.js +61 -181
- package/dist/types.d.ts +23 -19
- package/package.json +6 -4
- package/dist/node_modules/.bun/xxhash-wasm@1.1.0/node_modules/xxhash-wasm/esm/xxhash-wasm.js +0 -3157
- package/dist/roots/flush-coordinator.d.ts +0 -18
- 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
|
-
|
|
9
|
-
|
|
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
|
|
32
|
-
|
|
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>;
|
package/dist/render/pipeline.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import {
|
|
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 {
|
|
5
|
-
import { getImageCache,
|
|
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
|
|
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
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
|
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(
|
|
229
|
-
if (profiling)
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
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
|
|
229
|
+
let t2 = t0;
|
|
256
230
|
let treeHash;
|
|
257
231
|
let cacheKey;
|
|
258
|
-
if (config.caching && config.
|
|
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 =
|
|
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
|
-
|
|
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
|
-
|
|
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
|
|
279
|
+
const tEnd = performance.now();
|
|
309
280
|
emitProfile(config, {
|
|
310
281
|
t0,
|
|
311
282
|
t1,
|
|
312
|
-
t2
|
|
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.
|
|
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
|
-
|
|
301
|
+
getTouchStripCache(config.touchStripCacheMaxBytes).set(cacheKey, buf, buf.byteLength + 64);
|
|
334
302
|
}
|
|
335
|
-
metrics.recordRender(
|
|
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
|
|
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
|
-
|
|
337
|
+
function sliceToDataUri(raw, fullWidth, fullHeight, column, segmentWidth, segmentHeight) {
|
|
373
338
|
const cropped = cropSlice(raw, fullWidth, column, segmentWidth, segmentHeight);
|
|
374
|
-
const png =
|
|
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,
|
|
344
|
+
export { bufferToDataUri, buildTakumiRoot, measureTree, renderToDataUri, renderToRaw, sliceToDataUri };
|
package/dist/render/png.d.ts
CHANGED
|
@@ -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>;
|
package/dist/render/png.js
CHANGED
|
@@ -1,6 +1,5 @@
|
|
|
1
1
|
import { getBufferPool } from "./buffer-pool.js";
|
|
2
|
-
import {
|
|
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 (
|
|
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
|
-
|
|
78
|
+
function encodePng(width, height, rgba) {
|
|
82
79
|
const ihdr = buildIhdr(width, height);
|
|
83
80
|
const filtered = buildFilteredScanlines(width, height, rgba);
|
|
84
|
-
const compressed =
|
|
81
|
+
const compressed = deflateSync(filtered);
|
|
85
82
|
getBufferPool().release(filtered);
|
|
86
83
|
return assemblePng(ihdr, compressed);
|
|
87
84
|
}
|
|
88
85
|
//#endregion
|
|
89
|
-
export {
|
|
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);
|
package/dist/roots/registry.d.ts
CHANGED
|
@@ -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 {
|
|
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
|
|
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,
|
|
18
|
+
constructor(renderConfig: RenderConfig, adapter: StreamDeckAdapter, onGlobalSettingsChange: (settings: JsonObject) => Promise<void>, wrapper?: WrapperComponent);
|
|
21
19
|
setGlobalSettings(settings: JsonObject): void;
|
|
22
|
-
create(ev:
|
|
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;
|