first-commit
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
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
This commit is contained in:
@@ -0,0 +1,601 @@
|
||||
#!/usr/bin/env node
|
||||
// animation-map.mjs — HyperFrames animation map for agents
|
||||
//
|
||||
// Reads every GSAP timeline registered in window.__timelines, enumerates
|
||||
// tweens, samples bboxes at N points per tween, computes flags and
|
||||
// human-readable summaries. Outputs a single animation-map.json.
|
||||
//
|
||||
// Usage:
|
||||
// node skills/hyperframes/scripts/animation-map.mjs <composition-dir> \
|
||||
// [--frames N] [--out <dir>] [--min-duration S] [--width W] [--height H] [--fps N]
|
||||
|
||||
import { mkdir, writeFile } from "node:fs/promises";
|
||||
import { resolve, join } from "node:path";
|
||||
import { hyperframesPackageSpec, importPackagesOrBootstrap } from "./package-loader.mjs";
|
||||
|
||||
const {
|
||||
createFileServer,
|
||||
createCaptureSession,
|
||||
initializeSession,
|
||||
closeCaptureSession,
|
||||
getCompositionDuration,
|
||||
} = (
|
||||
await importPackagesOrBootstrap(["@hyperframes/producer"], {
|
||||
npmPackages: [hyperframesPackageSpec("@hyperframes/producer")],
|
||||
})
|
||||
)["@hyperframes/producer"];
|
||||
|
||||
// ─── CLI ─────────────────────────────────────────────────────────────────────
|
||||
|
||||
const args = parseArgs(process.argv.slice(2));
|
||||
if (!args.composition) die("missing <composition-dir>");
|
||||
|
||||
const FRAMES = Number(args.frames ?? 6);
|
||||
const OUT_DIR = resolve(args.out ?? ".hyperframes/anim-map");
|
||||
const MIN_DUR = Number(args["min-duration"] ?? 0.15);
|
||||
const WIDTH = Number(args.width ?? 1920);
|
||||
const HEIGHT = Number(args.height ?? 1080);
|
||||
const FPS = Number(args.fps ?? 30);
|
||||
const COMP_DIR = resolve(args.composition);
|
||||
|
||||
await mkdir(OUT_DIR, { recursive: true });
|
||||
|
||||
// ─── Main ────────────────────────────────────────────────────────────────────
|
||||
|
||||
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 tweens = await enumerateTweens(session);
|
||||
const kept = tweens.filter((tw) => tw.end - tw.start >= MIN_DUR);
|
||||
|
||||
const report = {
|
||||
composition: COMP_DIR,
|
||||
duration,
|
||||
totalTweens: tweens.length,
|
||||
mappedTweens: kept.length,
|
||||
skippedMicroTweens: tweens.length - kept.length,
|
||||
tweens: [],
|
||||
};
|
||||
|
||||
for (let i = 0; i < kept.length; i++) {
|
||||
const tw = kept[i];
|
||||
const times = Array.from(
|
||||
{ length: FRAMES },
|
||||
(_, k) => +(tw.start + ((k + 0.5) / FRAMES) * (tw.end - tw.start)).toFixed(3),
|
||||
);
|
||||
|
||||
const bboxes = [];
|
||||
for (const t of times) {
|
||||
await seekTo(session, t);
|
||||
const bbox = await measureTarget(session, tw.selectorHint);
|
||||
bboxes.push({ t, ...bbox });
|
||||
}
|
||||
|
||||
const animProps = tw.props.filter(
|
||||
(p) => !["parent", "overwrite", "immediateRender", "startAt", "runBackwards"].includes(p),
|
||||
);
|
||||
const flags = computeFlags(tw, bboxes, { width: WIDTH, height: HEIGHT });
|
||||
const summary = describeTween(tw, animProps, bboxes, flags);
|
||||
|
||||
report.tweens.push({
|
||||
index: i + 1,
|
||||
selector: tw.selectorHint,
|
||||
targets: tw.targetCount,
|
||||
props: animProps,
|
||||
start: +tw.start.toFixed(3),
|
||||
end: +tw.end.toFixed(3),
|
||||
duration: +(tw.end - tw.start).toFixed(3),
|
||||
ease: tw.ease,
|
||||
bboxes,
|
||||
flags,
|
||||
summary,
|
||||
});
|
||||
}
|
||||
|
||||
markCollisions(report.tweens);
|
||||
|
||||
for (const tw of report.tweens) {
|
||||
if (tw.flags.includes("collision") && !tw.summary.includes("collision")) {
|
||||
tw.summary += " Overlaps another animated element.";
|
||||
}
|
||||
}
|
||||
|
||||
// ── Composition-level analysis ──
|
||||
report.choreography = buildTimeline(report.tweens, duration);
|
||||
report.density = computeDensity(report.tweens, duration);
|
||||
report.staggers = detectStaggers(report.tweens);
|
||||
report.elements = buildElementLifecycles(report.tweens);
|
||||
report.deadZones = findDeadZones(report.density, duration);
|
||||
report.snapshots = await captureSnapshots(session, report.tweens, duration);
|
||||
|
||||
await writeFile(join(OUT_DIR, "animation-map.json"), JSON.stringify(report, null, 2));
|
||||
|
||||
printSummary(report);
|
||||
} finally {
|
||||
await closeCaptureSession(session).catch(() => {});
|
||||
server.close();
|
||||
}
|
||||
|
||||
// ─── Seek helper ────────────────────────────────────────────────────────────
|
||||
|
||||
async function seekTo(session, t) {
|
||||
await session.page.evaluate((time) => {
|
||||
if (window.__hf && typeof window.__hf.seek === "function") {
|
||||
window.__hf.seek(time);
|
||||
return;
|
||||
}
|
||||
const tls = window.__timelines;
|
||||
if (tls) {
|
||||
for (const tl of Object.values(tls)) {
|
||||
if (typeof tl.seek === "function") tl.seek(time);
|
||||
}
|
||||
}
|
||||
}, t);
|
||||
await new Promise((r) => setTimeout(r, 100));
|
||||
}
|
||||
|
||||
// ─── Timeline introspection ──────────────────────────────────────────────────
|
||||
|
||||
async function enumerateTweens(session) {
|
||||
return await session.page.evaluate(() => {
|
||||
const results = [];
|
||||
const registry = window.__timelines || {};
|
||||
|
||||
const selectorOf = (el) => {
|
||||
if (!el || !(el instanceof Element)) return null;
|
||||
if (el.id) return `#${el.id}`;
|
||||
const cls = [...el.classList].slice(0, 2).join(".");
|
||||
return cls ? `${el.tagName.toLowerCase()}.${cls}` : el.tagName.toLowerCase();
|
||||
};
|
||||
|
||||
const walk = (node, parentOffset = 0) => {
|
||||
if (!node) return;
|
||||
if (typeof node.getChildren === "function") {
|
||||
const offset = parentOffset + (node.startTime?.() ?? 0);
|
||||
for (const child of node.getChildren(true, true, true)) {
|
||||
walk(child, offset);
|
||||
}
|
||||
return;
|
||||
}
|
||||
const targets = (node.targets?.() ?? []).filter((t) => t instanceof Element);
|
||||
if (!targets.length) return;
|
||||
const vars = node.vars ?? {};
|
||||
const props = Object.keys(vars).filter(
|
||||
(k) =>
|
||||
![
|
||||
"duration",
|
||||
"ease",
|
||||
"delay",
|
||||
"repeat",
|
||||
"yoyo",
|
||||
"onStart",
|
||||
"onUpdate",
|
||||
"onComplete",
|
||||
"stagger",
|
||||
].includes(k),
|
||||
);
|
||||
const start = parentOffset + (node.startTime?.() ?? 0);
|
||||
const end = start + (node.duration?.() ?? 0);
|
||||
results.push({
|
||||
selectorHint: selectorOf(targets[0]) ?? "(unknown)",
|
||||
targetCount: targets.length,
|
||||
props,
|
||||
start,
|
||||
end,
|
||||
ease: typeof vars.ease === "string" ? vars.ease : (vars.ease?.toString?.() ?? "none"),
|
||||
});
|
||||
};
|
||||
|
||||
for (const tl of Object.values(registry)) walk(tl, 0);
|
||||
results.sort((a, b) => a.start - b.start);
|
||||
return results;
|
||||
});
|
||||
}
|
||||
|
||||
async function measureTarget(session, selector) {
|
||||
return await session.page.evaluate((sel) => {
|
||||
const el = document.querySelector(sel);
|
||||
if (!el) return { x: 0, y: 0, w: 0, h: 0, missing: true };
|
||||
const r = el.getBoundingClientRect();
|
||||
const cs = getComputedStyle(el);
|
||||
return {
|
||||
x: Math.round(r.x),
|
||||
y: Math.round(r.y),
|
||||
w: Math.round(r.width),
|
||||
h: Math.round(r.height),
|
||||
opacity: parseFloat(cs.opacity),
|
||||
visible: cs.visibility !== "hidden" && cs.display !== "none",
|
||||
};
|
||||
}, selector);
|
||||
}
|
||||
|
||||
// ─── Tween description (the key output for agents) ──────────────────────────
|
||||
|
||||
function describeTween(tw, props, bboxes, flags) {
|
||||
const dur = (tw.end - tw.start).toFixed(2);
|
||||
const parts = [];
|
||||
|
||||
parts.push(`${tw.selectorHint} animates ${props.join("+")} over ${dur}s (${tw.ease})`);
|
||||
|
||||
// Movement
|
||||
const first = bboxes[0];
|
||||
const last = bboxes[bboxes.length - 1];
|
||||
if (first && last) {
|
||||
const dx = last.x - first.x;
|
||||
const dy = last.y - first.y;
|
||||
if (Math.abs(dx) > 3 || Math.abs(dy) > 3) {
|
||||
const dirs = [];
|
||||
if (Math.abs(dy) > 3) dirs.push(dy < 0 ? `${Math.abs(dy)}px up` : `${Math.abs(dy)}px down`);
|
||||
if (Math.abs(dx) > 3)
|
||||
dirs.push(dx < 0 ? `${Math.abs(dx)}px left` : `${Math.abs(dx)}px right`);
|
||||
parts.push(`moves ${dirs.join(" and ")}`);
|
||||
}
|
||||
}
|
||||
|
||||
// Opacity
|
||||
if (first && last && first.opacity !== undefined && last.opacity !== undefined) {
|
||||
const o1 = first.opacity;
|
||||
const o2 = last.opacity;
|
||||
if (Math.abs(o2 - o1) > 0.1) {
|
||||
if (o1 < 0.1 && o2 > 0.5) parts.push("fades in");
|
||||
else if (o1 > 0.5 && o2 < 0.1) parts.push("fades out");
|
||||
else parts.push(`opacity ${o1.toFixed(1)}→${o2.toFixed(1)}`);
|
||||
}
|
||||
}
|
||||
|
||||
// Scale (from props)
|
||||
if (props.includes("scale") || props.includes("scaleX") || props.includes("scaleY")) {
|
||||
parts.push("scales");
|
||||
}
|
||||
|
||||
// Size changes
|
||||
if (first && last) {
|
||||
const dw = last.w - first.w;
|
||||
const dh = last.h - first.h;
|
||||
if (Math.abs(dw) > 5) parts.push(`width ${first.w}→${last.w}px`);
|
||||
if (Math.abs(dh) > 5) parts.push(`height ${first.h}→${last.h}px`);
|
||||
}
|
||||
|
||||
// Visibility
|
||||
if (first && last && first.visible !== last.visible) {
|
||||
parts.push(last.visible ? "becomes visible" : "becomes hidden");
|
||||
}
|
||||
|
||||
// Final position
|
||||
if (last && !last.missing) {
|
||||
parts.push(`ends at (${last.x}, ${last.y}) ${last.w}×${last.h}px`);
|
||||
}
|
||||
|
||||
// Flags
|
||||
if (flags.length > 0) {
|
||||
parts.push(`FLAGS: ${flags.join(", ")}`);
|
||||
}
|
||||
|
||||
return parts.join(". ") + ".";
|
||||
}
|
||||
|
||||
// ─── Flag computation ───────────────────────────────────────────────────────
|
||||
|
||||
function computeFlags(tw, bboxes, { width, height }) {
|
||||
const flags = [];
|
||||
const dur = tw.end - tw.start;
|
||||
|
||||
if (bboxes.every((b) => b.w === 0 || b.h === 0)) flags.push("degenerate");
|
||||
|
||||
const anyOffscreen = bboxes.some(
|
||||
(b) =>
|
||||
b.x + b.w <= 0 ||
|
||||
b.y + b.h <= 0 ||
|
||||
b.x >= width ||
|
||||
b.y >= height ||
|
||||
b.x < -b.w * 0.5 ||
|
||||
b.y < -b.h * 0.5 ||
|
||||
b.x + b.w > width + b.w * 0.5 ||
|
||||
b.y + b.h > height + b.h * 0.5,
|
||||
);
|
||||
if (anyOffscreen) flags.push("offscreen");
|
||||
|
||||
if (bboxes.every((b) => b.opacity !== undefined && b.opacity < 0.01 && b.visible)) {
|
||||
flags.push("invisible");
|
||||
}
|
||||
|
||||
if (dur < 0.2 && tw.props.some((p) => ["y", "x", "opacity", "scale"].includes(p))) {
|
||||
flags.push("paced-fast");
|
||||
}
|
||||
if (dur > 2.0) flags.push("paced-slow");
|
||||
|
||||
return flags;
|
||||
}
|
||||
|
||||
function markCollisions(tweens) {
|
||||
for (let i = 0; i < tweens.length; i++) {
|
||||
for (let j = i + 1; j < tweens.length; j++) {
|
||||
const a = tweens[i];
|
||||
const b = tweens[j];
|
||||
if (a.end <= b.start || b.end <= a.start) continue;
|
||||
for (const ba of a.bboxes) {
|
||||
const bb = b.bboxes.find((x) => Math.abs(x.t - ba.t) < 0.05);
|
||||
if (!bb) continue;
|
||||
const overlap = rectOverlapArea(ba, bb);
|
||||
const aArea = ba.w * ba.h;
|
||||
if (aArea > 0 && overlap / aArea > 0.3) {
|
||||
if (!a.flags.includes("collision")) a.flags.push("collision");
|
||||
if (!b.flags.includes("collision")) b.flags.push("collision");
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function rectOverlapArea(a, b) {
|
||||
const x1 = Math.max(a.x, b.x);
|
||||
const y1 = Math.max(a.y, b.y);
|
||||
const x2 = Math.min(a.x + a.w, b.x + b.w);
|
||||
const y2 = Math.min(a.y + a.h, b.y + b.h);
|
||||
return Math.max(0, x2 - x1) * Math.max(0, y2 - y1);
|
||||
}
|
||||
|
||||
// ─── Composition-level analysis ─────────────────────────────────────────────
|
||||
|
||||
function buildTimeline(tweens, duration) {
|
||||
const cols = 60;
|
||||
const lines = [];
|
||||
const secPerCol = duration / cols;
|
||||
|
||||
lines.push("Timeline (" + duration.toFixed(1) + "s, each char ≈ " + secPerCol.toFixed(2) + "s):");
|
||||
lines.push(" " + "0s" + " ".repeat(cols - 8) + duration.toFixed(0) + "s");
|
||||
lines.push(" " + "┼" + "─".repeat(cols - 1) + "┤");
|
||||
|
||||
for (const tw of tweens) {
|
||||
const startCol = Math.floor(tw.start / secPerCol);
|
||||
const endCol = Math.min(cols, Math.ceil(tw.end / secPerCol));
|
||||
const bar =
|
||||
" ".repeat(startCol) +
|
||||
"█".repeat(Math.max(1, endCol - startCol)) +
|
||||
" ".repeat(Math.max(0, cols - endCol));
|
||||
const label = tw.selector + " " + tw.props.join("+");
|
||||
lines.push(" " + bar + " " + label);
|
||||
}
|
||||
|
||||
return lines.join("\n");
|
||||
}
|
||||
|
||||
function computeDensity(tweens, duration) {
|
||||
const buckets = [];
|
||||
for (let t = 0; t < duration; t += 0.5) {
|
||||
const active = tweens.filter((tw) => tw.start <= t + 0.5 && tw.end >= t);
|
||||
buckets.push({ t: +t.toFixed(1), activeTweens: active.length });
|
||||
}
|
||||
return buckets;
|
||||
}
|
||||
|
||||
function findDeadZones(density, duration) {
|
||||
const zones = [];
|
||||
let zoneStart = null;
|
||||
for (const d of density) {
|
||||
if (d.activeTweens === 0) {
|
||||
if (zoneStart === null) zoneStart = d.t;
|
||||
} else {
|
||||
if (zoneStart !== null) {
|
||||
const zoneEnd = d.t;
|
||||
if (zoneEnd - zoneStart >= 1.0) {
|
||||
zones.push({
|
||||
start: zoneStart,
|
||||
end: zoneEnd,
|
||||
duration: +(zoneEnd - zoneStart).toFixed(1),
|
||||
note:
|
||||
"No animation for " +
|
||||
(zoneEnd - zoneStart).toFixed(1) +
|
||||
"s. Intentional hold or missing entrance?",
|
||||
});
|
||||
}
|
||||
zoneStart = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (zoneStart !== null && duration - zoneStart >= 1.0) {
|
||||
zones.push({
|
||||
start: zoneStart,
|
||||
end: +duration.toFixed(1),
|
||||
duration: +(duration - zoneStart).toFixed(1),
|
||||
note:
|
||||
"No animation for " +
|
||||
(duration - zoneStart).toFixed(1) +
|
||||
"s at end. Final hold or missing outro?",
|
||||
});
|
||||
}
|
||||
return zones;
|
||||
}
|
||||
|
||||
function detectStaggers(tweens) {
|
||||
const groups = [];
|
||||
const used = new Set();
|
||||
|
||||
for (let i = 0; i < tweens.length; i++) {
|
||||
if (used.has(i)) continue;
|
||||
const tw = tweens[i];
|
||||
const group = [tw];
|
||||
used.add(i);
|
||||
|
||||
for (let j = i + 1; j < tweens.length; j++) {
|
||||
if (used.has(j)) continue;
|
||||
const other = tweens[j];
|
||||
const sameProps = tw.props.join(",") === other.props.join(",");
|
||||
const sameDuration = Math.abs(tw.duration - other.duration) < 0.05;
|
||||
const closeInTime = other.start - tw.start < tw.duration * 4;
|
||||
if (sameProps && sameDuration && closeInTime) {
|
||||
group.push(other);
|
||||
used.add(j);
|
||||
}
|
||||
}
|
||||
|
||||
if (group.length >= 3) {
|
||||
const intervals = [];
|
||||
for (let k = 1; k < group.length; k++) {
|
||||
intervals.push(+(group[k].start - group[k - 1].start).toFixed(3));
|
||||
}
|
||||
const avgInterval = intervals.reduce((a, b) => a + b, 0) / intervals.length;
|
||||
const maxDrift = Math.max(...intervals.map((iv) => Math.abs(iv - avgInterval)));
|
||||
const consistent = maxDrift < avgInterval * 0.3;
|
||||
|
||||
groups.push({
|
||||
elements: group.map((g) => g.selector),
|
||||
props: tw.props,
|
||||
count: group.length,
|
||||
intervals,
|
||||
avgInterval: +avgInterval.toFixed(3),
|
||||
consistent,
|
||||
note: consistent
|
||||
? group.length +
|
||||
" elements stagger at " +
|
||||
(avgInterval * 1000).toFixed(0) +
|
||||
"ms intervals"
|
||||
: group.length +
|
||||
" elements stagger with uneven intervals (" +
|
||||
intervals.map((iv) => (iv * 1000).toFixed(0) + "ms").join(", ") +
|
||||
")",
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
return groups;
|
||||
}
|
||||
|
||||
function buildElementLifecycles(tweens) {
|
||||
const elements = {};
|
||||
for (const tw of tweens) {
|
||||
const sel = tw.selector;
|
||||
if (!elements[sel]) {
|
||||
elements[sel] = { firstTween: tw.start, lastTween: tw.end, tweenCount: 0, props: new Set() };
|
||||
}
|
||||
elements[sel].firstTween = Math.min(elements[sel].firstTween, tw.start);
|
||||
elements[sel].lastTween = Math.max(elements[sel].lastTween, tw.end);
|
||||
elements[sel].tweenCount++;
|
||||
tw.props.forEach((p) => elements[sel].props.add(p));
|
||||
}
|
||||
|
||||
const result = {};
|
||||
for (const [sel, data] of Object.entries(elements)) {
|
||||
const lastBbox = findLastBbox(tweens, sel);
|
||||
result[sel] = {
|
||||
firstAppears: +data.firstTween.toFixed(3),
|
||||
lastAnimates: +data.lastTween.toFixed(3),
|
||||
tweenCount: data.tweenCount,
|
||||
props: [...data.props],
|
||||
endsVisible: lastBbox ? lastBbox.opacity > 0.1 && lastBbox.visible : null,
|
||||
finalPosition: lastBbox
|
||||
? { x: lastBbox.x, y: lastBbox.y, w: lastBbox.w, h: lastBbox.h }
|
||||
: null,
|
||||
};
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
function findLastBbox(tweens, selector) {
|
||||
for (let i = tweens.length - 1; i >= 0; i--) {
|
||||
if (tweens[i].selector === selector && tweens[i].bboxes?.length > 0) {
|
||||
return tweens[i].bboxes[tweens[i].bboxes.length - 1];
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
async function captureSnapshots(session, tweens, duration) {
|
||||
const times = [0, duration * 0.25, duration * 0.5, duration * 0.75, duration - 0.1];
|
||||
const snapshots = [];
|
||||
|
||||
for (const t of times) {
|
||||
await seekTo(session, t);
|
||||
const visible = await session.page.evaluate(() => {
|
||||
const out = [];
|
||||
const els = document.querySelectorAll("[id]");
|
||||
for (const el of els) {
|
||||
const cs = getComputedStyle(el);
|
||||
if (cs.display === "none") continue;
|
||||
const opacity = parseFloat(cs.opacity);
|
||||
if (opacity < 0.01) continue;
|
||||
const rect = el.getBoundingClientRect();
|
||||
if (rect.width < 1 || rect.height < 1) continue;
|
||||
out.push({
|
||||
id: el.id,
|
||||
x: Math.round(rect.x),
|
||||
y: Math.round(rect.y),
|
||||
w: Math.round(rect.width),
|
||||
h: Math.round(rect.height),
|
||||
opacity: +opacity.toFixed(2),
|
||||
});
|
||||
}
|
||||
return out;
|
||||
});
|
||||
|
||||
const activeTweens = tweens
|
||||
.filter((tw) => tw.start <= t && tw.end >= t)
|
||||
.map((tw) => tw.selector);
|
||||
|
||||
snapshots.push({
|
||||
t: +t.toFixed(2),
|
||||
visibleElements: visible.length,
|
||||
animatingNow: activeTweens,
|
||||
elements: visible,
|
||||
});
|
||||
}
|
||||
|
||||
return snapshots;
|
||||
}
|
||||
|
||||
// ─── Output ─────────────────────────────────────────────────────────────────
|
||||
|
||||
function printSummary(report) {
|
||||
console.log(
|
||||
`\nAnimation map: ${report.mappedTweens}/${report.totalTweens} tweens (skipped ${report.skippedMicroTweens} micro-tweens)`,
|
||||
);
|
||||
|
||||
const flagCounts = {};
|
||||
for (const tw of report.tweens) {
|
||||
for (const f of tw.flags) flagCounts[f] = (flagCounts[f] ?? 0) + 1;
|
||||
}
|
||||
if (Object.keys(flagCounts).length > 0) {
|
||||
for (const [f, n] of Object.entries(flagCounts)) console.log(` ${f}: ${n}`);
|
||||
}
|
||||
if (report.staggers?.length > 0) {
|
||||
console.log(` staggers: ${report.staggers.map((s) => s.note).join("; ")}`);
|
||||
}
|
||||
if (report.deadZones?.length > 0) {
|
||||
console.log(
|
||||
` dead zones: ${report.deadZones.map((z) => z.start + "-" + z.end + "s").join(", ")}`,
|
||||
);
|
||||
}
|
||||
|
||||
console.log(report.choreography);
|
||||
}
|
||||
|
||||
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(`animation-map: ${msg}`);
|
||||
process.exit(2);
|
||||
}
|
||||
Reference in New Issue
Block a user