这个清单来自"一人公司"分享 PPT 的真实迭代过程。每一条都是踩过坑之后总结的,按重要性排序。
生成 PPT 前,先通读一遍;生成后,逐项自检。
现象:直接把 layouts.md 的骨架粘到新 HTML,结果样式全部丢失——大标题变成非衬线、数据大字报字体小得像正文、pipeline 多页糊成一坨、图片堆到浏览器底部。
根因:如果 template.html 的 <style> 里没有这些类的定义,浏览器就 fallback 到默认样式。
做法:
Read assets/template.html,确认 layouts.md 里用到的类都已定义h-hero / h-xl / h-sub / h-md / lead / meta-row / stat-card / stat-label / stat-nb / stat-unit / stat-note / pipeline-section / pipeline-label / pipeline / step / step-nb / step-title / step-desc / grid-2-7-5 / grid-2-6-6 / grid-2-8-4 / grid-3-3 / frame / img-cap / callout-src<style> 里补上,不要在每页 inline 重写现象:在中式杂志风格里用 emoji(🎯 💡 ✅)会立刻破坏格调。
做法:用 Lucide 图标库,CDN 方式引用:
<script src="https://unpkg.com/lucide@latest/dist/umd/lucide.min.js"></script>
...
<i data-lucide="target" class="ico-md"></i>
...
<script>lucide.createIcons();</script>
常用图标名:target / palette / search-check / compass / share-2 / crown / check-circle / x-circle / plus / arrow-right / grid-2x2 / network
现象:用 aspect-ratio 撑图,网格会在父容器不足时堆叠或切掉图片关键信息(比如截图上部的标题栏)。
做法:图片容器用固定 height + overflow hidden,图片走 object-fit:cover + object-position:top:
<figure class="frame-img" style="height:26vh">
<img src="screenshot.png">
</figure>
CSS 里 .frame-img img 已经预设 object-position:top,只裁底。
绝不用这种写法(会在网格中撑破容器):
<!-- 坏例 -->
<figure class="frame-img" style="aspect-ratio: 16/9">...</figure>
例外:单张主视觉(非网格内)可以用 aspect-ratio + max-height,因为父容器会兜底。
现象:所有 light 页面背景都像蒙了一层灰,甚至 hero light 也灰。
根因:JS 根据 slide 的主题切换两张 canvas 的 opacity。如果整个 deck 开场是 hero dark,而没有任何机制能把 bg 切到 light,body 永远不加 light-bg 类,canvas#bg-dark 一直在上面。
做法:
go() 函数已改为从 classList 推断主题(light / dark),所以 slide 必须明确带 light 或 dark 类。不要漏写,更不要用其他自定义主题名hero light / hero dark,正文页用 light / dark。只写 hero 不带主题色是坏的light-bg现象:除封面 hero dark 外,其余所有页面默认写 light——视觉平淡,没有呼吸感,白花花一片。
根因:layouts.md 的骨架默认全写 light,如果只是粘贴骨架不调整主题,就会全亮。
做法:
hero dark / hero light / light / dark 中的哪一个,对齐后再写代码hero dark + ≥1 hero light;不能全是 light 正文页——必须有 dark 正文页light / dark 交替light(截图/数字/流程需要亮底)hero darkhero dark 与 hero light 交替grep 'class="slide' index.html,目视确认节奏有交错现象:左上角 .chrome 写"Design First · 设计先行",同一页里 .kicker 又写"Phase 01 · 设计阶段"——同义翻译,AI 味浓。
做法:
现象:中文大标题字号设太大(比如 13vw),结果每行只容 1 个字,强制换行非常难看。
做法:
h-hero(最大):10vw,且标题长度 ≤ 5 字h-xl(次大):6vw-7vw<br> 手工断行,不要依赖自动换行white-space:nowrap示例:我不是程序员。(6 字)用 h-xl 7.2vw + nowrap,一行排完。
做法:
所有字体用 Google Fonts CDN 引入,模板里已预设。
align-self:end 贴底现象:左文右图布局里,为了让右列图片和左列 callout 底部对齐,在 <figure> 上加 align-self:end。结果:
align-self 完全失效,图片掉到文档流最下面被浏览器底栏遮挡.foot 和 #nav 圆点遮挡做法:
.frame.grid-2-7-5(或 .grid-2-6-6/.grid-2-8-4)<figure class="frame-img r-16x10"> 或 <figure class="frame-img r-4x3"> 自然贴顶即可justify-content:space-between,不要动右列margin-top:7vh 到 9vh,让图片跟正文内容区对齐现象:aspect-ratio: 2592/1798 这种从原图复制的比例,在不同屏幕下撑出奇怪的空白或溢出。
做法:无论原图什么比例,占位器固定用标准比例 16/10 / 4/3 / 3/2 / 1/1 / 16/9。图片自动 object-fit:cover + object-position:top,顶部不裁,底部裁掉一点无伤大雅。
现象:为了"高级感"加了强阴影或黑框,瞬间变成商务 PPT。
做法:最多 1-4px 的微圆角 + 极淡的底噪(已在模板里)。不要加 box-shadow,不要加 border(除非 1px 极淡的灰)。
推荐节奏(25-30 页):
Hero Cover → Act Divider (hero) → 3-4 pages non-hero → Act Divider (hero)
→ 4-5 pages non-hero → Hero Question → ... → Hero Close
连续 2 页以上 hero 会让人疲劳,连续 4 页以上 non-hero 会让节奏死。
大字报(big numbers / hero question)和密集页(pipeline / image grid)交替出现,听众眼睛才不累。
现象:一会儿写 "Skills",一会儿写 "技能",一会儿写 "薄承载厚技能",全篇不一致。
做法:
用 XX / 总页数 的格式(比如 05 / 27)。不要在右上角加动态页码(会和 .chrome 重复)。
现象:生成后打开浏览器,翻页时内容直接"啪"地出来,没有任何节奏感——杂志风完全靠排版硬撑,少了层级展开的仪式感。
根因:完全没给任何元素加 data-anim,Motion One 脚本找不到可播的元素,整页静态出现。
做法:
data-anim<section> 上加对应 data-animate
data-animate="quote" + 每行 <span data-anim="line" style="display:block">data-animate="directional" + 左列 data-anim="left" + 右列 data-anim="right"data-animate="pipeline" + 每 step 加 data-anim="step"data-anim)自检:生成后 grep -c 'data-anim' index.html,应该数十条以上。如果只有个位数,一定漏标了。
现象:流水线页直接全部淡入,失去"一步步讲"的节奏,但切到下一页时又只能往前翻,没法回到上一个 step。
做法:Layout 6 的 <section> 必须加 data-animate="pipeline"。演示时按 →/空格/滚轮下滑可以逐个点亮 step,全部点亮之后再按 → 才会翻到下一页。这个节奏是刻意的,不是 bug。
dark hero:遮罩 12-15%(WebGL 明显透出) light hero:遮罩 16-20%(WebGL 隐约可见,不抢字) 普通 light/dark 页:遮罩 92-95%(几乎不透)
如果页面文字非常少(hero question),遮罩可以再薄些;如果正文密集,必须加厚遮罩确保可读。
现象:Spiral Vortex、径向涟漪在 light 主题下太显眼,像 Windows 98 屏保。
做法:light hero 用 FBM 域扭曲驱动的无中心流动,底色保持银/纸色(接近 #F0F0F0 / #FBF8F3),彩虹偏色 subtle(0.05 以下)。
Dark hero 可以用 Holographic Dispersion(钛金色散)等带中心结构的 shader,因为黑底能容纳更多视觉信息。
justify-content:space-between:标题贴顶,引用框贴底align-self:endmargin-top:7vh 到 9vhalign-items:start(不是 center / end)margin-top:6vh 到 8vh.frame-img 用同一个固定高度类,如 .h-16 / .h-18 / .h-22,不要用同一个超宽容器硬塞所有 .frame-img 和 .frame-img img 都加 border-radius:4px,视觉上"柔和"但不软。不要超过 8px,否则像消费 app UI。
图片放在 images/ 文件夹下,HTML 里用相对路径 images/xxx.png,不要用绝对路径。
.chrome 里写死JS 会动态算总页数并扩展底部翻页圆点,但 .chrome 里的 XX / N 是写死的。加页/删页时要手工改 N。
模板默认支持:← → / 滚轮 / 触屏滑动 / 底部圆点 / Home·End。不要删 JS 里的导航逻辑。
height:100vh 硬设,用 min-height:80vh100vh 会让内容刚好卡满屏幕,但浏览器工具栏、标签栏会吃掉一部分高度,导致内容溢出。用 min-height:80vh + align-content:center 更稳。
生成完 PPT 后,逐项对照这个清单(勾一下):
预检(生成前)
□ 已读过 template.html 的 <style>,确认所需类都存在
□ 已决定每页用哪个 Layout(1-10)
□ 已画出"主题节奏表":每页明确 hero dark / hero light / light / dark
□ 节奏表满足硬规则:无连续 3 页同主题 / 有 ≥1 hero dark + ≥1 hero light(8 页以上) / 至少有 1 个 dark 正文页
□ `<title>` 已改为实际 deck 标题(grep "[必填]" 应无结果)
内容
□ 每一幕的页数比例合理(不会头重脚轻)
□ 没有使用 emoji 作图标
□ Skills / Harness 等术语用法统一
□ 每页的 kicker + 标题 + 正文 三级信息清晰
排版
□ 所有大标题没有出现 1 字 1 行的换行
□ 图片网格用 height:Nvh 而非 aspect-ratio
□ 图片只裁底部,顶部和左右完整
□ 衬线/非衬线字体分工符合模板
□ Pipeline 多组之间有明显分隔
视觉
□ hero 页和 non-hero 页交替
□ WebGL 背景在 hero 页可见
□ 图片有微弱圆角
□ 没有沉重的阴影和边框
交互
□ ← → 翻页正常
□ 底部圆点数量与总页数匹配
□ chrome 里的页码和实际页号一致
□ ESC 键触发索引视图(如果保留)
动效
□ `assets/motion.min.js` 存在(本地兜底)
□ 翻页时内容逐个淡入,不是"啪"一下全出
□ 大引用页 `<section>` 带 `data-animate="quote"`,每行 `<span data-anim="line">`
□ Before/After 对比页 `<section>` 带 `data-animate="directional"`,左右列标 left/right
□ Pipeline 页 `<section>` 带 `data-animate="pipeline"`,每 step 标 data-anim="step"
□ `grep -c 'data-anim' index.html` 数量 ≥ 页数 × 3(平均每页 3 个以上标记)
全勾完,才是合格的 PPT。