从 SKILL.md 下沉的完整版。SKILL.md 保留 7 条硬规则速查,本文件是每条规则的展开:架构选型、取图渠道与代码、AppPhone JSX 骨架、ios_frame 三步用法、品位锚点全表。
做 iOS/Android/移动 app 原型时(触发:「app 原型」「iOS mockup」「移动应用」「做个 app」),下面四条覆盖通用 placeholder 原则——app 原型是 demo 现场,静态摆拍和米白占位卡没有说服力。
默认单文件 inline React——所有 JSX/data/styles 直接写进主 HTML 的 <script type="text/babel">...</script> 标签,不要用 <script src="components.jsx"> 外部加载。原因:file:// 协议下浏览器把外部 JS 当跨 origin 拦截,强制用户起 HTTP server 违反「双击就能开」的原型直觉。引用本地图片必须 base64 内嵌 data URL,别假设有 server。
拆外部文件只在两种情况:
components.jsx + data.js,同时明确交付说明(python3 -m http.server 命令 + 访问 URL)index.html + 每屏独立 HTML(today.html/graph.html...),iframe 聚合,每屏也都是自包含单文件选型速查:
| 场景 | 架构 | 交付方式 |
|---|---|---|
| 单人做 4-6 屏原型(主流) | 单文件 inline | 一个 .html 双击开 |
| 单人做大型 App(>10 屏) | 多 jsx + server | 附启动命令 |
| 多 agent 并行 | 多 HTML + iframe | index.html 聚合,每屏独立可开 |
默认主动去取真实图片填充,不要画 SVG、不要拿米白卡摆着、不要等用户要求。常用渠道:
| 场景 | 首选渠道 |
|---|---|
| 美术/博物馆/历史内容 | Wikimedia Commons(公共领域)、Met Museum Open Access、Art Institute of Chicago API |
| 通用生活/摄影 | Unsplash、Pexels(免版权) |
| 用户本地已有素材 | ~/Downloads、项目 _archive/ 或用户配置的素材库 |
Wikimedia 下载避坑(本机 curl 走代理 TLS 会炸,Python urllib 直接走得通):
# 合规 User-Agent 是硬性要求,否则 429
UA = 'ProjectName/0.1 (https://github.com/you; you@example.com)'
# 用 MediaWiki API 查真实 URL
api = 'https://commons.wikimedia.org/w/api.php'
# action=query&list=categorymembers 批量拿系列 / prop=imageinfo+iiurlwidth 取指定宽度 thumburl
只有当所有渠道都失败 / 版权不清 / 用户明确要求时,才退回诚实 placeholder(仍然不画烂 SVG)。
真图诚实性测试(关键):取图之前先问自己——「如果去掉这张图,信息是否有损?」
| 场景 | 判断 | 动作 |
|---|---|---|
| 文章/Essay 列表的封面、Profile 页的风景头图、设置页的装饰 banner | 装饰,与内容无内在关联 | 不要加。加了就是 AI slop,等同紫色渐变 |
| 博物馆/人物内容的肖像、产品详情的实物、地图卡片的地点 | 内容本身,有内在关联 | 必须加 |
| 图谱/可视化背景的极淡纹理 | 氛围,服从内容不抢戏 | 加,但 opacity ≤ 0.08 |
反例:给文字 Essay 配 Unsplash「灵感图」、给笔记 App 配 stock photo 模特——都是 AI slop。取真图的许可不等于滥用真图的通行证。
iOS App 原型的默认交付形态就一种,不要再问用户「要平铺还是可操作」:平铺 4-6 个主界面,且每一台都能交互。一眼看全貌(多台 iPhone 并排),又每台都能点 tab 切换、在界面上做基本操作(展开、切换、选中、打开弹层)。两个好处一次给齐,别让用户二选一。
| 维度 | 默认做法 |
|---|---|
| 屏数 | 平铺 4-6 个主界面(覆盖 app 的核心功能面,不是随便摆几个)。多于 6 个抓最主要的 4-6 个,其余可在单台内通过 tab/导航到达 |
| 布局 | 多台独立 iPhone 横向 flexWrap 并排,每台上方一行 italic 小字标签说明这是哪个界面 |
| 每台交互 | 每台都是独立的迷你状态机:tab bar 可切、界面内按钮/卡片/开关可点、能弹 modal——不是静态摆拍 |
只有两种特例才偏离默认(用户明确说了才走,否则一律默认):
ScreenComponent,不挂状态机)AppPhone 走完整 flow默认骨架(平铺多台,每台各自一个带 state 的 AppPhone):
// 每台 = 一个独立状态机,初始落在自己负责的主界面
function AppPhone({ initial }) {
const [screen, setScreen] = React.useState(initial);
const [modal, setModal] = React.useState(null);
// 按 screen 渲染对应 ScreenComponent,传入 onTabChange/onOpen/onClose/onToggle 等 callback
return (
<IosFrame>
<ScreenComponent
screen={screen}
onTabChange={setScreen}
onOpen={setModal}
onClose={() => setModal(null)}
/>
</IosFrame>
);
}
// 平铺:4-6 台并排,每台 initial 落在不同主界面
<div style={{display: 'flex', gap: 32, flexWrap: 'wrap', padding: 48, alignItems: 'flex-start'}}>
{mainScreens.map(s => (
<div key={s.id}>
<div style={{fontSize: 13, color: '#666', marginBottom: 8, fontStyle: 'italic'}}>{s.label}</div>
<AppPhone initial={s.id} />
</div>
))}
</div>
Screen 组件接 callback props(onTabChange、onOpen、onClose、onToggle、onAnnotation),不硬编码状态。TabBar、按钮、作品卡、开关加 cursor: pointer + hover 反馈。每台落在不同主界面,但 tab 切换后能到达彼此——平铺给全貌,点击给纵深。
静态截图只能看 layout,交互 bug 要点过才发现。用 Playwright 跑 3 项最小点击测试:进入详情 / 关键标注点 / tab 切换。检查 pageerror 为 0 再交付。Playwright 可用 npx playwright 调用,或按本机全局安装路径(npm root -g + /playwright)。
没有 design system 时默认往这些方向走,避免撞 AI slop:
| 维度 | 首选 | 避免 |
|---|---|---|
| 字体 | 衬线 display(Newsreader/Source Serif/EB Garamond)+ -apple-system body |
全场 SF Pro 或 Inter——太像系统默认,没风格 |
| 色彩 | 一个有温度的底色 + 单个 accent 贯穿全场(rust 橙/墨绿/深红) | 多色聚类(除非数据真的有 ≥3 个分类维度) |
| 信息密度·克制型(默认) | 少一层容器、少一个 border、少一个装饰性 icon——给内容留气口 | 每条卡片都配无意义的 icon + tag + status dot |
| 信息密度·高密度型(例外) | 当产品核心卖点是「智能 / 数据 / 上下文感知」时(AI 工具、Dashboard、Tracker、Copilot、番茄钟、健康监测、记账类),每屏需至少 3 处可见的产品差异化信息:非装饰性数据、对话/推理片段、状态推断、上下文关联 | 只放一个按钮一个时钟——AI 的智能感没表达出来,跟普通 App 没区别 |
| 细节签名 | 留一处「值得截图」的质感:极淡油画底纹 / serif 斜体引语 / 全屏黑底录音波形 | 到处平均用力,结果处处平淡 |
两条原则同时生效:
assets/ios_frame.jsx——禁止手写 Dynamic Island / status bar做 iPhone mockup 时硬性绑定 assets/ios_frame.jsx。这是已经对齐过 iPhone 15 Pro 精确规格的标准外壳:bezel、Dynamic Island(124×36、top:12、居中)、status bar(时间/信号/电池、两侧避让岛、vertical center 对齐岛中线)、Home Indicator、content 区 top padding 都处理好了。
禁止在你的 HTML 里自己写以下任何一项:
.dynamic-island / .island / position: absolute; top: 11/12px; width: ~120; 居中的黑圆角矩形.status-bar with 手写的时间/信号/电池图标.home-indicator / 底部 home bar自己写 99% 会撞位置 bug——status bar 的时间/电池被岛挤压、或 content top padding 算错导致第一行内容盖在岛下。iPhone 15 Pro 的刘海是固定 124×36 像素,留给 status bar 两侧的可用宽度很窄,不是你凭空估的。
用法(严格三步):
// 步骤 1: Read 本 skill 的 assets/ios_frame.jsx(相对本 SKILL.md 的路径)
// 步骤 2: 把整个 iosFrameStyles 常量 + IosFrame 组件贴进你的 <script type="text/babel">
// 步骤 3: 你自己的屏组件包在 <IosFrame>...</IosFrame> 里,不碰 island/status bar/home indicator
<IosFrame time="9:41" battery={85}>
<YourScreen /> {/* 内容从 top 54 开始渲染,下边留给 home indicator,你不用管 */}
</IosFrame>
例外:只有用户明确要求「假装是 iPhone 14 非 Pro 的刘海」「做 Android 不是 iOS」「自定义设备形态」时才绕过——此时读对应 android_frame.jsx 或修改 ios_frame.jsx 的常量,不要在项目 HTML 里另起一套 island/status bar。