@fcannizzaro/streamdeck-react 0.1.12 → 0.1.14
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/README.md +11 -8
- package/dist/action.d.ts +2 -2
- package/dist/action.js +2 -1
- package/dist/bundler-shared.d.ts +31 -0
- package/dist/bundler-shared.js +64 -2
- package/dist/font-inline.js +1 -1
- package/dist/google-font.d.ts +61 -0
- package/dist/google-font.js +124 -0
- package/dist/index.d.ts +4 -1
- package/dist/index.js +2 -1
- package/dist/manifest-extract.d.ts +32 -0
- package/dist/manifest-extract.js +141 -0
- package/dist/manifest-gen.d.ts +52 -0
- package/dist/manifest-gen.js +229 -0
- package/dist/manifest-types.d.ts +238 -0
- package/dist/plugin.js +44 -35
- package/dist/render/render-pool.js +1 -1
- package/dist/rollup.d.ts +37 -10
- package/dist/rollup.js +43 -14
- package/dist/roots/registry.js +20 -20
- package/dist/types.d.ts +35 -35
- package/dist/vite.d.ts +24 -8
- package/dist/vite.js +48 -13
- package/package.json +10 -5
- package/dist/manifest-codegen.d.ts +0 -38
- package/dist/manifest-codegen.js +0 -110
|
@@ -0,0 +1,229 @@
|
|
|
1
|
+
import { dirname } from "node:path";
|
|
2
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
|
|
3
|
+
//#region src/manifest-gen.ts
|
|
4
|
+
/**
|
|
5
|
+
* Validate just the plugin UUID format.
|
|
6
|
+
*
|
|
7
|
+
* Used in `buildStart` for early error reporting before action
|
|
8
|
+
* extraction is complete.
|
|
9
|
+
*
|
|
10
|
+
* @returns A validation error if the UUID is invalid, or `null` if valid.
|
|
11
|
+
*/
|
|
12
|
+
function validatePluginUUID(uuid) {
|
|
13
|
+
if (!/^([a-z0-9-]+)(\.[a-z0-9-]+)+$/.test(uuid)) return {
|
|
14
|
+
field: "uuid",
|
|
15
|
+
message: `Plugin UUID "${uuid}" must be in reverse-DNS format (lowercase alphanumeric, hyphens, periods)`
|
|
16
|
+
};
|
|
17
|
+
return null;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Validate a full ManifestConfig for correctness.
|
|
21
|
+
*
|
|
22
|
+
* Checks:
|
|
23
|
+
* - All action UUIDs are prefixed with the plugin UUID
|
|
24
|
+
* - No duplicate action UUIDs
|
|
25
|
+
* - Plugin UUID matches reverse-DNS pattern
|
|
26
|
+
*
|
|
27
|
+
* Called in `writeBundle` after action extraction is complete.
|
|
28
|
+
*/
|
|
29
|
+
function validateManifestConfig(config, warn) {
|
|
30
|
+
const errors = [];
|
|
31
|
+
const uuidError = validatePluginUUID(config.uuid);
|
|
32
|
+
if (uuidError) errors.push(uuidError);
|
|
33
|
+
const prefix = config.uuid + ".";
|
|
34
|
+
const seenUuids = /* @__PURE__ */ new Set();
|
|
35
|
+
for (const action of config.actions) {
|
|
36
|
+
if (!action.uuid.startsWith(prefix)) {
|
|
37
|
+
const error = {
|
|
38
|
+
field: `actions[${action.uuid}].uuid`,
|
|
39
|
+
message: `Action UUID "${action.uuid}" must be prefixed with plugin UUID "${config.uuid}."`
|
|
40
|
+
};
|
|
41
|
+
errors.push(error);
|
|
42
|
+
warn?.(`[@fcannizzaro/streamdeck-react] ${error.message}`);
|
|
43
|
+
}
|
|
44
|
+
if (seenUuids.has(action.uuid)) errors.push({
|
|
45
|
+
field: `actions[${action.uuid}].uuid`,
|
|
46
|
+
message: `Duplicate action UUID "${action.uuid}"`
|
|
47
|
+
});
|
|
48
|
+
seenUuids.add(action.uuid);
|
|
49
|
+
}
|
|
50
|
+
return errors;
|
|
51
|
+
}
|
|
52
|
+
/** Add a key to the record only if the value is defined. */
|
|
53
|
+
function addIf(record, key, value) {
|
|
54
|
+
if (value !== void 0 && value !== null) record[key] = value;
|
|
55
|
+
}
|
|
56
|
+
function buildTriggerDescription(td) {
|
|
57
|
+
const out = {};
|
|
58
|
+
addIf(out, "Rotate", td.rotate);
|
|
59
|
+
addIf(out, "Push", td.push);
|
|
60
|
+
addIf(out, "Touch", td.touch);
|
|
61
|
+
addIf(out, "LongTouch", td.longTouch);
|
|
62
|
+
return out;
|
|
63
|
+
}
|
|
64
|
+
function buildEncoder(encoder) {
|
|
65
|
+
const out = {};
|
|
66
|
+
addIf(out, "layout", encoder.layout);
|
|
67
|
+
addIf(out, "Icon", encoder.icon);
|
|
68
|
+
addIf(out, "StackColor", encoder.stackColor);
|
|
69
|
+
addIf(out, "background", encoder.background);
|
|
70
|
+
if (encoder.triggerDescription) out["TriggerDescription"] = buildTriggerDescription(encoder.triggerDescription);
|
|
71
|
+
return out;
|
|
72
|
+
}
|
|
73
|
+
function buildState(state) {
|
|
74
|
+
const out = { Image: state.image };
|
|
75
|
+
addIf(out, "Name", state.name);
|
|
76
|
+
addIf(out, "Title", state.title);
|
|
77
|
+
addIf(out, "ShowTitle", state.showTitle);
|
|
78
|
+
addIf(out, "TitleAlignment", state.titleAlignment);
|
|
79
|
+
addIf(out, "TitleColor", state.titleColor);
|
|
80
|
+
addIf(out, "FontFamily", state.fontFamily);
|
|
81
|
+
addIf(out, "FontSize", state.fontSize);
|
|
82
|
+
addIf(out, "FontStyle", state.fontStyle);
|
|
83
|
+
addIf(out, "FontUnderline", state.fontUnderline);
|
|
84
|
+
addIf(out, "MultiActionImage", state.multiActionImage);
|
|
85
|
+
return out;
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Derive Controllers from an action definition.
|
|
89
|
+
*
|
|
90
|
+
* Priority:
|
|
91
|
+
* 1. Explicit `controllers` on info → use as-is
|
|
92
|
+
* 2. Derived from key/dial/touchStrip presence:
|
|
93
|
+
* - key → includes "Keypad"
|
|
94
|
+
* - dial or touchStrip → includes "Encoder"
|
|
95
|
+
* - both → ["Keypad", "Encoder"]
|
|
96
|
+
* 3. Default → ["Keypad"]
|
|
97
|
+
*/
|
|
98
|
+
function deriveControllers(action) {
|
|
99
|
+
const info = action.info;
|
|
100
|
+
if (info?.controllers) return info.controllers.filter((c) => c != null);
|
|
101
|
+
const hasKey = action.key != null;
|
|
102
|
+
const hasEncoder = action.dial != null || action.touchStrip != null;
|
|
103
|
+
if (hasKey && hasEncoder) return ["Keypad", "Encoder"];
|
|
104
|
+
if (hasEncoder) return ["Encoder"];
|
|
105
|
+
if (hasKey) return ["Keypad"];
|
|
106
|
+
if (info?.encoder) return ["Encoder"];
|
|
107
|
+
return ["Keypad"];
|
|
108
|
+
}
|
|
109
|
+
function buildAction(action) {
|
|
110
|
+
const info = action.info;
|
|
111
|
+
if (!info) throw new Error(`[@fcannizzaro/streamdeck-react] Action "${action.uuid}" is missing \`info\`. Add info: { name, icon } to the defineAction() call.`);
|
|
112
|
+
const controllers = deriveControllers(action);
|
|
113
|
+
const states = info.states ? info.states.map(buildState) : [{ Image: info.icon }];
|
|
114
|
+
const out = {
|
|
115
|
+
UUID: action.uuid,
|
|
116
|
+
Name: info.name,
|
|
117
|
+
Icon: info.icon,
|
|
118
|
+
Controllers: controllers,
|
|
119
|
+
States: states
|
|
120
|
+
};
|
|
121
|
+
addIf(out, "Tooltip", info.tooltip);
|
|
122
|
+
addIf(out, "DisableAutomaticStates", info.disableAutomaticStates);
|
|
123
|
+
addIf(out, "DisableCaching", info.disableCaching);
|
|
124
|
+
addIf(out, "SupportedInMultiActions", info.supportedInMultiActions);
|
|
125
|
+
addIf(out, "SupportedInKeyLogicActions", info.supportedInKeyLogicActions);
|
|
126
|
+
addIf(out, "VisibleInActionsList", info.visibleInActionsList);
|
|
127
|
+
addIf(out, "UserTitleEnabled", info.userTitleEnabled);
|
|
128
|
+
addIf(out, "PropertyInspectorPath", info.propertyInspectorPath);
|
|
129
|
+
addIf(out, "SupportURL", info.supportUrl);
|
|
130
|
+
if (info.encoder) out["Encoder"] = buildEncoder(info.encoder);
|
|
131
|
+
if (info.os) out["OS"] = info.os;
|
|
132
|
+
return out;
|
|
133
|
+
}
|
|
134
|
+
var DEFAULT_OS = [{
|
|
135
|
+
platform: "mac",
|
|
136
|
+
minimumVersion: "13"
|
|
137
|
+
}, {
|
|
138
|
+
platform: "windows",
|
|
139
|
+
minimumVersion: "10"
|
|
140
|
+
}];
|
|
141
|
+
function buildOS(osEntries) {
|
|
142
|
+
return osEntries.filter((o) => o != null).map((o) => ({
|
|
143
|
+
Platform: o.platform,
|
|
144
|
+
MinimumVersion: o.minimumVersion
|
|
145
|
+
}));
|
|
146
|
+
}
|
|
147
|
+
function buildNodejs(nodejs) {
|
|
148
|
+
const out = { Version: nodejs.version };
|
|
149
|
+
addIf(out, "Debug", nodejs.debug);
|
|
150
|
+
addIf(out, "GenerateProfilerOutput", nodejs.generateProfilerOutput);
|
|
151
|
+
return out;
|
|
152
|
+
}
|
|
153
|
+
function buildProfile(profile) {
|
|
154
|
+
const out = {
|
|
155
|
+
Name: profile.name,
|
|
156
|
+
DeviceType: profile.deviceType
|
|
157
|
+
};
|
|
158
|
+
addIf(out, "AutoInstall", profile.autoInstall);
|
|
159
|
+
addIf(out, "DontAutoSwitchWhenInstalled", profile.dontAutoSwitchWhenInstalled);
|
|
160
|
+
addIf(out, "Readonly", profile.readonly);
|
|
161
|
+
return out;
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Build the full manifest JSON object from a ManifestConfig.
|
|
165
|
+
*
|
|
166
|
+
* Applies all auto-derivation defaults and transforms camelCase
|
|
167
|
+
* to the PascalCase format expected by the Elgato schema.
|
|
168
|
+
*
|
|
169
|
+
* @param config - The manifest configuration
|
|
170
|
+
* @param codePath - Override CodePath (typically derived from bundler output)
|
|
171
|
+
*/
|
|
172
|
+
function buildManifestJson(config, codePath) {
|
|
173
|
+
const osEntries = config.os ? config.os.filter((o) => o != null) : DEFAULT_OS;
|
|
174
|
+
const nodejs = config.nodejs ?? { version: "24" };
|
|
175
|
+
const out = {
|
|
176
|
+
$schema: "https://schemas.elgato.com/streamdeck/plugins/manifest.json",
|
|
177
|
+
UUID: config.uuid,
|
|
178
|
+
Name: config.name,
|
|
179
|
+
Author: config.author,
|
|
180
|
+
Description: config.description,
|
|
181
|
+
Icon: config.icon,
|
|
182
|
+
Version: config.version,
|
|
183
|
+
CodePath: codePath ?? config.codePath ?? "bin/plugin.mjs",
|
|
184
|
+
OS: buildOS(osEntries),
|
|
185
|
+
Nodejs: buildNodejs(nodejs),
|
|
186
|
+
SDKVersion: config.sdkVersion ?? 2,
|
|
187
|
+
Software: { MinimumVersion: config.software?.minimumVersion ?? "7.1" },
|
|
188
|
+
Category: config.category ?? config.name,
|
|
189
|
+
CategoryIcon: config.categoryIcon ?? config.icon,
|
|
190
|
+
Actions: config.actions.map(buildAction)
|
|
191
|
+
};
|
|
192
|
+
addIf(out, "URL", config.url);
|
|
193
|
+
addIf(out, "SupportURL", config.supportUrl);
|
|
194
|
+
addIf(out, "PropertyInspectorPath", config.propertyInspectorPath);
|
|
195
|
+
addIf(out, "DefaultWindowSize", config.defaultWindowSize);
|
|
196
|
+
addIf(out, "CodePathMac", config.codePathMac);
|
|
197
|
+
addIf(out, "CodePathWin", config.codePathWin);
|
|
198
|
+
if (config.applicationsToMonitor) out["ApplicationsToMonitor"] = config.applicationsToMonitor;
|
|
199
|
+
if (config.profiles?.length) out["Profiles"] = config.profiles.map(buildProfile);
|
|
200
|
+
return out;
|
|
201
|
+
}
|
|
202
|
+
/**
|
|
203
|
+
* Generate the full manifest JSON string from a ManifestConfig.
|
|
204
|
+
*
|
|
205
|
+
* @param config - The manifest configuration
|
|
206
|
+
* @param codePath - Override CodePath (typically derived from bundler output)
|
|
207
|
+
* @returns Formatted JSON string
|
|
208
|
+
*/
|
|
209
|
+
function generateManifestJsonString(config, codePath) {
|
|
210
|
+
const json = buildManifestJson(config, codePath);
|
|
211
|
+
return JSON.stringify(json, null, 2) + "\n";
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* Write manifest.json only when the content has changed.
|
|
215
|
+
* Creates the parent directory if it does not exist.
|
|
216
|
+
*
|
|
217
|
+
* @returns `true` if the file was written, `false` if content was unchanged.
|
|
218
|
+
*/
|
|
219
|
+
function writeManifestIfChanged(outPath, content) {
|
|
220
|
+
if (existsSync(outPath)) {
|
|
221
|
+
if (readFileSync(outPath, "utf-8") === content) return false;
|
|
222
|
+
}
|
|
223
|
+
const dir = dirname(outPath);
|
|
224
|
+
if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
|
|
225
|
+
writeFileSync(outPath, content);
|
|
226
|
+
return true;
|
|
227
|
+
}
|
|
228
|
+
//#endregion
|
|
229
|
+
export { generateManifestJsonString, validateManifestConfig, validatePluginUUID, writeManifestIfChanged };
|
|
@@ -0,0 +1,238 @@
|
|
|
1
|
+
export type ManifestController = "Encoder" | "Keypad";
|
|
2
|
+
export interface ManifestOSInfo {
|
|
3
|
+
platform: "mac" | "windows";
|
|
4
|
+
minimumVersion: string;
|
|
5
|
+
}
|
|
6
|
+
export interface ManifestNodejsInfo {
|
|
7
|
+
/**
|
|
8
|
+
* Node.js version to use.
|
|
9
|
+
* @default "24"
|
|
10
|
+
*/
|
|
11
|
+
version: "20" | "24";
|
|
12
|
+
/**
|
|
13
|
+
* Debug configuration. `"enabled"` and `"break"` are shortcuts for
|
|
14
|
+
* `--inspect` and `--inspect-brk` respectively. A custom string is
|
|
15
|
+
* passed as-is (e.g. `"--inspect=127.0.0.1:8090"`).
|
|
16
|
+
*/
|
|
17
|
+
debug?: string;
|
|
18
|
+
/** Enable profiler output. */
|
|
19
|
+
generateProfilerOutput?: boolean;
|
|
20
|
+
}
|
|
21
|
+
export interface ManifestStateInfo {
|
|
22
|
+
/** Path to state image (extension omitted). 72×72 and 144×144 @2x. */
|
|
23
|
+
image: string;
|
|
24
|
+
/** State name (shown in multi-action state selector). */
|
|
25
|
+
name?: string;
|
|
26
|
+
/** Default title text. */
|
|
27
|
+
title?: string;
|
|
28
|
+
/** Whether to show the title. */
|
|
29
|
+
showTitle?: boolean;
|
|
30
|
+
/** Title alignment. */
|
|
31
|
+
titleAlignment?: "bottom" | "middle" | "top";
|
|
32
|
+
/** Title color (hex). */
|
|
33
|
+
titleColor?: string;
|
|
34
|
+
/** Font family for the title. */
|
|
35
|
+
fontFamily?: string;
|
|
36
|
+
/** Font size for the title. */
|
|
37
|
+
fontSize?: number;
|
|
38
|
+
/** Font style for the title. */
|
|
39
|
+
fontStyle?: "" | "Bold" | "Bold Italic" | "Italic" | "Regular";
|
|
40
|
+
/** Whether the title is underlined. */
|
|
41
|
+
fontUnderline?: boolean;
|
|
42
|
+
/** Image shown when the action is in a multi-action. */
|
|
43
|
+
multiActionImage?: string;
|
|
44
|
+
}
|
|
45
|
+
export interface ManifestTriggerDescription {
|
|
46
|
+
/** Dial rotation description. */
|
|
47
|
+
rotate?: string;
|
|
48
|
+
/** Dial press description. */
|
|
49
|
+
push?: string;
|
|
50
|
+
/** Touch tap description. */
|
|
51
|
+
touch?: string;
|
|
52
|
+
/** Long touch description. */
|
|
53
|
+
longTouch?: string;
|
|
54
|
+
}
|
|
55
|
+
export interface ManifestEncoderInfo {
|
|
56
|
+
/**
|
|
57
|
+
* Touch screen layout. Pre-defined: `$X1`, `$A0`, `$A1`, `$B1`,
|
|
58
|
+
* `$B2`, `$C1`. Or a path to a custom `.json` layout file.
|
|
59
|
+
*/
|
|
60
|
+
layout?: string;
|
|
61
|
+
/** Encoder icon (extension omitted). 72×72 and 144×144 @2x. */
|
|
62
|
+
icon?: string;
|
|
63
|
+
/** Background color for dial stack (hex). */
|
|
64
|
+
stackColor?: string;
|
|
65
|
+
/** Touchscreen background image (extension omitted). 200×100 and 400×200 @2x. */
|
|
66
|
+
background?: string;
|
|
67
|
+
/** Descriptions shown to the user for each interaction type. */
|
|
68
|
+
triggerDescription?: ManifestTriggerDescription;
|
|
69
|
+
}
|
|
70
|
+
export interface ManifestProfileInfo {
|
|
71
|
+
/** Path to .streamDeckProfile file (extension omitted). */
|
|
72
|
+
name: string;
|
|
73
|
+
/**
|
|
74
|
+
* Target device type.
|
|
75
|
+
*
|
|
76
|
+
* Common values:
|
|
77
|
+
* 0 = Stream Deck, 1 = Mini, 2 = XL, 5 = Pedal,
|
|
78
|
+
* 7 = Stream Deck +, 9 = Neo, 10 = Studio
|
|
79
|
+
*/
|
|
80
|
+
deviceType: number;
|
|
81
|
+
/** Auto-install when plugin is installed. @default true */
|
|
82
|
+
autoInstall?: boolean;
|
|
83
|
+
/** Don't auto-switch to profile on first install. @default false */
|
|
84
|
+
dontAutoSwitchWhenInstalled?: boolean;
|
|
85
|
+
/** Profile is read-only. @default false */
|
|
86
|
+
readonly?: boolean;
|
|
87
|
+
}
|
|
88
|
+
export interface ActionManifestInfo {
|
|
89
|
+
/** Action display name in Stream Deck's action list. */
|
|
90
|
+
name: string;
|
|
91
|
+
/** Path to action icon (extension omitted). 20×20 and 40×40 @2x. */
|
|
92
|
+
icon: string;
|
|
93
|
+
/**
|
|
94
|
+
* Skip this action during auto-extraction for manifest generation.
|
|
95
|
+
* The action is still registered at runtime but excluded from the
|
|
96
|
+
* generated manifest.json.
|
|
97
|
+
*
|
|
98
|
+
* @default false
|
|
99
|
+
*/
|
|
100
|
+
disabled?: boolean;
|
|
101
|
+
/** Hover tooltip in the actions list. */
|
|
102
|
+
tooltip?: string;
|
|
103
|
+
/**
|
|
104
|
+
* Custom states. When omitted, a single state is generated
|
|
105
|
+
* using the `icon` field as the state image.
|
|
106
|
+
* @default [{ image: icon }]
|
|
107
|
+
*/
|
|
108
|
+
states?: ManifestStateInfo[];
|
|
109
|
+
/** Encoder config (layout, triggerDescription, background). */
|
|
110
|
+
encoder?: ManifestEncoderInfo;
|
|
111
|
+
/** Disable automatic state toggling. @default false */
|
|
112
|
+
disableAutomaticStates?: boolean;
|
|
113
|
+
/** Disable Stream Deck image caching. @default false */
|
|
114
|
+
disableCaching?: boolean;
|
|
115
|
+
/** Available in multi-actions. @default true */
|
|
116
|
+
supportedInMultiActions?: boolean;
|
|
117
|
+
/** Available in key logic actions (SD 7.0+). @default true */
|
|
118
|
+
supportedInKeyLogicActions?: boolean;
|
|
119
|
+
/** Visible in the actions list. @default true */
|
|
120
|
+
visibleInActionsList?: boolean;
|
|
121
|
+
/** Allow user to edit title. @default true */
|
|
122
|
+
userTitleEnabled?: boolean;
|
|
123
|
+
/** Action-specific property inspector HTML path. */
|
|
124
|
+
propertyInspectorPath?: string;
|
|
125
|
+
/** Action support URL. */
|
|
126
|
+
supportUrl?: string;
|
|
127
|
+
/** OS restriction for this action. */
|
|
128
|
+
os?: ("mac" | "windows")[];
|
|
129
|
+
/**
|
|
130
|
+
* Controller types.
|
|
131
|
+
*
|
|
132
|
+
* Auto-derived in bundler plugin:
|
|
133
|
+
* - `encoder` field present → `["Encoder"]`
|
|
134
|
+
* - otherwise → `["Keypad"]`
|
|
135
|
+
*
|
|
136
|
+
* @default ["Keypad"]
|
|
137
|
+
*/
|
|
138
|
+
controllers?: [ManifestController, ManifestController?];
|
|
139
|
+
}
|
|
140
|
+
export interface ManifestActionSource {
|
|
141
|
+
uuid: string;
|
|
142
|
+
/** Presence indicates Keypad controller support. */
|
|
143
|
+
key?: unknown;
|
|
144
|
+
/** Presence indicates Encoder controller support. */
|
|
145
|
+
dial?: unknown;
|
|
146
|
+
/** Presence indicates Encoder controller support (touchstrip variant). */
|
|
147
|
+
touchStrip?: unknown;
|
|
148
|
+
/** Action manifest metadata. Required for manifest generation. */
|
|
149
|
+
info?: ActionManifestInfo;
|
|
150
|
+
}
|
|
151
|
+
export interface ManifestConfig extends PluginManifestInfo {
|
|
152
|
+
/**
|
|
153
|
+
* Action definitions from defineAction().
|
|
154
|
+
* Each action must have `info` populated with at least `name` and `icon`.
|
|
155
|
+
* Controllers are auto-derived from key/dial/touchStrip presence.
|
|
156
|
+
*/
|
|
157
|
+
actions: ManifestActionSource[];
|
|
158
|
+
}
|
|
159
|
+
export interface PluginManifestInfo {
|
|
160
|
+
/**
|
|
161
|
+
* Unique plugin identifier in reverse-DNS format.
|
|
162
|
+
*
|
|
163
|
+
* All action UUIDs must be prefixed with this value.
|
|
164
|
+
*/
|
|
165
|
+
uuid: string;
|
|
166
|
+
/** Plugin display name. */
|
|
167
|
+
name: string;
|
|
168
|
+
/** Author name shown on Marketplace. */
|
|
169
|
+
author: string;
|
|
170
|
+
/** Plugin description. */
|
|
171
|
+
description: string;
|
|
172
|
+
/** Path to plugin icon (extension omitted). 256×256 and 512×512 @2x. */
|
|
173
|
+
icon: string;
|
|
174
|
+
/**
|
|
175
|
+
* Plugin version.
|
|
176
|
+
* @example "1.0.0.0"
|
|
177
|
+
*/
|
|
178
|
+
version: string;
|
|
179
|
+
/**
|
|
180
|
+
* Actions list group name.
|
|
181
|
+
* @default Same as `name`
|
|
182
|
+
*/
|
|
183
|
+
category?: string;
|
|
184
|
+
/**
|
|
185
|
+
* Category icon path (extension omitted).
|
|
186
|
+
* @default Same as `icon`
|
|
187
|
+
*/
|
|
188
|
+
categoryIcon?: string;
|
|
189
|
+
/** Plugin website URL. */
|
|
190
|
+
url?: string;
|
|
191
|
+
/** Support website URL. */
|
|
192
|
+
supportUrl?: string;
|
|
193
|
+
/** Global property inspector HTML path. */
|
|
194
|
+
propertyInspectorPath?: string;
|
|
195
|
+
/** Pre-defined profiles distributed with the plugin. */
|
|
196
|
+
profiles?: ManifestProfileInfo[];
|
|
197
|
+
/** Applications to monitor on Mac/Windows. */
|
|
198
|
+
applicationsToMonitor?: {
|
|
199
|
+
mac?: string[];
|
|
200
|
+
windows?: string[];
|
|
201
|
+
};
|
|
202
|
+
/**
|
|
203
|
+
* Default window size for `window.open()` from the property inspector.
|
|
204
|
+
* @example [500, 650]
|
|
205
|
+
*/
|
|
206
|
+
defaultWindowSize?: [number, number];
|
|
207
|
+
/** macOS-specific entry point override. */
|
|
208
|
+
codePathMac?: string;
|
|
209
|
+
/** Windows-specific entry point override. */
|
|
210
|
+
codePathWin?: string;
|
|
211
|
+
/**
|
|
212
|
+
* Plugin entry point path.
|
|
213
|
+
* @default Derived from bundler output path (e.g. "bin/plugin.mjs")
|
|
214
|
+
*/
|
|
215
|
+
codePath?: string;
|
|
216
|
+
/**
|
|
217
|
+
* Operating system requirements.
|
|
218
|
+
* @default [{ platform: "mac", minimumVersion: "13" }, { platform: "windows", minimumVersion: "10" }]
|
|
219
|
+
*/
|
|
220
|
+
os?: [ManifestOSInfo, ManifestOSInfo?];
|
|
221
|
+
/**
|
|
222
|
+
* Node.js configuration.
|
|
223
|
+
* @default { version: "24" }
|
|
224
|
+
*/
|
|
225
|
+
nodejs?: ManifestNodejsInfo;
|
|
226
|
+
/**
|
|
227
|
+
* SDK version.
|
|
228
|
+
* @default 2
|
|
229
|
+
*/
|
|
230
|
+
sdkVersion?: 2 | 3;
|
|
231
|
+
/**
|
|
232
|
+
* Stream Deck software requirements.
|
|
233
|
+
* @default { minimumVersion: "7.1" }
|
|
234
|
+
*/
|
|
235
|
+
software?: {
|
|
236
|
+
minimumVersion: string;
|
|
237
|
+
};
|
|
238
|
+
}
|
package/dist/plugin.js
CHANGED
|
@@ -7,46 +7,55 @@ import { Renderer } from "@takumi-rs/core";
|
|
|
7
7
|
//#region src/plugin.ts
|
|
8
8
|
function createPlugin(config) {
|
|
9
9
|
const adapter = config.adapter ?? physicalDevice();
|
|
10
|
-
const renderer = new Renderer({ fonts: config.fonts.map((f) => ({
|
|
11
|
-
name: f.name,
|
|
12
|
-
data: f.data,
|
|
13
|
-
weight: f.weight,
|
|
14
|
-
style: f.style
|
|
15
|
-
})) });
|
|
16
|
-
const renderPool = config.useWorker !== false ? new RenderPool(config.fonts) : null;
|
|
17
|
-
const renderConfig = {
|
|
18
|
-
renderer,
|
|
19
|
-
imageFormat: config.imageFormat ?? "png",
|
|
20
|
-
caching: config.caching ?? true,
|
|
21
|
-
devicePixelRatio: config.devicePixelRatio ?? 1,
|
|
22
|
-
debug: config.debug ?? process.env.NODE_ENV !== "production",
|
|
23
|
-
imageCacheMaxBytes: config.imageCacheMaxBytes ?? 16 * 1024 * 1024,
|
|
24
|
-
touchStripCacheMaxBytes: config.touchStripCacheMaxBytes ?? 8 * 1024 * 1024,
|
|
25
|
-
renderPool
|
|
26
|
-
};
|
|
27
|
-
const registry = new RootRegistry(renderConfig, adapter, async (settings) => {
|
|
28
|
-
await adapter.setGlobalSettings(settings);
|
|
29
|
-
}, config.wrapper);
|
|
30
|
-
adapter.getGlobalSettings().then((gs) => {
|
|
31
|
-
registry.setGlobalSettings(gs);
|
|
32
|
-
}).catch((err) => {
|
|
33
|
-
console.error("[@fcannizzaro/streamdeck-react] Failed to load global settings:", err);
|
|
34
|
-
});
|
|
35
|
-
adapter.onGlobalSettingsChanged((settings) => {
|
|
36
|
-
registry.setGlobalSettings(settings);
|
|
37
|
-
});
|
|
38
|
-
for (const definition of config.actions) registerActionWithAdapter(adapter, definition, registry, config.onActionError);
|
|
39
|
-
if (renderConfig.debug) metrics.enable();
|
|
40
|
-
if (config.devtools) startDevtoolsServer({
|
|
41
|
-
devtoolsName: adapter.pluginUUID,
|
|
42
|
-
registry,
|
|
43
|
-
renderConfig
|
|
44
|
-
});
|
|
45
10
|
return { async connect() {
|
|
11
|
+
const takumiMode = config.takumi ?? "native-binding";
|
|
12
|
+
const renderPool = (takumiMode === "wasm" ? false : config.useWorker !== false) ? new RenderPool(config.fonts) : null;
|
|
13
|
+
const renderConfig = {
|
|
14
|
+
renderer: await initializeRenderer(takumiMode, config.fonts),
|
|
15
|
+
imageFormat: config.imageFormat ?? "png",
|
|
16
|
+
caching: config.caching ?? true,
|
|
17
|
+
devicePixelRatio: config.devicePixelRatio ?? 1,
|
|
18
|
+
debug: config.debug ?? process.env.NODE_ENV !== "production",
|
|
19
|
+
imageCacheMaxBytes: config.imageCacheMaxBytes ?? 16 * 1024 * 1024,
|
|
20
|
+
touchStripCacheMaxBytes: config.touchStripCacheMaxBytes ?? 8 * 1024 * 1024,
|
|
21
|
+
renderPool
|
|
22
|
+
};
|
|
23
|
+
const registry = new RootRegistry(renderConfig, adapter, async (settings) => {
|
|
24
|
+
await adapter.setGlobalSettings(settings);
|
|
25
|
+
}, config.wrapper);
|
|
26
|
+
adapter.getGlobalSettings().then((gs) => {
|
|
27
|
+
registry.setGlobalSettings(gs);
|
|
28
|
+
}).catch((err) => {
|
|
29
|
+
console.error("[@fcannizzaro/streamdeck-react] Failed to load global settings:", err);
|
|
30
|
+
});
|
|
31
|
+
adapter.onGlobalSettingsChanged((settings) => {
|
|
32
|
+
registry.setGlobalSettings(settings);
|
|
33
|
+
});
|
|
34
|
+
for (const definition of config.actions) registerActionWithAdapter(adapter, definition, registry, config.onActionError);
|
|
35
|
+
if (renderConfig.debug) metrics.enable();
|
|
36
|
+
if (config.devtools) startDevtoolsServer({
|
|
37
|
+
devtoolsName: adapter.pluginUUID,
|
|
38
|
+
registry,
|
|
39
|
+
renderConfig
|
|
40
|
+
});
|
|
46
41
|
if (renderPool != null) renderPool.initialize().catch(() => {});
|
|
47
42
|
await adapter.connect();
|
|
48
43
|
} };
|
|
49
44
|
}
|
|
45
|
+
async function initializeRenderer(mode, fonts) {
|
|
46
|
+
const fontData = fonts.map((f) => ({
|
|
47
|
+
name: f.name,
|
|
48
|
+
data: f.data,
|
|
49
|
+
weight: f.weight,
|
|
50
|
+
style: f.style
|
|
51
|
+
}));
|
|
52
|
+
if (mode === "wasm") {
|
|
53
|
+
const wasm = await import("@takumi-rs/wasm");
|
|
54
|
+
await wasm.default();
|
|
55
|
+
return new wasm.Renderer({ fonts: fontData });
|
|
56
|
+
}
|
|
57
|
+
return new Renderer({ fonts: fontData });
|
|
58
|
+
}
|
|
50
59
|
function registerActionWithAdapter(adapter, definition, registry, onError) {
|
|
51
60
|
const handleError = (actionId, err) => {
|
|
52
61
|
const error = err instanceof Error ? err : new Error(String(err));
|
|
@@ -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
        // Cast: TakumiNode is a local structural duplicate of the
        // @takumi-rs/helpers Node union — the shapes match at runtime
        // but TypeScript can't verify the discriminated union members
        // from the loose `type: string` index signature.
        const buffer = await renderer.render(
          rootNode as unknown as import("@takumi-rs/helpers").Node,
          {
            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,
|