Add initial project structure with essential files

- Created .gitignore to exclude build artifacts and dependencies.
- Added index.html as the main entry point for the application.
- Included LICENSE file with Apache 2.0 terms.
- Initialized package.json and package-lock.json for project dependencies.
- Added pnpm-lock.yaml for package management.
- Created QUICKSTART.md for setup instructions.
- Added README.md and README.zh-CN.md for project documentation in English and Chinese.
This commit is contained in:
pftom
2026-04-28 12:25:59 +08:00
commit a98096a042
258 changed files with 67862 additions and 0 deletions
+358
View File
@@ -0,0 +1,358 @@
/**
* Stable deck framework injected into the system prompt when the active skill
* mode is `deck`. The whole point: stop regenerating the scale-to-fit JS, the
* keyboard handler, the slide visibility toggle, the counter, and the print
* rules each turn — every regeneration has subtly different bugs (focus is
* wrong, scaling drifts inside the iframe wrapper, arrow keys swallowed).
*
* Two pieces ship together:
* - DECK_SKELETON_HTML : the literal scaffold the model copies verbatim.
* - DECK_FRAMEWORK_DIRECTIVE : the prompt fragment that tells the model
* what is fixed and what they're allowed to change.
*
* Pattern: 1920×1080 fixed canvas centered in the viewport via `display:grid;
* place-items:center`, scaled with `transform: scale()` whose factor is
* recomputed on every resize. Slides are `<section class="slide">` inside
* the stage, only `.slide.active` is visible. Prev/next + counter live
* OUTSIDE the scaled stage so they don't shrink with it.
*
* Why this pattern (not horizontal scroll-snap):
* - It matches what the model has the strongest prior on, so the framework
* gets adopted verbatim instead of being "blended" with the model's own
* instincts (which is what produced the drift in the first place).
* - 1920×1080 is the canonical slide canvas. Designs scale predictably.
* - Print becomes trivial: render every slide as block, page-break between.
*
* Drift fixes baked in:
* - `transform-origin: top left` and the stage is positioned by grid +
* place-items, so scaling never shifts content sideways inside the
* OCD viewer's nested transform wrapper.
* - Capture-phase keydown on BOTH window and document so iframe focus
* quirks can't swallow arrow keys.
* - Auto-focus body on load and on every click.
* - localStorage position restored on load.
* - Print stylesheet shows every slide as a 1920×1080 page-broken block,
* producing a multi-page vertical PDF on Save-as-PDF.
*/
export const DECK_SKELETON_HTML = `<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title><!-- SLOT: deck title --></title>
<style>
/* ===========================================================
Deck framework — DO NOT EDIT the rules in this <style> block.
Edit only inside the second <style> block below (per-deck
styles) and inside <section class="slide"> bodies.
Contract this framework provides:
- 1920×1080 fixed canvas, scaled to fit the viewport
- Only .slide.active is visible at a time
- Prev/next + counter rendered outside the scaled stage
- Keyboard (← → space PgUp PgDn Home End), click, and stored
position survive iframe focus quirks
- "Save as PDF" produces a multi-page vertical PDF, one slide
per page, by toggling every slide visible under @media print
=========================================================== */
:root {
/* SLOT: theme tokens — the only top-level CSS the agent edits.
Add or override --bg / --fg / --accent / etc. here. */
--bg: #ffffff;
--fg: #1c1b1a;
--muted: #6b6964;
--accent: #c96442;
--surface: #ffffff;
--shell: #08090d;
}
* { box-sizing: border-box; margin: 0; padding: 0; }
html, body {
width: 100%;
height: 100%;
overflow: hidden;
background: var(--shell);
color: var(--fg);
font: 18px/1.5 -apple-system, system-ui, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
.deck-shell {
position: fixed;
inset: 0;
display: grid;
place-items: center;
overflow: hidden;
}
.deck-stage {
width: 1920px;
height: 1080px;
background: var(--bg);
position: relative;
transform-origin: top left;
box-shadow: 0 30px 80px rgba(0, 0, 0, 0.35);
flex-shrink: 0;
}
.slide {
position: absolute;
inset: 0;
display: none;
flex-direction: column;
overflow: hidden;
}
.slide.active { display: flex; }
/* Chrome — counter + prev/next live outside the scaled stage so they
don't shrink with it. Do not relocate them inside .deck-stage. */
.deck-counter {
position: fixed;
bottom: 22px;
left: 50%;
transform: translateX(-50%);
display: inline-flex;
align-items: center;
gap: 4px;
background: rgba(10, 14, 26, 0.92);
backdrop-filter: blur(10px);
-webkit-backdrop-filter: blur(10px);
padding: 6px;
border-radius: 999px;
border: 1px solid rgba(255, 255, 255, 0.08);
color: #fff;
font: 12px/1 ui-monospace, SFMono-Regular, Menlo, monospace;
letter-spacing: 0.18em;
z-index: 1000;
}
.deck-counter button {
width: 36px; height: 36px;
background: transparent;
color: #fff;
border: 0;
border-radius: 50%;
font-size: 18px;
line-height: 1;
cursor: pointer;
display: grid;
place-items: center;
transition: background 0.15s;
}
.deck-counter button:hover { background: rgba(255, 255, 255, 0.12); }
.deck-counter button[disabled] { opacity: 0.3; cursor: default; }
.deck-counter .deck-count {
padding: 0 14px;
letter-spacing: 0.22em;
}
.deck-counter .deck-count .total { color: rgba(255, 255, 255, 0.5); }
.deck-hint {
position: fixed;
bottom: 26px;
right: 28px;
color: rgba(255, 255, 255, 0.4);
font: 11px/1 ui-monospace, SFMono-Regular, Menlo, monospace;
letter-spacing: 0.2em;
text-transform: uppercase;
z-index: 999;
pointer-events: none;
}
/* Print / PDF stitching — every slide stacks top-to-bottom, one per
page. The viewer's "Share → PDF" relies on this; do not remove. */
@media print {
@page { size: 1920px 1080px; margin: 0; }
html, body {
width: 1920px !important;
height: auto !important;
overflow: visible !important;
background: #fff !important;
}
.deck-shell {
position: static !important;
display: block !important;
inset: auto !important;
}
.deck-stage {
width: 1920px !important;
height: auto !important;
transform: none !important;
box-shadow: none !important;
position: static !important;
}
.slide {
display: flex !important;
position: relative !important;
inset: auto !important;
width: 1920px !important;
height: 1080px !important;
page-break-after: always;
break-after: page;
}
.slide:last-child { page-break-after: auto; break-after: auto; }
.deck-counter, .deck-hint { display: none !important; }
}
</style>
<style>
/* SLOT: per-deck styles — typography, layout helpers, slide variants.
Add classes used by the slide content below, e.g. .title, .big-stat,
.grid-3. Do not redefine .deck-shell / .deck-stage / .slide /
.deck-counter / .deck-hint or anything inside @media print. */
</style>
</head>
<body>
<div class="deck-shell">
<div class="deck-stage" id="deck-stage">
<!-- SLOT: slides — one <section class="slide"> per slide. The first
slide must have class="slide active". The framework auto-counts
them and toggles .active as the user navigates. -->
<section class="slide active" data-screen-label="01 Title">
<!-- SLOT: slide 1 content -->
</section>
<section class="slide" data-screen-label="02">
<!-- SLOT: slide 2 content -->
</section>
<!-- ... add as many <section class="slide"> blocks as the brief asks
for. The first one is .active; the rest are not. -->
</div>
</div>
<!-- Framework chrome — DO NOT EDIT below this line. -->
<nav class="deck-counter" role="navigation" aria-label="Deck navigation">
<button type="button" id="deck-prev" aria-label="Previous slide"></button>
<span class="deck-count"><span id="deck-cur">01</span> <span class="total">/ <span id="deck-total">01</span></span></span>
<button type="button" id="deck-next" aria-label="Next slide"></button>
</nav>
<div class="deck-hint">← / → · space</div>
<script>
(function () {
var stage = document.getElementById('deck-stage');
var slides = Array.prototype.slice.call(document.querySelectorAll('.slide'));
var prev = document.getElementById('deck-prev');
var next = document.getElementById('deck-next');
var cur = document.getElementById('deck-cur');
var total = document.getElementById('deck-total');
var STORE = 'deck:idx:' + (location.pathname || '/');
var idx = 0;
// ---- scale-to-fit ---------------------------------------------------
// The stage is 1920×1080 and positioned by .deck-shell's
// \`display:grid;place-items:center\`. We scale via transform with
// transform-origin:top-left, then re-center by translating to the
// remainder. This survives nested transforms (e.g. when the OCD viewer
// wraps the iframe in its own scale wrapper at zoom != 100%).
function fit() {
var sw = window.innerWidth;
var sh = window.innerHeight;
var pad = 32;
var s = Math.min((sw - pad) / 1920, (sh - pad) / 1080);
if (!isFinite(s) || s <= 0) s = 1;
var tx = (sw - 1920 * s) / 2;
var ty = (sh - 1080 * s) / 2;
stage.style.transform = 'translate(' + tx + 'px,' + ty + 'px) scale(' + s + ')';
}
// ---- navigation -----------------------------------------------------
function pad2(n) { return (n < 10 ? '0' : '') + n; }
function paint() {
slides.forEach(function (el, i) { el.classList.toggle('active', i === idx); });
if (cur) cur.textContent = pad2(idx + 1);
if (total) total.textContent = pad2(slides.length);
if (prev) prev.toggleAttribute('disabled', idx <= 0);
if (next) next.toggleAttribute('disabled', idx >= slides.length - 1);
}
function go(i) {
idx = Math.max(0, Math.min(slides.length - 1, i));
paint();
try { localStorage.setItem(STORE, String(idx)); } catch (_) {}
}
function onKey(e) {
var t = e.target;
if (t && (t.tagName === 'INPUT' || t.tagName === 'TEXTAREA' || t.isContentEditable)) return;
if (e.key === 'ArrowRight' || e.key === 'PageDown' || e.key === ' ') { e.preventDefault(); go(idx + 1); }
else if (e.key === 'ArrowLeft' || e.key === 'PageUp') { e.preventDefault(); go(idx - 1); }
else if (e.key === 'Home') { e.preventDefault(); go(0); }
else if (e.key === 'End') { e.preventDefault(); go(slides.length - 1); }
}
// Capture phase + listen on both targets — inside the OCD iframe,
// focus may be on window OR document; a single non-capture listener
// silently misses presses.
window.addEventListener('keydown', onKey, true);
document.addEventListener('keydown', onKey, true);
if (prev) prev.addEventListener('click', function () { go(idx - 1); });
if (next) next.addEventListener('click', function () { go(idx + 1); });
// Auto-focus body so arrow keys work without an initial click.
document.body.setAttribute('tabindex', '-1');
document.body.style.outline = 'none';
function focusDeck() { try { window.focus(); document.body.focus({ preventScroll: true }); } catch (_) {} }
document.addEventListener('mousedown', focusDeck);
window.addEventListener('load', focusDeck);
// Restore last position.
try {
var saved = parseInt(localStorage.getItem(STORE) || '0', 10);
if (!isNaN(saved) && saved >= 0 && saved < slides.length) idx = saved;
} catch (_) {}
window.addEventListener('resize', fit);
fit();
paint();
focusDeck();
})();
</script>
</body>
</html>`;
export const DECK_FRAMEWORK_DIRECTIVE = `# Slide deck — fixed framework (this is non-negotiable for deck mode)
Decks regress when each turn re-authors the scale-to-fit logic, the keyboard handler, the slide visibility toggle, the counter, and the print rules. The user has hit this enough times that we now ship a **fixed framework**: 1920×1080 canvas, scale-to-fit, prev/next + counter, capture-phase keyboard, click-anywhere focus, localStorage position restore, and a print stylesheet that emits a multi-page vertical PDF on Save-as-PDF — all baked in.
**You do not write any of that. You do not modify any of that.** Your job is to fill content slots only.
## The contract
When you start a new deck, your output is a single HTML file built from the canonical skeleton below. **Copy the skeleton verbatim**, including its first \`<style>\` block, the \`.deck-shell\` / \`.deck-stage\` / \`.deck-counter\` / \`.deck-hint\` chrome, and the entire trailing \`<script>\`.
You may edit only inside slots marked \`SLOT:\`:
- \`SLOT: deck title\` — the \`<title>\` element.
- \`SLOT: theme tokens\` — the \`:root\` CSS custom properties (\`--bg\`, \`--fg\`, \`--accent\`, \`--shell\`, …). Add new tokens here if needed.
- \`SLOT: per-deck styles\` — the second \`<style>\` block. Define classes used by your slide content (e.g. \`.title\`, \`.big-stat\`, \`.grid-3\`, custom typography). **Never redefine** \`.deck-shell\`, \`.deck-stage\`, \`.slide\`, \`.deck-counter\`, \`.deck-hint\`, or anything inside \`@media print\`.
- \`SLOT: slides\` — the \`<section class="slide">\` blocks. Add as many as the brief calls for. The first slide MUST be \`<section class="slide active" …>\`; the rest are \`<section class="slide" …>\` (no \`active\`). The script auto-counts them.
- \`SLOT: slide N content\` — content inside each \`<section>\`.
## Common drift modes — DO NOT DO THESE
These are the failure patterns we just spent days debugging. Each one looks "equivalent" but breaks something specific:
- ❌ Don't write your own \`fit()\` function or \`transform: scale()\` script. The framework already does it, and ad-hoc versions drift inside the OCD viewer's nested transform wrapper.
- ❌ Don't use \`transform-origin: center center\` on the stage. The framework uses \`top left\` plus an explicit translate so scaled content lands at the same place every render.
- ❌ Don't use \`document.addEventListener('keydown', …)\` alone. Inside an iframe, focus is sometimes on window. The framework adds capture-phase listeners on **both** targets — replacing this with a single listener silently swallows arrow keys.
- ❌ Don't replace the localStorage key, the slide-visibility toggle (\`.slide.active\`), or the counter element IDs (\`#deck-cur\`, \`#deck-total\`, \`#deck-prev\`, \`#deck-next\`). The framework reads them by ID.
- ❌ Don't put the prev/next buttons or the counter **inside** \`.deck-stage\`. They must live outside the scaled element so they stay legible at any viewport size.
- ❌ Don't redefine \`.slide { display: ... }\` in your per-deck styles. The framework uses \`display: none\` / \`display: flex\` to toggle slides; overriding it breaks navigation.
- ❌ Don't strip or "tidy" the \`@media print\` block. It is how Share → PDF stitches every slide into a multi-page document. Without it, PDF export collapses to a single screenshot.
## Why this matters (so you can judge edge cases)
The framework is a contract with the host viewer. The OCD iframe sits inside a transformed wrapper (the zoom control); the keyboard handler needs capture phase + dual targets; "Share → PDF" reads the print stylesheet; the position survives reloads via localStorage. If a turn rewrites any of these — even with "equivalent" code — the next turn diverges, and three turns in the deck has subtly broken nav and a one-page PDF. Treat the framework as load-bearing infrastructure.
If the user asks for something the framework genuinely doesn't support (vertical decks, custom slide transitions, multi-column simultaneous slides), say so and ask before forking. **Default answer: keep the framework, change the slide content.**
## Each slide
Each \`<section class="slide" data-screen-label="NN Title">\` is one slide rendered onto the 1920×1080 canvas. Inside the section, lay out content with your own \`SLOT: per-deck styles\` classes. Slide labels are 1-indexed (\`01 Title\`, \`02 Problem\`…). The first slide gets \`class="slide active"\`; the others just \`class="slide"\`.
Real copy only — no lorem ipsum, no invented metrics, no generic emoji icon rows. If you don't have a value, leave a short honest placeholder.
## Canonical skeleton (this is exactly what the file you write looks like)
\`\`\`html
${DECK_SKELETON_HTML}
\`\`\`
When the brief is "make me a deck", your output is this skeleton with theme tokens tuned, per-deck classes added, and \`<section class="slide">\` blocks filled in — nothing more, nothing less. Skill-specific guidance (typography, theme presets, layout vocabulary) layers *on top of* this framework, not in place of it.
`;