appshot-cli 1.0.2 → 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (151) hide show
  1. package/README.md +300 -559
  2. package/dist/cli.js +5 -3
  3. package/dist/cli.js.map +1 -1
  4. package/dist/commands/backgrounds.d.ts.map +1 -1
  5. package/dist/commands/backgrounds.js +164 -9
  6. package/dist/commands/backgrounds.js.map +1 -1
  7. package/dist/commands/build.d.ts.map +1 -1
  8. package/dist/commands/build.js +154 -54
  9. package/dist/commands/build.js.map +1 -1
  10. package/dist/commands/caption.d.ts.map +1 -1
  11. package/dist/commands/caption.js +350 -6
  12. package/dist/commands/caption.js.map +1 -1
  13. package/dist/commands/check.d.ts.map +1 -1
  14. package/dist/commands/check.js +174 -97
  15. package/dist/commands/check.js.map +1 -1
  16. package/dist/commands/clean.d.ts.map +1 -1
  17. package/dist/commands/clean.js +16 -0
  18. package/dist/commands/clean.js.map +1 -1
  19. package/dist/commands/fonts.d.ts.map +1 -1
  20. package/dist/commands/fonts.js +44 -26
  21. package/dist/commands/fonts.js.map +1 -1
  22. package/dist/commands/gradients.d.ts.map +1 -1
  23. package/dist/commands/gradients.js +32 -15
  24. package/dist/commands/gradients.js.map +1 -1
  25. package/dist/commands/init.d.ts.map +1 -1
  26. package/dist/commands/init.js +18 -16
  27. package/dist/commands/init.js.map +1 -1
  28. package/dist/commands/migrate.d.ts.map +1 -1
  29. package/dist/commands/migrate.js +51 -104
  30. package/dist/commands/migrate.js.map +1 -1
  31. package/dist/commands/preset.d.ts.map +1 -1
  32. package/dist/commands/preset.js +16 -14
  33. package/dist/commands/preset.js.map +1 -1
  34. package/dist/commands/presets.d.ts.map +1 -1
  35. package/dist/commands/presets.js +15 -0
  36. package/dist/commands/presets.js.map +1 -1
  37. package/dist/commands/quickstart.d.ts.map +1 -1
  38. package/dist/commands/quickstart.js +40 -53
  39. package/dist/commands/quickstart.js.map +1 -1
  40. package/dist/commands/style.d.ts.map +1 -1
  41. package/dist/commands/style.js +101 -57
  42. package/dist/commands/style.js.map +1 -1
  43. package/dist/commands/template.d.ts.map +1 -1
  44. package/dist/commands/template.js +68 -80
  45. package/dist/commands/template.js.map +1 -1
  46. package/dist/commands/validate.d.ts.map +1 -1
  47. package/dist/commands/validate.js +141 -0
  48. package/dist/commands/validate.js.map +1 -1
  49. package/dist/commands/wizard.d.ts +25 -0
  50. package/dist/commands/wizard.d.ts.map +1 -0
  51. package/dist/commands/wizard.js +366 -0
  52. package/dist/commands/wizard.js.map +1 -0
  53. package/dist/core/compose.d.ts +60 -1
  54. package/dist/core/compose.d.ts.map +1 -1
  55. package/dist/core/compose.js +150 -0
  56. package/dist/core/compose.js.map +1 -1
  57. package/dist/core/device-strategies/index.d.ts +8 -0
  58. package/dist/core/device-strategies/index.d.ts.map +1 -0
  59. package/dist/core/device-strategies/index.js +22 -0
  60. package/dist/core/device-strategies/index.js.map +1 -0
  61. package/dist/core/device-strategies/ipad.d.ts +3 -0
  62. package/dist/core/device-strategies/ipad.d.ts.map +1 -0
  63. package/dist/core/device-strategies/ipad.js +13 -0
  64. package/dist/core/device-strategies/ipad.js.map +1 -0
  65. package/dist/core/device-strategies/iphone.d.ts +3 -0
  66. package/dist/core/device-strategies/iphone.d.ts.map +1 -0
  67. package/dist/core/device-strategies/iphone.js +13 -0
  68. package/dist/core/device-strategies/iphone.js.map +1 -0
  69. package/dist/core/device-strategies/mac.d.ts +3 -0
  70. package/dist/core/device-strategies/mac.d.ts.map +1 -0
  71. package/dist/core/device-strategies/mac.js +13 -0
  72. package/dist/core/device-strategies/mac.js.map +1 -0
  73. package/dist/core/device-strategies/watch.d.ts +3 -0
  74. package/dist/core/device-strategies/watch.d.ts.map +1 -0
  75. package/dist/core/device-strategies/watch.js +13 -0
  76. package/dist/core/device-strategies/watch.js.map +1 -0
  77. package/dist/core/files.d.ts +3 -3
  78. package/dist/core/files.d.ts.map +1 -1
  79. package/dist/core/files.js.map +1 -1
  80. package/dist/core/layouts/device-fit.d.ts +10 -0
  81. package/dist/core/layouts/device-fit.d.ts.map +1 -0
  82. package/dist/core/layouts/device-fit.js +18 -0
  83. package/dist/core/layouts/device-fit.js.map +1 -0
  84. package/dist/core/layouts/index.d.ts +4 -0
  85. package/dist/core/layouts/index.d.ts.map +1 -0
  86. package/dist/core/layouts/index.js +4 -0
  87. package/dist/core/layouts/index.js.map +1 -0
  88. package/dist/core/layouts/math.d.ts +23 -0
  89. package/dist/core/layouts/math.d.ts.map +1 -0
  90. package/dist/core/layouts/math.js +63 -0
  91. package/dist/core/layouts/math.js.map +1 -0
  92. package/dist/core/layouts/text-layout.d.ts +10 -0
  93. package/dist/core/layouts/text-layout.d.ts.map +1 -0
  94. package/dist/core/layouts/text-layout.js +60 -0
  95. package/dist/core/layouts/text-layout.js.map +1 -0
  96. package/dist/mcp/cli-options.d.ts +15 -0
  97. package/dist/mcp/cli-options.d.ts.map +1 -1
  98. package/dist/mcp/cli-options.js +16 -0
  99. package/dist/mcp/cli-options.js.map +1 -1
  100. package/dist/mcp/server.d.ts.map +1 -1
  101. package/dist/mcp/server.js +268 -52
  102. package/dist/mcp/server.js.map +1 -1
  103. package/dist/services/caption-enhancement.d.ts +32 -0
  104. package/dist/services/caption-enhancement.d.ts.map +1 -0
  105. package/dist/services/caption-enhancement.js +156 -0
  106. package/dist/services/caption-enhancement.js.map +1 -0
  107. package/dist/services/compose-bridge.d.ts +3 -3
  108. package/dist/services/compose-bridge.d.ts.map +1 -1
  109. package/dist/services/compose-bridge.js +64 -21
  110. package/dist/services/compose-bridge.js.map +1 -1
  111. package/dist/services/doctor.d.ts.map +1 -1
  112. package/dist/services/doctor.js +13 -2
  113. package/dist/services/doctor.js.map +1 -1
  114. package/dist/services/translation.d.ts +8 -0
  115. package/dist/services/translation.d.ts.map +1 -1
  116. package/dist/services/translation.js +117 -29
  117. package/dist/services/translation.js.map +1 -1
  118. package/dist/services/watch-service.d.ts.map +1 -1
  119. package/dist/services/watch-service.js.map +1 -1
  120. package/dist/templates/registry.d.ts +18 -59
  121. package/dist/templates/registry.d.ts.map +1 -1
  122. package/dist/templates/registry.js +138 -500
  123. package/dist/templates/registry.js.map +1 -1
  124. package/dist/types.d.ts +35 -0
  125. package/dist/types.d.ts.map +1 -1
  126. package/dist/utils/config-migration.d.ts +7 -0
  127. package/dist/utils/config-migration.d.ts.map +1 -0
  128. package/dist/utils/config-migration.js +58 -0
  129. package/dist/utils/config-migration.js.map +1 -0
  130. package/dist/utils/config-version.d.ts +5 -0
  131. package/dist/utils/config-version.d.ts.map +1 -0
  132. package/dist/utils/config-version.js +10 -0
  133. package/dist/utils/config-version.js.map +1 -0
  134. package/dist/utils/language.d.ts +2 -2
  135. package/dist/utils/language.d.ts.map +1 -1
  136. package/dist/utils/language.js +1 -1
  137. package/dist/utils/language.js.map +1 -1
  138. package/dist/utils/spinner.d.ts +21 -0
  139. package/dist/utils/spinner.d.ts.map +1 -0
  140. package/dist/utils/spinner.js +60 -0
  141. package/dist/utils/spinner.js.map +1 -0
  142. package/dist/utils/v2-banner.d.ts +2 -0
  143. package/dist/utils/v2-banner.d.ts.map +1 -0
  144. package/dist/utils/v2-banner.js +14 -0
  145. package/dist/utils/v2-banner.js.map +1 -0
  146. package/dist/utils/validation.d.ts.map +1 -1
  147. package/dist/utils/validation.js +4 -3
  148. package/dist/utils/validation.js.map +1 -1
  149. package/package.json +1 -1
  150. package/skill/SKILL.md +5 -1
  151. package/skill/references/templates.md +26 -0
@@ -0,0 +1,366 @@
1
+ import { Command } from 'commander';
2
+ import pc from 'picocolors';
3
+ import inquirer from 'inquirer';
4
+ import { spawn } from 'child_process';
5
+ import path from 'path';
6
+ import { fileURLToPath } from 'url';
7
+ import { promises as fs } from 'fs';
8
+ import { loadConfig, saveConfig, fileExists } from '../core/files.js';
9
+ import { detectConfigVersion } from '../utils/config-version.js';
10
+ import { showV1DeprecationBanner } from '../utils/v2-banner.js';
11
+ import { translationService } from '../services/translation.js';
12
+ import { applyTemplateToConfig, getTemplate, resolveTemplateId, templates } from '../templates/registry.js';
13
+ import { getImageDimensions } from '../core/devices.js';
14
+ const DEFAULT_LANGS = 'en,es,fr';
15
+ const DEFAULT_MODEL = 'gpt-5-mini';
16
+ export default function wizardCmd() {
17
+ const cmd = new Command('wizard')
18
+ .description('One-shot v2 setup and build flow')
19
+ .option('--devices <list>', 'comma-separated device list (iphone,ipad,mac,watch)')
20
+ .option('--layout <mode>', 'layout mode (header, footer, screenshot-only)')
21
+ .option('--template <id>', 'apply a v2 template (ocean-header, sunset-footer, clean-screenshot, etc.)')
22
+ .option('--caption-source <mode>', 'filenames or manual')
23
+ .option('--langs <codes>', 'language codes (comma-separated)')
24
+ .option('--model <name>', 'OpenAI model to use', DEFAULT_MODEL)
25
+ .option('--enhance', 'enhance captions after translation')
26
+ .option('--no-enhance', 'skip caption enhancement')
27
+ .option('--no-interactive', 'run without prompts (use flags)')
28
+ .option('--dry-run', 'print actions without executing')
29
+ .option('--migrate', 'auto-migrate v1 config to v2')
30
+ .option('--require-ai', 'fail if AI steps are requested but no API key is set')
31
+ .addHelpText('after', `
32
+ ${pc.bold('Examples:')}
33
+ ${pc.dim('# Interactive wizard')}
34
+ $ appshot wizard
35
+
36
+ ${pc.dim('# Interactive with template')}
37
+ $ appshot wizard --template ocean-header
38
+
39
+ ${pc.dim('# Non-interactive (CI)')}
40
+ $ appshot wizard --no-interactive --devices iphone --template ocean-header --caption-source filenames --langs en,es,fr --model gpt-5-mini
41
+
42
+ ${pc.bold('Notes:')}
43
+ • If OPENAI_API_KEY is missing, translation/enhance steps are skipped unless --require-ai is set.
44
+ • Manual captions require interactive entry per file.`)
45
+ .action(async (opts) => {
46
+ try {
47
+ await runWizard(opts);
48
+ }
49
+ catch (error) {
50
+ console.error(pc.red('Error:'), error instanceof Error ? error.message : String(error));
51
+ process.exit(1);
52
+ }
53
+ });
54
+ return cmd;
55
+ }
56
+ export async function runWizard(options, deps = {}) {
57
+ const runner = deps.runner ?? runCli;
58
+ const interactive = !options.noInteractive;
59
+ const plan = {
60
+ devices: [],
61
+ layout: options.layout ?? 'header',
62
+ templateId: options.template,
63
+ captionSource: options.captionSource ?? 'filenames',
64
+ langs: options.langs ?? DEFAULT_LANGS,
65
+ model: options.model ?? DEFAULT_MODEL,
66
+ enhance: options.noEnhance ? false : (options.enhance ?? true)
67
+ };
68
+ if (interactive) {
69
+ const answers = await inquirer.prompt([
70
+ {
71
+ type: 'checkbox',
72
+ name: 'devices',
73
+ message: 'Which devices?',
74
+ choices: [
75
+ { name: 'iPhone', value: 'iphone', checked: true },
76
+ { name: 'iPad', value: 'ipad' },
77
+ { name: 'Mac', value: 'mac' },
78
+ { name: 'Watch', value: 'watch' }
79
+ ],
80
+ validate: (values) => (values.length > 0 ? true : 'Select at least one device')
81
+ },
82
+ {
83
+ type: 'list',
84
+ name: 'templateId',
85
+ message: 'Use a template preset?',
86
+ choices: [
87
+ { name: 'No template (custom)', value: '' },
88
+ ...templates.map(template => ({
89
+ name: `${template.name} — ${template.description}`,
90
+ value: template.id
91
+ }))
92
+ ],
93
+ default: plan.templateId || ''
94
+ },
95
+ {
96
+ type: 'list',
97
+ name: 'layout',
98
+ message: 'Layout mode:',
99
+ choices: [
100
+ { name: 'Header (caption on top)', value: 'header' },
101
+ { name: 'Footer (caption on bottom)', value: 'footer' },
102
+ { name: 'Screenshot-only', value: 'screenshot-only' }
103
+ ],
104
+ default: plan.layout,
105
+ when: (answers) => !answers.templateId
106
+ },
107
+ {
108
+ type: 'list',
109
+ name: 'captionSource',
110
+ message: 'Caption source:',
111
+ choices: [
112
+ { name: 'Filenames (auto)', value: 'filenames' },
113
+ { name: 'Manual entry', value: 'manual' }
114
+ ],
115
+ default: plan.captionSource
116
+ },
117
+ {
118
+ type: 'confirm',
119
+ name: 'localize',
120
+ message: 'Localize captions to other languages?',
121
+ default: true
122
+ },
123
+ {
124
+ type: 'checkbox',
125
+ name: 'langs',
126
+ message: 'Select languages:',
127
+ choices: translationService.getSupportedLanguages().map(lang => ({
128
+ name: `${lang.code} — ${lang.name}`,
129
+ value: lang.code,
130
+ checked: DEFAULT_LANGS.split(',').includes(lang.code)
131
+ })),
132
+ when: (answers) => answers.localize,
133
+ validate: (values) => {
134
+ const normalized = (values || []).map(value => {
135
+ if (typeof value === 'string')
136
+ return value.toLowerCase();
137
+ if (value && typeof value === 'object' && 'value' in value) {
138
+ return String(value.value ?? '').toLowerCase();
139
+ }
140
+ return String(value ?? '').toLowerCase();
141
+ }).filter(Boolean);
142
+ if (normalized.length === 0) {
143
+ return 'Select at least one language';
144
+ }
145
+ if (!normalized.some(code => code === 'en' || code.startsWith('en-'))) {
146
+ return 'English (en) is required as the base language';
147
+ }
148
+ return true;
149
+ }
150
+ },
151
+ {
152
+ type: 'confirm',
153
+ name: 'enhance',
154
+ message: 'Enhance captions (rewrite to be shorter/clearer and fit layout limits)?',
155
+ default: plan.enhance
156
+ },
157
+ {
158
+ type: 'input',
159
+ name: 'model',
160
+ message: 'OpenAI model:',
161
+ default: plan.model,
162
+ when: (answers) => Boolean(answers.localize) || Boolean(answers.enhance)
163
+ }
164
+ ]);
165
+ plan.devices = answers.devices;
166
+ plan.templateId = answers.templateId || undefined;
167
+ plan.layout = answers.layout || plan.layout;
168
+ plan.captionSource = answers.captionSource;
169
+ plan.langs = answers.localize ? answers.langs.join(',') : 'en';
170
+ plan.enhance = answers.enhance;
171
+ plan.model = answers.model || plan.model;
172
+ }
173
+ else {
174
+ plan.devices = options.devices ? options.devices.split(',').map(d => d.trim()).filter(Boolean) : ['iphone'];
175
+ }
176
+ if (plan.devices.length === 0) {
177
+ throw new Error('No devices selected.');
178
+ }
179
+ if (plan.templateId) {
180
+ const resolved = resolveTemplateId(plan.templateId);
181
+ const template = getTemplate(resolved.id);
182
+ if (!template) {
183
+ throw new Error(`Template "${plan.templateId}" not found.`);
184
+ }
185
+ plan.templateId = resolved.id;
186
+ plan.layout = template.layout;
187
+ if (options.layout) {
188
+ console.log(pc.yellow('⚠'), `Template "${template.name}" sets layout to "${template.layout}" (ignoring --layout).`);
189
+ }
190
+ }
191
+ const needsAi = plan.langs.split(',').length > 1 || plan.enhance;
192
+ const hasApiKey = !!process.env.OPENAI_API_KEY;
193
+ if (needsAi && !hasApiKey) {
194
+ if (options.requireAi) {
195
+ throw new Error('OPENAI_API_KEY is required for translation/enhancement.');
196
+ }
197
+ console.log(pc.yellow('⚠'), 'OPENAI_API_KEY not found. Skipping translation/enhancement.');
198
+ plan.langs = 'en';
199
+ plan.enhance = false;
200
+ }
201
+ const configExists = await fileExists(path.join(process.cwd(), '.appshot', 'config.json'));
202
+ if (!configExists) {
203
+ await maybeRun(runner, ['init'], options.dryRun);
204
+ }
205
+ let config;
206
+ try {
207
+ config = await loadConfig();
208
+ }
209
+ catch {
210
+ throw new Error('Could not load config. Run "appshot init" first.');
211
+ }
212
+ const version = detectConfigVersion(config);
213
+ if (version === 1) {
214
+ if (interactive && !options.migrate) {
215
+ showV1DeprecationBanner();
216
+ const confirm = await inquirer.prompt([{
217
+ type: 'confirm',
218
+ name: 'migrate',
219
+ message: 'Migrate v1 config to v2 now?',
220
+ default: true
221
+ }]);
222
+ if (!confirm.migrate) {
223
+ throw new Error('Wizard canceled. Run "appshot migrate" to upgrade.');
224
+ }
225
+ }
226
+ else if (!options.migrate) {
227
+ throw new Error('v1 config detected. Run "appshot migrate" or pass --migrate.');
228
+ }
229
+ await maybeRun(runner, ['migrate', '--yes'], options.dryRun);
230
+ config = await loadConfig();
231
+ }
232
+ if (config.version === 2) {
233
+ let nextConfig = config;
234
+ if (plan.templateId) {
235
+ nextConfig = applyTemplateToConfig(plan.templateId, nextConfig);
236
+ }
237
+ else {
238
+ nextConfig.layout = plan.layout;
239
+ }
240
+ await applyDetectedResolutions(nextConfig, plan.devices);
241
+ await saveConfig(nextConfig);
242
+ }
243
+ const langs = plan.langs;
244
+ const langList = langs.split(',').map(l => l.trim()).filter(Boolean);
245
+ const targetLangs = langList.filter(l => l !== 'en');
246
+ for (const device of plan.devices) {
247
+ if (plan.captionSource === 'filenames') {
248
+ const args = ['caption', '--device', device, '--auto-caption'];
249
+ if (targetLangs.length > 0) {
250
+ args.push('--translate', '--langs', targetLangs.join(','), '--model', plan.model);
251
+ }
252
+ await maybeRun(runner, args, options.dryRun);
253
+ }
254
+ else {
255
+ const args = ['caption', '--device', device];
256
+ if (targetLangs.length > 0) {
257
+ args.push('--translate', '--langs', targetLangs.join(','), '--model', plan.model);
258
+ }
259
+ await maybeRun(runner, args, options.dryRun);
260
+ }
261
+ if (plan.enhance) {
262
+ const enhanceArgs = ['caption', 'enhance', '--device', device, '--model', plan.model];
263
+ if (langList.length > 0) {
264
+ enhanceArgs.push('--langs', langList.join(','));
265
+ }
266
+ await maybeRun(runner, enhanceArgs, options.dryRun);
267
+ }
268
+ }
269
+ const buildArgs = ['build', '--devices', plan.devices.join(',')];
270
+ if (langList.length > 0) {
271
+ buildArgs.push('--langs', langList.join(','));
272
+ }
273
+ await maybeRun(runner, buildArgs, options.dryRun);
274
+ }
275
+ async function applyDetectedResolutions(config, devices) {
276
+ const targetDevices = devices.length > 0 ? devices : Object.keys(config.devices);
277
+ for (const device of targetDevices) {
278
+ const entry = config.devices[device];
279
+ if (!entry)
280
+ continue;
281
+ const input = typeof entry === 'string' ? entry : entry.input;
282
+ const inputDir = path.resolve(process.cwd(), input);
283
+ try {
284
+ const files = (await fs.readdir(inputDir))
285
+ .filter(f => f.match(/\.(png|jpg|jpeg)$/i))
286
+ .sort();
287
+ if (files.length === 0) {
288
+ if (typeof entry === 'object' && 'resolution' in entry) {
289
+ delete entry.resolution;
290
+ }
291
+ continue;
292
+ }
293
+ const sample = path.join(inputDir, files[0]);
294
+ const { width, height } = await getImageDimensions(sample);
295
+ if (!width || !height) {
296
+ continue;
297
+ }
298
+ const resolution = `${width}x${height}`;
299
+ if (typeof entry === 'string') {
300
+ config.devices[device] = { input, resolution };
301
+ }
302
+ else {
303
+ entry.resolution = resolution;
304
+ }
305
+ console.log(pc.dim(`Detected ${device} resolution: ${resolution}`));
306
+ }
307
+ catch {
308
+ if (typeof entry === 'object' && 'resolution' in entry) {
309
+ delete entry.resolution;
310
+ }
311
+ }
312
+ }
313
+ }
314
+ async function maybeRun(runner, args, dryRun) {
315
+ if (dryRun) {
316
+ console.log(pc.dim(`• Would run: appshot ${args.join(' ')}`));
317
+ return;
318
+ }
319
+ await runner(args);
320
+ }
321
+ function resolveCliEntry() {
322
+ const currentFile = fileURLToPath(import.meta.url);
323
+ const dir = path.dirname(currentFile);
324
+ const pathParts = dir.split(path.sep);
325
+ const isDevMode = pathParts.includes('src');
326
+ if (isDevMode) {
327
+ return path.resolve(dir, '../cli.ts');
328
+ }
329
+ return path.resolve(dir, '../cli.js');
330
+ }
331
+ function getCliCommand() {
332
+ const entry = resolveCliEntry();
333
+ if (entry.endsWith('.ts')) {
334
+ return {
335
+ execPath: process.execPath,
336
+ args: [path.resolve(path.dirname(entry), '../node_modules/.bin/tsx'), entry]
337
+ };
338
+ }
339
+ return {
340
+ execPath: process.execPath,
341
+ args: [entry]
342
+ };
343
+ }
344
+ function runCli(args, opts) {
345
+ return new Promise((resolve, reject) => {
346
+ const cli = getCliCommand();
347
+ const child = spawn(cli.execPath, [...cli.args, ...args], {
348
+ cwd: opts?.cwd ?? process.cwd(),
349
+ stdio: 'inherit',
350
+ env: {
351
+ ...process.env,
352
+ FORCE_COLOR: '1'
353
+ }
354
+ });
355
+ child.once('error', (error) => reject(error));
356
+ child.once('close', (code) => {
357
+ if (code === 0) {
358
+ resolve();
359
+ }
360
+ else {
361
+ reject(new Error(`Command failed: appshot ${args.join(' ')}`));
362
+ }
363
+ });
364
+ });
365
+ }
366
+ //# sourceMappingURL=wizard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wizard.js","sourceRoot":"","sources":["../../src/commands/wizard.ts"],"names":[],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import type { GradientConfig, CaptionConfig, DeviceConfig, BackgroundConfig } from '../types.js';
1
+ import type { GradientConfig, CaptionConfig, DeviceConfig, BackgroundConfig, CaptionConfigV2, LayoutModeV2, DeviceStrategyV2 } from '../types.js';
2
2
  export interface LayoutDebugInfo {
3
3
  mode: 'above' | 'below' | 'overlay';
4
4
  framePosition: string | number | undefined;
@@ -45,6 +45,65 @@ export interface ComposeOptions {
45
45
  * Compose a complete App Store screenshot with gradient, caption, and framed device
46
46
  */
47
47
  export declare function composeAppStoreScreenshot(options: ComposeOptions): Promise<Buffer>;
48
+ export interface LayoutV2DebugInfo {
49
+ layout: LayoutModeV2;
50
+ usable: {
51
+ x: number;
52
+ y: number;
53
+ width: number;
54
+ height: number;
55
+ };
56
+ caption?: {
57
+ x: number;
58
+ y: number;
59
+ width: number;
60
+ height: number;
61
+ };
62
+ device: {
63
+ x: number;
64
+ y: number;
65
+ width: number;
66
+ height: number;
67
+ };
68
+ deviceFit?: {
69
+ x: number;
70
+ y: number;
71
+ width: number;
72
+ height: number;
73
+ scale: number;
74
+ };
75
+ fontSize?: number;
76
+ captionTruncated?: boolean;
77
+ }
78
+ export interface ComposeV2Options {
79
+ screenshot: Buffer;
80
+ frame?: Buffer | null;
81
+ frameMetadata?: {
82
+ frameWidth: number;
83
+ frameHeight: number;
84
+ screenRect: {
85
+ x: number;
86
+ y: number;
87
+ width: number;
88
+ height: number;
89
+ };
90
+ maskPath?: string;
91
+ deviceType?: 'iphone' | 'ipad' | 'mac' | 'watch';
92
+ displayName?: string;
93
+ name?: string;
94
+ };
95
+ caption?: string;
96
+ captionConfig: CaptionConfigV2;
97
+ backgroundConfig?: BackgroundConfig;
98
+ outputWidth: number;
99
+ outputHeight: number;
100
+ layout: LayoutModeV2;
101
+ deviceType: DeviceStrategyV2['deviceType'];
102
+ deviceInputPath?: string;
103
+ verbose?: boolean;
104
+ onDebug?: (info: LayoutV2DebugInfo) => void;
105
+ }
106
+ export declare function composeV2(options: ComposeV2Options): Promise<Buffer>;
48
107
  /**
49
108
  * Compose a framed device with a fully transparent background.
50
109
  * No gradient or captions are applied; output is frame-sized.
@@ -1 +1 @@
1
- {"version":3,"file":"compose.d.ts","sourceRoot":"","sources":["../../src/core/compose.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"compose.d.ts","sourceRoot":"","sources":["../../src/core/compose.ts"],"names":[],"mappings":"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"}
@@ -7,6 +7,10 @@ import { applyRoundedCorners } from './mask-generator.js';
7
7
  import { calculateAdaptiveCaptionHeight, wrapText } from './text-utils.js';
8
8
  import { FontService } from '../services/fonts.js';
9
9
  import { resolveLayoutSpacing } from './layout-utils.js';
10
+ import { computeFontSize, computeRegions } from './layouts/math.js';
11
+ import { fitBoxToRegion } from './layouts/device-fit.js';
12
+ import { layoutCaptionText } from './layouts/text-layout.js';
13
+ import { getDeviceStrategyV2 } from './device-strategies/index.js';
10
14
  /**
11
15
  * Parse font name to extract style and weight
12
16
  */
@@ -119,6 +123,34 @@ function generateCaptionSVG(lines, width, height, fontSize, fontFamily, fontStyl
119
123
  ${svgElements.join('\n ')}
120
124
  </svg>`;
121
125
  }
126
+ function generateCaptionSvgV2(options) {
127
+ const { lines, width, height, fontSize, fontFamily, fontStyle, fontWeight, textColor, lineHeight, background } = options;
128
+ const svgElements = [];
129
+ if (background?.color) {
130
+ const opacity = background.opacity !== undefined ? background.opacity : 0.8;
131
+ svgElements.push(`<rect x="0" y="0" width="${width}" height="${height}" ` +
132
+ `fill="${background.color}" opacity="${opacity}"/>`);
133
+ }
134
+ const padding = Math.round(height * 0.1);
135
+ const innerHeight = Math.max(0, height - padding * 2);
136
+ const totalTextHeight = lines.length * fontSize * lineHeight;
137
+ const startY = padding + Math.max(fontSize, (innerHeight - totalTextHeight) / 2 + fontSize);
138
+ const centerX = Math.floor(width / 2);
139
+ const textElements = lines.map((line, index) => {
140
+ const y = startY + (index * fontSize * lineHeight);
141
+ return `<text x="${centerX}" y="${y}" ` +
142
+ `font-family="${fontFamily}" ` +
143
+ `font-size="${fontSize}" ` +
144
+ `font-style="${fontStyle}" ` +
145
+ `font-weight="${fontWeight}" ` +
146
+ `fill="${textColor}" ` +
147
+ `text-anchor="middle">${escapeXml(line)}</text>`;
148
+ });
149
+ svgElements.push(...textElements);
150
+ return `<svg width="${width}" height="${height}" xmlns="http://www.w3.org/2000/svg">
151
+ ${svgElements.join('\n ')}
152
+ </svg>`;
153
+ }
122
154
  /**
123
155
  * Compose a complete App Store screenshot with gradient, caption, and framed device
124
156
  */
@@ -991,6 +1023,124 @@ export async function composeAppStoreScreenshot(options) {
991
1023
  .toBuffer();
992
1024
  return result;
993
1025
  }
1026
+ export async function composeV2(options) {
1027
+ const { screenshot, frame, frameMetadata, caption, captionConfig, backgroundConfig, outputWidth, outputHeight, layout, deviceType, deviceInputPath, verbose = false, onDebug } = options;
1028
+ const hasCaption = typeof caption === 'string' && caption.trim().length > 0;
1029
+ const effectiveLayout = hasCaption ? layout : 'screenshot-only';
1030
+ const strategy = getDeviceStrategyV2(deviceType);
1031
+ const regions = computeRegions({ canvasWidth: outputWidth, canvasHeight: outputHeight, layout: effectiveLayout, strategy });
1032
+ if (verbose) {
1033
+ console.log(pc.dim(' v2 layout:'));
1034
+ console.log(pc.dim(` Layout: ${effectiveLayout}`));
1035
+ if (!hasCaption && layout !== 'screenshot-only') {
1036
+ console.log(pc.dim(' Layout override: no caption → screenshot-only'));
1037
+ }
1038
+ console.log(pc.dim(` Usable: ${regions.usable.width}x${regions.usable.height} @ (${regions.usable.x}, ${regions.usable.y})`));
1039
+ if (regions.caption) {
1040
+ console.log(pc.dim(` Caption: ${regions.caption.width}x${regions.caption.height} @ (${regions.caption.x}, ${regions.caption.y})`));
1041
+ }
1042
+ console.log(pc.dim(` Device: ${regions.device.width}x${regions.device.height} @ (${regions.device.x}, ${regions.device.y})`));
1043
+ }
1044
+ let backgroundBuffer;
1045
+ if (backgroundConfig) {
1046
+ backgroundBuffer = await renderBackground(outputWidth, outputHeight, backgroundConfig, deviceInputPath || '');
1047
+ if (backgroundConfig.warnOnMismatch && backgroundConfig.image) {
1048
+ const validation = await validateBackgroundDimensions(backgroundConfig.image, outputWidth, outputHeight);
1049
+ if (validation.warnings.length > 0) {
1050
+ console.warn(pc.yellow('⚠️ Background dimension warnings:'));
1051
+ validation.warnings.forEach(warning => {
1052
+ console.warn(pc.dim(` ${warning}`));
1053
+ });
1054
+ }
1055
+ }
1056
+ }
1057
+ else {
1058
+ const defaultGradient = {
1059
+ colors: ['#4A90E2', '#7B68EE'],
1060
+ direction: 'top-bottom'
1061
+ };
1062
+ backgroundBuffer = await renderGradient(outputWidth, outputHeight, defaultGradient);
1063
+ }
1064
+ const composites = [];
1065
+ let deviceFit;
1066
+ if (frame && frameMetadata) {
1067
+ const framedBuffer = await composeFrameOnly({
1068
+ screenshot,
1069
+ frame,
1070
+ frameMetadata,
1071
+ outputFormat: 'png',
1072
+ verbose
1073
+ });
1074
+ deviceFit = fitBoxToRegion(frameMetadata.frameWidth, frameMetadata.frameHeight, regions.device);
1075
+ const resizedFrame = await sharp(framedBuffer)
1076
+ .resize(deviceFit.width, deviceFit.height, { fit: 'fill' })
1077
+ .toBuffer();
1078
+ composites.push({
1079
+ input: resizedFrame,
1080
+ left: deviceFit.x,
1081
+ top: deviceFit.y
1082
+ });
1083
+ }
1084
+ else {
1085
+ const metadata = await sharp(screenshot).metadata();
1086
+ const baseWidth = metadata.width || regions.device.width;
1087
+ const baseHeight = metadata.height || regions.device.height;
1088
+ deviceFit = fitBoxToRegion(baseWidth, baseHeight, regions.device);
1089
+ const resizedScreenshot = await sharp(screenshot)
1090
+ .resize(deviceFit.width, deviceFit.height, { fit: 'fill' })
1091
+ .toBuffer();
1092
+ composites.push({
1093
+ input: resizedScreenshot,
1094
+ left: deviceFit.x,
1095
+ top: deviceFit.y
1096
+ });
1097
+ }
1098
+ let captionTruncated = false;
1099
+ let fontSize;
1100
+ if (hasCaption && regions.caption && effectiveLayout !== 'screenshot-only') {
1101
+ const screenHeight = frameMetadata?.screenRect.height || regions.device.height;
1102
+ fontSize = computeFontSize(screenHeight, strategy);
1103
+ const textLayout = layoutCaptionText(caption, regions.caption, fontSize, strategy);
1104
+ captionTruncated = textLayout.truncated;
1105
+ const parsedFont = parseFontName(captionConfig.font);
1106
+ const fontFamily = getFontStack(parsedFont.family);
1107
+ const fontStyle = parsedFont.style || 'normal';
1108
+ const fontWeight = parsedFont.weight === 'bold' ? '700' : '400';
1109
+ const svgText = generateCaptionSvgV2({
1110
+ lines: textLayout.lines,
1111
+ width: regions.caption.width,
1112
+ height: regions.caption.height,
1113
+ fontSize,
1114
+ fontFamily,
1115
+ fontStyle,
1116
+ fontWeight,
1117
+ textColor: captionConfig.color,
1118
+ lineHeight: strategy.captionLineHeight,
1119
+ background: captionConfig.background
1120
+ });
1121
+ const captionImage = await sharp(Buffer.from(svgText)).png().toBuffer();
1122
+ composites.push({
1123
+ input: captionImage,
1124
+ left: regions.caption.x,
1125
+ top: regions.caption.y
1126
+ });
1127
+ if (captionTruncated) {
1128
+ console.warn(pc.yellow('⚠️ Caption truncated. Consider shortening the text.'));
1129
+ }
1130
+ }
1131
+ if (onDebug) {
1132
+ onDebug({
1133
+ layout: effectiveLayout,
1134
+ usable: regions.usable,
1135
+ caption: regions.caption,
1136
+ device: regions.device,
1137
+ deviceFit,
1138
+ fontSize,
1139
+ captionTruncated
1140
+ });
1141
+ }
1142
+ return sharp(backgroundBuffer).composite(composites).png().toBuffer();
1143
+ }
994
1144
  /**
995
1145
  * Compose a framed device with a fully transparent background.
996
1146
  * No gradient or captions are applied; output is frame-sized.