Finish Electron shell and workflow wiring
Template tests / tests (push) Failing after 4s

This commit is contained in:
Iisyourdad
2026-06-10 18:32:30 -05:00
parent a5bbdde480
commit f47aca67c2
22 changed files with 5002 additions and 2 deletions
+253
View File
@@ -0,0 +1,253 @@
#!/usr/bin/env node
'use strict';
const fs = require('node:fs');
const path = require('node:path');
const { GuideStore } = require('../core/store');
const raster = require('../core/raster');
const { encodePng } = require('../core/png');
const { buildRenderAst } = require('../core/renderast');
const { exportGuideArchive } = require('../core/archive');
const { runExport } = require('../exporters');
const { writeJsonSync, slugify } = require('../core/util');
const ROOT_DIR = path.resolve(__dirname, '..');
const DEFAULT_ROOT = path.join(ROOT_DIR, 'examples');
function parseArgs(argv) {
const out = { root: DEFAULT_ROOT };
for (let i = 0; i < argv.length; i++) {
const arg = argv[i];
if (arg === '--root' && argv[i + 1]) out.root = path.resolve(argv[++i]);
else if (arg === '--help' || arg === '-h') out.help = true;
}
return out;
}
function cleanDir(dir) {
fs.rmSync(dir, { recursive: true, force: true });
fs.mkdirSync(dir, { recursive: true });
}
function drawChrome(img, { accent, title, subtitle, sidebarLabel, bodyLabel }) {
const W = img.width;
const H = img.height;
raster.fillRect(img, 0, 0, W, H, [245, 247, 250, 255]);
raster.fillRect(img, 0, 0, W, 68, accent);
raster.fillRect(img, 28, 94, 270, H - 138, [255, 255, 255, 255]);
raster.fillRect(img, 326, 94, W - 354, H - 138, [255, 255, 255, 255]);
raster.fillRect(img, 48, 118, 212, 18, [232, 237, 243, 255]);
raster.fillRect(img, 48, 152, 212, 18, [232, 237, 243, 255]);
raster.fillRect(img, 48, 186, 212, 18, [232, 237, 243, 255]);
raster.fillRect(img, 362, 148, 220, 152, [230, 237, 245, 255]);
raster.fillRect(img, 608, 148, 276, 40, [235, 241, 248, 255]);
raster.fillRect(img, 608, 202, 276, 40, [235, 241, 248, 255]);
raster.fillRect(img, 608, 256, 276, 40, [235, 241, 248, 255]);
raster.drawText(img, 28, 20, title, 26, [255, 255, 255, 255]);
raster.drawText(img, 28, 44, subtitle, 12, [214, 226, 240, 255]);
raster.drawText(img, 48, 102, sidebarLabel, 12, [78, 90, 105, 255]);
raster.drawText(img, 356, 102, bodyLabel, 12, [78, 90, 105, 255]);
}
function makeShotOne() {
const img = raster.createImage(1280, 760, [245, 247, 250, 255]);
drawChrome(img, {
accent: [0, 104, 255, 255],
title: 'Reset password',
subtitle: 'Users > Security > Reset',
sidebarLabel: 'Users',
bodyLabel: 'Admin Portal',
});
raster.fillRect(img, 392, 156, 176, 36, [0, 104, 255, 255]);
raster.drawTextCentered(img, 480, 175, 'Open Users', 16, [255, 255, 255, 255]);
raster.fillRect(img, 644, 160, 160, 20, [255, 255, 255, 255]);
raster.fillRect(img, 644, 196, 240, 20, [255, 255, 255, 255]);
raster.fillRect(img, 644, 232, 220, 20, [255, 255, 255, 255]);
raster.drawText(img, 360, 336, '1. Open the Users list and confirm the target account is visible.', 12, [48, 59, 71, 255]);
raster.drawText(img, 360, 360, 'The highlight shows the next action target.', 12, [96, 108, 121, 255]);
return img;
}
function makeShotTwo() {
const img = raster.createImage(1280, 760, [245, 247, 250, 255]);
drawChrome(img, {
accent: [20, 115, 90, 255],
title: 'Security settings',
subtitle: '2-factor authentication and resets',
sidebarLabel: 'Security',
bodyLabel: 'Account settings',
});
raster.fillRect(img, 366, 160, 252, 56, [20, 115, 90, 255]);
raster.drawTextCentered(img, 492, 180, 'Enable 2FA', 18, [255, 255, 255, 255]);
raster.fillRect(img, 648, 160, 250, 22, [233, 238, 244, 255]);
raster.fillRect(img, 648, 196, 250, 22, [233, 238, 244, 255]);
raster.fillRect(img, 648, 232, 250, 22, [233, 238, 244, 255]);
raster.drawText(img, 360, 336, '2. Enable the reset policy and save the change.', 12, [48, 59, 71, 255]);
raster.drawText(img, 360, 360, 'The annotation number points at the primary action.', 12, [96, 108, 121, 255]);
return img;
}
function makeShotThree() {
const img = raster.createImage(1280, 760, [245, 247, 250, 255]);
drawChrome(img, {
accent: [36, 50, 78, 255],
title: 'Confirmation',
subtitle: 'Review before closing the workflow',
sidebarLabel: 'Review',
bodyLabel: 'Change summary',
});
raster.fillRect(img, 366, 150, 472, 210, [255, 255, 255, 255]);
raster.fillRect(img, 396, 182, 120, 18, [232, 237, 243, 255]);
raster.fillRect(img, 396, 220, 316, 18, [232, 237, 243, 255]);
raster.fillRect(img, 396, 256, 356, 18, [232, 237, 243, 255]);
raster.fillRect(img, 396, 292, 270, 18, [232, 237, 243, 255]);
raster.fillRect(img, 778, 298, 36, 36, [36, 50, 78, 255]);
raster.drawText(img, 396, 406, '3. Confirm the summary, then close the dialog.', 12, [48, 59, 71, 255]);
raster.drawText(img, 396, 430, 'A blur redacts the account number in the sample export.', 12, [96, 108, 121, 255]);
return img;
}
function createGuide(store) {
const guide = store.createGuide({
title: 'Reset a password in Admin Portal',
descriptionHtml: '<p>Offline sample guide showing capture, annotations, rich text, and exports.</p>',
placeholders: {
Product: 'Admin Portal',
Author: 'StepForge',
Department: 'Support',
},
flags: {
focusedViewDefault: true,
hideSkippedStepsInExports: true,
},
});
const steps = [
{
title: 'Open [[Product]] users',
descriptionHtml: '<p>Open the users list and select the target account.</p>',
annotations: [
{ type: 'rect', x: 0.275, y: 0.18, w: 0.19, h: 0.18, style: { stroke: '#0068ff', strokeWidth: 6, fill: 'transparent' } },
{ type: 'number', value: 1, x: 0.30, y: 0.08, w: 0.08, h: 0.12, style: { stroke: '#0068ff' } },
],
textBlocks: [
{ position: 'after-description', level: 'info', title: 'Tip', descriptionHtml: '<p>Use the search box to avoid scrolling.</p>' },
],
image: makeShotOne(),
},
{
title: 'Enable the reset policy',
descriptionHtml: '<p>Make sure the policy is active before continuing.</p>',
annotations: [
{ type: 'arrow', x: 0.47, y: 0.24, w: 0.23, h: -0.04, style: { stroke: '#14a375', strokeWidth: 5 } },
{ type: 'tooltip', x: 0.53, y: 0.13, w: 0.17, h: 0.08, text: 'Primary action', style: { fill: '#111827', textColor: '#ffffff', stroke: '#111827', tail: 'bottom' } },
{ type: 'number', value: 2, x: 0.31, y: 0.08, w: 0.08, h: 0.12, style: { stroke: '#14a375' } },
],
codeBlocks: [
{ id: 'cmd', language: 'bash', code: 'stepforge --capture --window --delay 300' },
],
image: makeShotTwo(),
},
{
title: 'Review the confirmation',
descriptionHtml: '<p>Confirm the summary and close the modal.</p>',
annotations: [
{ type: 'blur', x: 0.49, y: 0.32, w: 0.21, h: 0.08, radius: 12, style: { stroke: '#9ca3af', strokeWidth: 2 } },
{ type: 'highlight', x: 0.47, y: 0.24, w: 0.28, h: 0.20, style: { fill: '#ffeeb0', stroke: '#f0a500', strokeWidth: 2 } },
{ type: 'number', value: 3, x: 0.31, y: 0.08, w: 0.08, h: 0.12, style: { stroke: '#36a' } },
],
tableBlocks: [
{ id: 't1', rows: [['Field', 'Value'], ['Title', 'Admin Portal'], ['Owner', 'Support']] },
],
image: makeShotThree(),
},
];
steps.forEach((entry, index) => {
const buf = encodePng(entry.image);
store.addStep(guide.guideId, {
title: entry.title,
descriptionHtml: entry.descriptionHtml,
annotations: entry.annotations,
textBlocks: entry.textBlocks || [],
codeBlocks: entry.codeBlocks || [],
tableBlocks: entry.tableBlocks || [],
focusedView: { enabled: true, zoom: 1.1, panX: 0.5, panY: 0.5 },
}, buf, { width: entry.image.width, height: entry.image.height }, { position: index });
});
const substep = store.addStep(guide.guideId, {
kind: 'empty',
parentStepId: store.getGuide(guide.guideId).stepsOrder[1],
title: 'Confirm permission prompt',
descriptionHtml: '<p>Only administrators can complete this step.</p>',
textBlocks: [{ position: 'after-description', level: 'warn', title: 'Access', descriptionHtml: '<p>Admin rights required.</p>' }],
}, null, null, { position: 2 });
store.addStep(guide.guideId, {
kind: 'empty',
title: 'Legacy note',
hidden: true,
descriptionHtml: '<p>This hidden step exercises filtering in exports.</p>',
}, null, null, { position: 4 });
store.addStep(guide.guideId, {
kind: 'empty',
title: 'Deprecated flow',
skipped: true,
descriptionHtml: '<p>This skipped step remains in the library but is excluded from exports.</p>',
}, null, null, { position: 5 });
return { guideId: guide.guideId, substepId: substep.stepId };
}
function exportOutputs(store, guideId, root, manifest) {
const ast = buildRenderAst(store, guideId);
const formats = ['json', 'markdown', 'html-simple', 'html-rich', 'pdf', 'gif', 'image-bundle', 'docx', 'pptx'];
const outputs = {};
for (const format of formats) {
const outDir = path.join(root, 'sample-exports', format);
fs.mkdirSync(outDir, { recursive: true });
const result = runExport(format, ast, outDir, {});
outputs[format] = path.relative(root, result.file || outDir);
}
const archiveFile = path.join(root, 'sample-guide.sfgz');
exportGuideArchive(store, guideId, archiveFile);
manifest.archive = path.relative(root, archiveFile);
manifest.exports = outputs;
}
function main() {
const args = parseArgs(process.argv.slice(2));
if (args.help) {
console.log('Usage: node scripts/make-sample-guide.js [--root <dir>]');
process.exit(0);
}
const root = args.root;
const dataDir = path.join(root, 'sample-data');
const exportsDir = path.join(root, 'sample-exports');
cleanDir(root);
fs.mkdirSync(dataDir, { recursive: true });
fs.mkdirSync(exportsDir, { recursive: true });
const store = new GuideStore(dataDir);
const { guideId, substepId } = createGuide(store);
const manifest = {
format: 'stepforge-sample-manifest',
version: 1,
generatedAt: new Date().toISOString(),
guideId,
title: store.getGuide(guideId).title,
dataDir: path.relative(root, dataDir),
note: 'The sample guide is generated entirely offline from local assets.',
};
exportOutputs(store, guideId, root, manifest);
manifest.substepId = substepId;
manifest.slug = slugify(manifest.title);
writeJsonSync(path.join(root, 'sample-manifest.json'), manifest);
console.log(`Sample guide written to ${root}`);
}
if (require.main === module) main();