a46764fb1b
ci / Validate workspace (push) Has been cancelled
landing-page-ci / Validate landing page (push) Has been cancelled
landing-page-deploy / Deploy landing page (push) Has been cancelled
github-metrics / Generate repository metrics SVG (push) Has been cancelled
refresh-contributors-wall / Refresh contributors wall cache bust (push) Waiting to run
339 lines
12 KiB
JavaScript
339 lines
12 KiB
JavaScript
#!/usr/bin/env node
|
||
// contrast-report.mjs — HyperFrames contrast audit
|
||
//
|
||
// Reads a composition, seeks to N sample timestamps, walks the DOM for text
|
||
// elements, measures the WCAG 2.1 contrast ratio between each element's
|
||
// declared foreground color and the pixels behind it, and emits:
|
||
//
|
||
// - contrast-report.json (machine-readable, one entry per text element × sample)
|
||
// - contrast-overlay.png (sprite grid; magenta=fail AA, yellow=pass AA only, green=AAA)
|
||
//
|
||
// Usage:
|
||
// node skills/hyperframes/scripts/contrast-report.mjs <composition-dir> \
|
||
// [--samples N] [--out <dir>] [--width W] [--height H] [--fps N]
|
||
//
|
||
// The composition directory must contain an index.html. Raw authoring HTML
|
||
// works — the producer's file server auto-injects the runtime at serve time.
|
||
// Exits 1 if any text element fails WCAG AA.
|
||
|
||
import { mkdir, writeFile } from "node:fs/promises";
|
||
import { resolve } from "node:path";
|
||
import { hyperframesPackageSpec, importPackagesOrBootstrap } from "./package-loader.mjs";
|
||
|
||
// Use the producer's file server — it auto-injects the HyperFrames runtime
|
||
// and render-seek bridge, so raw authoring HTML works without a build step.
|
||
const packages = await importPackagesOrBootstrap(["@hyperframes/producer", "sharp"], {
|
||
npmPackages: [hyperframesPackageSpec("@hyperframes/producer"), "sharp@0.34.5"],
|
||
});
|
||
const sharp = packages.sharp.default;
|
||
const {
|
||
createFileServer,
|
||
createCaptureSession,
|
||
initializeSession,
|
||
closeCaptureSession,
|
||
captureFrameToBuffer,
|
||
getCompositionDuration,
|
||
} = packages["@hyperframes/producer"];
|
||
|
||
// ─── CLI ─────────────────────────────────────────────────────────────────────
|
||
|
||
const args = parseArgs(process.argv.slice(2));
|
||
if (!args.composition) die("missing <composition-dir>");
|
||
|
||
const SAMPLES = Number(args.samples ?? 10);
|
||
const OUT_DIR = resolve(args.out ?? ".hyperframes/contrast");
|
||
const WIDTH = Number(args.width ?? 1920);
|
||
const HEIGHT = Number(args.height ?? 1080);
|
||
const FPS = Number(args.fps ?? 30);
|
||
const COMP_DIR = resolve(args.composition);
|
||
|
||
// ─── Main ────────────────────────────────────────────────────────────────────
|
||
|
||
await mkdir(OUT_DIR, { recursive: true });
|
||
|
||
const server = await createFileServer({ projectDir: COMP_DIR, port: 0 });
|
||
const session = await createCaptureSession(
|
||
server.url,
|
||
OUT_DIR,
|
||
{ width: WIDTH, height: HEIGHT, fps: FPS, format: "png" },
|
||
null,
|
||
);
|
||
await initializeSession(session);
|
||
|
||
try {
|
||
const duration = await getCompositionDuration(session);
|
||
const times = Array.from(
|
||
{ length: SAMPLES },
|
||
(_, i) => +(((i + 0.5) / SAMPLES) * duration).toFixed(3),
|
||
);
|
||
|
||
const allEntries = [];
|
||
const overlayFrames = [];
|
||
|
||
for (let i = 0; i < times.length; i++) {
|
||
const t = times[i];
|
||
const { buffer: pngBuf } = await captureFrameToBuffer(session, i, t);
|
||
const elements = await probeTextElements(session, t);
|
||
const annotated = await annotateFrame(pngBuf, elements);
|
||
overlayFrames.push({ t, png: annotated });
|
||
for (const el of elements) allEntries.push({ time: t, ...el });
|
||
}
|
||
|
||
const report = {
|
||
composition: COMP_DIR,
|
||
width: WIDTH,
|
||
height: HEIGHT,
|
||
duration,
|
||
samples: times,
|
||
entries: allEntries,
|
||
summary: summarize(allEntries),
|
||
};
|
||
|
||
await writeFile(resolve(OUT_DIR, "contrast-report.json"), JSON.stringify(report, null, 2));
|
||
await writeOverlaySprite(overlayFrames, resolve(OUT_DIR, "contrast-overlay.png"));
|
||
|
||
printSummary(report);
|
||
process.exitCode = report.summary.failAA > 0 ? 1 : 0;
|
||
} finally {
|
||
await closeCaptureSession(session).catch(() => {});
|
||
server.close();
|
||
}
|
||
|
||
// ─── DOM probe (runs in the page) ────────────────────────────────────────────
|
||
|
||
async function probeTextElements(session, _t) {
|
||
// `session.page` is the Puppeteer Page owned by the capture session.
|
||
// We pass a pure function to `evaluate`: it walks the DOM and returns
|
||
// enough info for us to compute a ratio in Node using the frame buffer.
|
||
return await session.page.evaluate(() => {
|
||
/** @type {Array<{selector: string, text: string, fg: [number,number,number,number], fontSize: number, fontWeight: number, bbox: {x:number,y:number,w:number,h:number}}>} */
|
||
const out = [];
|
||
const walker = document.createTreeWalker(document.body, NodeFilter.SHOW_ELEMENT);
|
||
const parseColor = (c) => {
|
||
const m = c.match(/rgba?\(([^)]+)\)/);
|
||
if (!m) return [0, 0, 0, 1];
|
||
const parts = m[1].split(",").map((s) => parseFloat(s.trim()));
|
||
return [parts[0], parts[1], parts[2], parts[3] ?? 1];
|
||
};
|
||
const selectorOf = (el) => {
|
||
if (el.id) return `#${el.id}`;
|
||
const cls = [...el.classList].slice(0, 2).join(".");
|
||
return cls ? `${el.tagName.toLowerCase()}.${cls}` : el.tagName.toLowerCase();
|
||
};
|
||
let el;
|
||
while ((el = walker.nextNode())) {
|
||
// must have direct text
|
||
const direct = [...el.childNodes].some(
|
||
(n) => n.nodeType === 3 && n.textContent.trim().length,
|
||
);
|
||
if (!direct) continue;
|
||
const cs = getComputedStyle(el);
|
||
if (cs.visibility === "hidden" || cs.display === "none") continue;
|
||
if (parseFloat(cs.opacity) <= 0.01) continue;
|
||
const rect = el.getBoundingClientRect();
|
||
if (rect.width < 8 || rect.height < 8) continue;
|
||
out.push({
|
||
selector: selectorOf(el),
|
||
text: el.textContent.trim().slice(0, 60),
|
||
fg: parseColor(cs.color),
|
||
fontSize: parseFloat(cs.fontSize),
|
||
fontWeight: Number(cs.fontWeight) || 400,
|
||
bbox: { x: rect.x, y: rect.y, w: rect.width, h: rect.height },
|
||
});
|
||
}
|
||
return out;
|
||
});
|
||
}
|
||
|
||
// ─── Pixel sampling + WCAG math ──────────────────────────────────────────────
|
||
|
||
async function annotateFrame(pngBuf, elements) {
|
||
const img = sharp(pngBuf);
|
||
const meta = await img.metadata();
|
||
const { width, height } = meta;
|
||
const raw = await img.ensureAlpha().raw().toBuffer();
|
||
const channels = 4;
|
||
|
||
const measured = [];
|
||
for (const el of elements) {
|
||
const bg = sampleRingMedian(raw, width, height, channels, el.bbox);
|
||
const fg = compositeOver(el.fg, bg); // flatten any alpha against measured bg
|
||
const ratio = wcagRatio(fg, bg);
|
||
const large = isLargeText(el.fontSize, el.fontWeight);
|
||
el.bg = bg;
|
||
el.ratio = +ratio.toFixed(2);
|
||
el.wcagAA = large ? ratio >= 3 : ratio >= 4.5;
|
||
el.wcagAALarge = ratio >= 3;
|
||
el.wcagAAA = large ? ratio >= 4.5 : ratio >= 7;
|
||
measured.push(el);
|
||
}
|
||
|
||
// Draw boxes + ratio labels as an SVG overlay (sharp composite).
|
||
const svg = buildOverlaySVG(measured, width, height);
|
||
return await sharp(pngBuf)
|
||
.composite([{ input: Buffer.from(svg), top: 0, left: 0 }])
|
||
.png()
|
||
.toBuffer();
|
||
}
|
||
|
||
function sampleRingMedian(raw, width, height, channels, bbox) {
|
||
// 4-px ring immediately outside the element bbox. Median of each channel.
|
||
const r = [],
|
||
g = [],
|
||
b = [];
|
||
const x0 = Math.max(0, Math.floor(bbox.x) - 4);
|
||
const x1 = Math.min(width - 1, Math.ceil(bbox.x + bbox.w) + 4);
|
||
const y0 = Math.max(0, Math.floor(bbox.y) - 4);
|
||
const y1 = Math.min(height - 1, Math.ceil(bbox.y + bbox.h) + 4);
|
||
const pushPixel = (x, y) => {
|
||
const i = (y * width + x) * channels;
|
||
r.push(raw[i]);
|
||
g.push(raw[i + 1]);
|
||
b.push(raw[i + 2]);
|
||
};
|
||
for (let x = x0; x <= x1; x++) {
|
||
pushPixel(x, y0);
|
||
pushPixel(x, y1);
|
||
}
|
||
for (let y = y0; y <= y1; y++) {
|
||
pushPixel(x0, y);
|
||
pushPixel(x1, y);
|
||
}
|
||
return [median(r), median(g), median(b), 1];
|
||
}
|
||
|
||
function median(arr) {
|
||
const s = [...arr].sort((a, b) => a - b);
|
||
return s[Math.floor(s.length / 2)];
|
||
}
|
||
|
||
function compositeOver([fr, fg, fb, fa], [br, bg, bb]) {
|
||
return [
|
||
Math.round(fr * fa + br * (1 - fa)),
|
||
Math.round(fg * fa + bg * (1 - fa)),
|
||
Math.round(fb * fa + bb * (1 - fa)),
|
||
1,
|
||
];
|
||
}
|
||
|
||
function relLum([r, g, b]) {
|
||
const ch = (v) => {
|
||
const s = v / 255;
|
||
return s <= 0.03928 ? s / 12.92 : ((s + 0.055) / 1.055) ** 2.4;
|
||
};
|
||
return 0.2126 * ch(r) + 0.7152 * ch(g) + 0.0722 * ch(b);
|
||
}
|
||
|
||
function wcagRatio(a, b) {
|
||
const la = relLum(a);
|
||
const lb = relLum(b);
|
||
const [L1, L2] = la > lb ? [la, lb] : [lb, la];
|
||
return (L1 + 0.05) / (L2 + 0.05);
|
||
}
|
||
|
||
function isLargeText(fontSize, fontWeight) {
|
||
return fontSize >= 24 || (fontSize >= 19 && fontWeight >= 700);
|
||
}
|
||
|
||
// ─── Overlay rendering ───────────────────────────────────────────────────────
|
||
|
||
function buildOverlaySVG(elements, w, h) {
|
||
const rects = elements
|
||
.map((el) => {
|
||
const color = !el.wcagAA ? "#ff00aa" : !el.wcagAAA ? "#ffcc00" : "#00e08a";
|
||
const { x, y, w: bw, h: bh } = el.bbox;
|
||
return `
|
||
<rect x="${x}" y="${y}" width="${bw}" height="${bh}"
|
||
fill="none" stroke="${color}" stroke-width="3"/>
|
||
<rect x="${x}" y="${y - 18}" width="${48}" height="16" fill="${color}"/>
|
||
<text x="${x + 4}" y="${y - 5}" font-family="monospace" font-size="12" fill="#000">
|
||
${el.ratio.toFixed(1)}:1
|
||
</text>`;
|
||
})
|
||
.join("");
|
||
return `<svg xmlns="http://www.w3.org/2000/svg" width="${w}" height="${h}">${rects}</svg>`;
|
||
}
|
||
|
||
async function writeOverlaySprite(frames, outPath) {
|
||
if (!frames.length) return;
|
||
const cols = Math.min(frames.length, 5);
|
||
const rows = Math.ceil(frames.length / cols);
|
||
const { width, height } = await sharp(frames[0].png).metadata();
|
||
const scale = 0.25;
|
||
const cellW = Math.round(width * scale);
|
||
const cellH = Math.round(height * scale);
|
||
|
||
const cells = await Promise.all(
|
||
frames.map(async (f) => ({
|
||
input: await sharp(f.png).resize(cellW, cellH).png().toBuffer(),
|
||
time: f.t,
|
||
})),
|
||
);
|
||
|
||
const composites = cells.map((c, i) => ({
|
||
input: c.input,
|
||
top: Math.floor(i / cols) * cellH,
|
||
left: (i % cols) * cellW,
|
||
}));
|
||
|
||
await sharp({
|
||
create: {
|
||
width: cols * cellW,
|
||
height: rows * cellH,
|
||
channels: 3,
|
||
background: { r: 16, g: 16, b: 20 },
|
||
},
|
||
})
|
||
.composite(composites)
|
||
.png()
|
||
.toFile(outPath);
|
||
}
|
||
|
||
// ─── Summary ────────────────────────────────────────────────────────────────
|
||
|
||
function summarize(entries) {
|
||
const total = entries.length;
|
||
const failAA = entries.filter((e) => !e.wcagAA).length;
|
||
const passAAonly = entries.filter((e) => e.wcagAA && !e.wcagAAA).length;
|
||
const passAAA = entries.filter((e) => e.wcagAAA).length;
|
||
return { total, failAA, passAAonly, passAAA };
|
||
}
|
||
|
||
function printSummary({ summary, entries }) {
|
||
const { total, failAA, passAAonly, passAAA } = summary;
|
||
console.log(`\nContrast report: ${total} text-element samples`);
|
||
console.log(` fail WCAG AA: ${failAA}`);
|
||
console.log(` pass AA, not AAA: ${passAAonly}`);
|
||
console.log(` pass AAA: ${passAAA}`);
|
||
if (failAA) {
|
||
console.log("\nFailures:");
|
||
for (const e of entries.filter((x) => !x.wcagAA)) {
|
||
console.log(` t=${e.time}s ${e.selector.padEnd(24)} ${e.ratio.toFixed(2)}:1 "${e.text}"`);
|
||
}
|
||
}
|
||
}
|
||
|
||
// ─── Utilities ──────────────────────────────────────────────────────────────
|
||
|
||
function parseArgs(argv) {
|
||
const out = {};
|
||
let positional = 0;
|
||
for (let i = 0; i < argv.length; i++) {
|
||
const a = argv[i];
|
||
if (a.startsWith("--")) {
|
||
const k = a.slice(2);
|
||
const v = argv[i + 1]?.startsWith("--") ? true : argv[++i];
|
||
out[k] = v;
|
||
} else if (positional === 0) {
|
||
out.composition = a;
|
||
positional++;
|
||
}
|
||
}
|
||
return out;
|
||
}
|
||
|
||
function die(msg) {
|
||
console.error(`contrast-report: ${msg}`);
|
||
process.exit(2);
|
||
}
|