feat(core/frontend): restore ModelPhotoDemoPage A/B static showcase to baselines (§1 last item)

ModelPhotoDemoPage variant A/B pixel-restored to model-photo-demo-a/b.html (product rail + model grid +
batch result cards / task-stream layout). Sibling AssetFactoryPage/ImageWorkbenchPage untouched.
+ QA: shot-aitools.mjs / shot-pipeline.mjs. verified: tsc --noEmit clean; A/B + model-photo screenshots match.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
seaislee1209 2026-06-05 17:14:56 +08:00
parent 2242241c3b
commit 64b0f3a1aa
3 changed files with 1008 additions and 2 deletions

View File

@ -978,3 +978,518 @@
.image-workbench .ic-param-menu .mi.selected { color: var(--heat); font-weight: 600; }
.image-workbench .ic-param-menu .mi .mi-check { margin-left: auto; opacity: 0; color: var(--heat); }
.image-workbench .ic-param-menu .mi.selected .mi-check { opacity: 1; }
/* ============================================================
C · 模特图方案对比 Demo(.model-demo · 纯静态展示)
基线:public/exact/model-photo-demo-a.html(参数面板 + 结果双栏)
public/exact/model-photo-demo-b.html(任务流 + 底部 fixed 参数栏)
逐字对齐 .content 级正文全局 token + 共享 .pill;.content padding 抵消同 .image-workbench
============================================================ */
.model-demo {
/* 抵消 .content 的 36/48/60 padding,贴边铺满(同 .image-workbench 思路) */
margin: -36px -48px -60px;
height: calc(100vh - 65px);
display: flex; flex-direction: column;
background: var(--background-base);
overflow: hidden;
}
/* 顶部 demo 提示条(两版共用) */
.model-demo .dm-banner {
flex-shrink: 0;
margin: 12px 28px 0;
padding: 8px 12px;
background: var(--heat-12);
border: 1px dashed var(--heat-20);
border-radius: var(--r-sm);
font-size: 12px; color: var(--accent-black);
font-family: var(--font-mono); letter-spacing: .02em; line-height: 1.5;
}
.model-demo .dm-banner b { color: var(--heat); }
/* 两栏:左侧栏 260px + 主区 */
.model-demo .dm-grid {
flex: 1; min-height: 0;
display: grid;
grid-template-columns: 260px minmax(0, 1fr);
}
@media (max-width: 1100px) {
.model-demo .dm-grid { grid-template-columns: 220px minmax(0, 1fr); }
}
/* ── 左侧栏 · 商品空间(两版共用) ── */
.model-demo .dm-side {
border-right: 1px solid var(--border-faint);
background: var(--surface);
display: flex; flex-direction: column;
min-height: 0;
}
.model-demo .dm-side-h { padding: 14px 14px 10px; flex-shrink: 0; }
.model-demo .dm-side-h .ti-row { display: flex; align-items: center; margin-bottom: 10px; }
.model-demo .dm-side-h .ti {
font-size: 11px; font-family: var(--font-mono);
color: var(--black-alpha-48); letter-spacing: .08em; text-transform: uppercase;
}
.model-demo .dm-side-h .add {
margin-left: auto;
width: 22px; height: 22px;
display: grid; place-items: center;
background: var(--heat-12); color: var(--heat);
border: 0; border-radius: var(--r-sm);
cursor: pointer;
}
.model-demo .dm-side-h .add svg { width: 11px; height: 11px; }
.model-demo .dm-search {
display: flex; align-items: center; gap: 8px;
height: 32px; padding: 0 10px;
background: var(--background-lighter);
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
transition: border-color var(--t-base), background var(--t-base);
}
.model-demo .dm-search:focus-within { border-color: var(--heat-40); background: var(--surface); }
.model-demo .dm-search svg { width: 13px; height: 13px; color: var(--black-alpha-48); flex-shrink: 0; }
.model-demo .dm-search input {
flex: 1; min-width: 0; height: 100%;
border: 0; outline: 0; background: transparent;
font-size: 12.5px; color: var(--accent-black); font-family: inherit;
}
.model-demo .dm-search input::placeholder { color: var(--black-alpha-48); }
.model-demo .dm-prod-list {
flex: 1; min-height: 0;
overflow-y: auto;
padding: 4px 10px 10px;
display: flex; flex-direction: column; gap: 4px;
}
.model-demo .dm-prod {
display: flex; align-items: center; gap: 10px;
padding: 8px;
border: 1px solid transparent;
border-radius: var(--r-sm);
cursor: pointer; text-align: left;
background: transparent; font-family: inherit;
transition: background var(--t-base), border-color var(--t-base);
}
.model-demo .dm-prod:hover { background: var(--background-lighter); }
.model-demo .dm-prod.active { background: var(--heat-12); border-color: var(--heat-20); }
.model-demo .dm-prod .thumb {
flex-shrink: 0;
width: 40px; height: 40px;
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
overflow: hidden;
background: repeating-linear-gradient(135deg, transparent 0 4px, var(--black-alpha-4) 4px 5px);
display: grid; place-items: center;
font-family: var(--font-mono); font-size: 9px; color: var(--black-alpha-32);
}
.model-demo .dm-prod.active .thumb { border-color: var(--heat); }
.model-demo .dm-prod .body { flex: 1; min-width: 0; }
.model-demo .dm-prod .nm {
font-size: 12.5px; color: var(--accent-black); font-weight: 500; line-height: 1.3;
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.model-demo .dm-prod.active .nm { color: var(--heat); font-weight: 600; }
.model-demo .dm-prod .sub {
margin-top: 2px;
font-family: var(--font-mono); font-size: 10px;
color: var(--black-alpha-48); letter-spacing: .02em;
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.model-demo .dm-all {
flex-shrink: 0;
margin: 0 10px 12px;
padding: 10px 12px;
background: var(--background-lighter);
border: 1px dashed var(--border-faint);
border-radius: var(--r-sm);
color: var(--black-alpha-72);
font-size: 12px; font-family: inherit; cursor: pointer;
display: flex; align-items: center; gap: 6px;
transition: border-color var(--t-base), color var(--t-base), background var(--t-base);
}
.model-demo .dm-all:hover { border-color: var(--heat); color: var(--heat); background: var(--heat-12); }
.model-demo .dm-all .ct {
color: var(--black-alpha-48);
font-family: var(--font-mono); font-size: 10.5px;
margin-left: auto;
}
.model-demo .dm-all svg { width: 12px; height: 12px; }
.model-demo .dm-all .arrow { margin-left: 4px; }
/* 主区 · 返回 pill(基线无,挂主区头部,保留 onBack) */
.model-demo .dm-back {
flex-shrink: 0;
display: inline-flex; align-items: center; gap: 6px;
height: 30px; padding: 0 12px 0 10px;
background: var(--surface);
border: 1px solid var(--border-faint);
border-radius: var(--r-pill);
color: var(--accent-black);
font-size: 12.5px; font-weight: 500; font-family: inherit;
cursor: pointer;
transition: background var(--t-base), border-color var(--t-base), color var(--t-base);
}
.model-demo .dm-back:hover { background: var(--black-alpha-4); border-color: var(--black-alpha-24); }
.model-demo .dm-back svg { width: 14px; height: 14px; }
/* ── 主区(两版共用骨架) ── */
.model-demo .dm-main {
display: flex; flex-direction: column;
min-height: 0; overflow: hidden;
position: relative;
}
/* 批次卡(两版共用) */
.model-demo .dm-batch {
background: var(--surface);
border: 1px solid var(--border-faint);
border-radius: var(--r-md);
padding: 14px 16px;
margin-bottom: 14px;
}
.model-demo .dm-batch-h .pic {
flex-shrink: 0;
width: 28px; height: 28px;
background: var(--background-lighter);
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
display: grid; place-items: center;
color: var(--heat);
font-family: var(--font-mono); font-size: 11px; font-weight: 600;
}
.model-demo .dm-batch-h .meta { flex: 1; min-width: 0; }
.model-demo .dm-batch-h .nm { font-size: 13px; font-weight: 600; color: var(--accent-black); }
.model-demo .dm-batch-h .info { margin-top: 2px; font-family: var(--font-mono); font-size: 10.5px; color: var(--black-alpha-48); letter-spacing: .02em; }
.model-demo .dm-batch-h .info .sep { color: var(--black-alpha-24); }
.model-demo .dm-batch-h .ops { display: flex; gap: 4px; }
.model-demo .dm-batch-h .ops button {
width: 28px; height: 28px;
background: var(--surface);
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
color: var(--black-alpha-56); cursor: pointer;
display: grid; place-items: center;
transition: border-color var(--t-base), color var(--t-base);
}
.model-demo .dm-batch-h .ops button:hover { border-color: var(--heat-20); color: var(--heat); }
.model-demo .dm-batch-h .ops button svg { width: 13px; height: 13px; }
.model-demo .dm-batch-grid { display: grid; grid-template-columns: repeat(4, 1fr); gap: 10px; }
@media (max-width: 1400px) { .model-demo .dm-batch-grid { grid-template-columns: repeat(3, 1fr); } }
.model-demo .dm-cell {
aspect-ratio: 3 / 4;
background: var(--background-lighter);
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
overflow: hidden; position: relative; cursor: pointer;
}
.model-demo .dm-cell .ph {
position: absolute; inset: 0;
display: grid; place-items: center;
font-family: var(--font-mono); font-size: 11px; color: var(--black-alpha-32);
background: repeating-linear-gradient(135deg, transparent 0 6px, var(--black-alpha-3) 6px 7px);
}
.model-demo .dm-cell .tag {
position: absolute; top: 6px; left: 6px;
padding: 2px 6px;
background: var(--accent-black);
color: var(--accent-white);
border-radius: var(--r-sm);
font-size: 10px; font-weight: 500;
}
/*
方案 A · 参数面板 + 结果双栏
*/
.model-demo.dm-a .dm-main-h {
flex-shrink: 0;
display: flex; align-items: center; gap: 14px;
padding: 14px 28px;
border-bottom: 1px solid var(--border-faint);
background: var(--surface);
}
.model-demo.dm-a .dm-main-h .cur { min-width: 0; }
.model-demo.dm-a .dm-main-h .crumb {
font-family: var(--font-mono); font-size: 10.5px;
color: var(--black-alpha-48); letter-spacing: .04em;
}
.model-demo.dm-a .dm-main-h h2 {
font-size: 20px; font-weight: 600;
letter-spacing: -.015em; color: var(--accent-black);
}
.model-demo.dm-a .dm-main-h .stats {
margin-left: auto;
display: flex; gap: 6px;
font-family: var(--font-mono); font-size: 11px;
color: var(--black-alpha-48); letter-spacing: .02em;
}
.model-demo.dm-a .dm-main-h .stats b { color: var(--accent-black); font-weight: 600; }
.model-demo.dm-a .dm-main-h .stats .sep { color: var(--black-alpha-24); }
.model-demo.dm-a .dm-body {
flex: 1; min-height: 0;
display: grid; grid-template-columns: 320px minmax(0, 1fr);
}
@media (max-width: 1100px) { .model-demo.dm-a .dm-body { grid-template-columns: 280px minmax(0, 1fr); } }
/* 左侧参数面板 */
.model-demo.dm-a .dm-form {
border-right: 1px solid var(--border-faint);
background: var(--surface);
display: flex; flex-direction: column;
min-height: 0;
}
.model-demo.dm-a .dm-form-scroll { flex: 1; min-height: 0; overflow-y: auto; padding: 16px 18px; }
.model-demo.dm-a .dm-field { margin-bottom: 16px; }
.model-demo.dm-a .dm-field:last-child { margin-bottom: 0; }
.model-demo.dm-a .dm-field-h { font-size: 12px; font-weight: 600; color: var(--accent-black); margin-bottom: 8px; }
.model-demo.dm-a .dm-field-h .opt { font-weight: 400; font-size: 11px; color: var(--black-alpha-48); margin-left: 4px; }
.model-demo.dm-a .dm-models { display: grid; grid-template-columns: repeat(3, 1fr); gap: 8px; }
.model-demo.dm-a .dm-model {
aspect-ratio: 3 / 4;
border: 1px solid var(--border-faint);
background: var(--background-lighter);
border-radius: var(--r-sm);
position: relative; cursor: pointer; overflow: hidden;
transition: border-color var(--t-base);
}
.model-demo.dm-a .dm-model:hover { border-color: var(--black-alpha-32); }
.model-demo.dm-a .dm-model.selected { border-color: var(--heat); border-width: 2px; }
.model-demo.dm-a .dm-model .ph {
position: absolute; inset: 0;
display: grid; place-items: center;
font-family: var(--font-mono); font-size: 10px; color: var(--black-alpha-32);
background: repeating-linear-gradient(135deg, transparent 0 6px, var(--black-alpha-3) 6px 7px);
}
.model-demo.dm-a .dm-model .nm {
position: absolute; bottom: 0; left: 0; right: 0;
padding: 4px 6px;
background: linear-gradient(transparent, var(--black-alpha-48));
font-size: 10px; color: var(--accent-white); font-weight: 500;
}
.model-demo.dm-a .dm-model.selected::after {
content: ''; position: absolute; top: 4px; right: 4px;
width: 14px; height: 14px;
background: var(--heat) url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='none' stroke='%23fff' stroke-width='3' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M20 6 9 17l-5-5'/%3E%3C/svg%3E") no-repeat center / 9px;
border-radius: 50%;
}
.model-demo.dm-a .dm-model.add {
border-style: dashed;
display: flex; align-items: center; justify-content: center;
color: var(--black-alpha-48);
}
.model-demo.dm-a .dm-chip-row { display: flex; flex-wrap: wrap; gap: 6px; }
.model-demo.dm-a .dm-chip {
display: inline-flex; align-items: center;
height: 28px; padding: 0 11px;
background: var(--surface);
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
font-size: 12px; color: var(--black-alpha-72);
font-family: inherit; cursor: pointer;
transition: border-color var(--t-base), color var(--t-base), background var(--t-base);
}
.model-demo.dm-a .dm-chip:hover { border-color: var(--black-alpha-32); color: var(--accent-black); }
.model-demo.dm-a .dm-chip.active { background: var(--heat-12); color: var(--heat); border-color: var(--heat-20); font-weight: 600; }
.model-demo.dm-a .dm-textarea {
width: 100%; min-height: 60px;
padding: 8px 10px;
background: var(--background-lighter);
border: 1px solid var(--black-alpha-12);
border-radius: var(--r-sm);
font-family: inherit; font-size: 12.5px;
color: var(--accent-black);
outline: none; resize: vertical;
}
.model-demo.dm-a .dm-textarea::placeholder { color: var(--black-alpha-48); }
.model-demo.dm-a .dm-textarea:focus { border-color: var(--heat-40); background: var(--surface); }
.model-demo.dm-a .dm-form-cta {
flex-shrink: 0;
padding: 14px 18px;
border-top: 1px solid var(--border-faint);
background: var(--surface);
}
.model-demo.dm-a .dm-cost {
display: flex; align-items: center; justify-content: space-between;
margin-bottom: 10px;
font-family: var(--font-mono); font-size: 11px;
color: var(--black-alpha-48); letter-spacing: .02em;
}
.model-demo.dm-a .dm-cost .v { color: var(--accent-black); font-weight: 600; }
.model-demo.dm-a .dm-gen {
width: 100%; height: 42px;
background: var(--heat); color: var(--accent-white);
border: 1px solid var(--heat); border-radius: var(--r-md);
font-size: 14px; font-weight: 600; font-family: inherit;
cursor: pointer;
display: inline-flex; align-items: center; justify-content: center; gap: 8px;
box-shadow: var(--shadow-cta);
}
.model-demo.dm-a .dm-gen svg { width: 15px; height: 15px; }
/* 右侧结果区 */
.model-demo.dm-a .dm-result {
background: var(--background-base);
min-height: 0; overflow-y: auto;
padding: 22px 24px;
}
.model-demo.dm-a .dm-result-h { display: flex; align-items: baseline; gap: 10px; margin-bottom: 14px; }
.model-demo.dm-a .dm-result-h .ti { font-size: 15px; font-weight: 600; color: var(--accent-black); }
.model-demo.dm-a .dm-result-h .sub { font-family: var(--font-mono); font-size: 10.5px; color: var(--black-alpha-48); letter-spacing: .02em; }
.model-demo.dm-a .dm-batch-h {
display: flex; align-items: center; gap: 10px;
margin-bottom: 12px; padding-bottom: 10px;
border-bottom: 1px solid var(--border-faint);
}
/*
方案 B · v2:任务流主区 + 底部 fixed 参数栏
*/
.model-demo.dm-b .dm-main-h {
flex-shrink: 0;
padding: 16px 28px 12px;
border-bottom: 1px solid var(--border-faint);
background: var(--surface);
}
.model-demo.dm-b .dm-main-h .crumb {
font-family: var(--font-mono); font-size: 10.5px;
color: var(--black-alpha-48); letter-spacing: .04em; margin-bottom: 4px;
}
.model-demo.dm-b .dm-main-h .title-row { display: flex; align-items: center; gap: 14px; }
.model-demo.dm-b .dm-main-h h2 {
font-size: 22px; font-weight: 600;
letter-spacing: -.015em; color: var(--accent-black);
}
.model-demo.dm-b .dm-main-h .title-row .dm-back { margin-left: auto; }
.model-demo.dm-b .dm-main-h .row { display: flex; align-items: center; gap: 16px; margin-top: 6px; }
.model-demo.dm-b .dm-main-h .stats {
font-family: var(--font-mono); font-size: 11px;
color: var(--black-alpha-48); letter-spacing: .02em;
display: flex; gap: 4px;
}
.model-demo.dm-b .dm-main-h .stats b { color: var(--accent-black); font-weight: 600; }
.model-demo.dm-b .dm-main-h .stats .sep { color: var(--black-alpha-24); }
.model-demo.dm-b .dm-main-h .spacer { flex: 1; }
.model-demo.dm-b .dm-tb { display: flex; gap: 8px; align-items: center; }
.model-demo.dm-b .dm-tb .icbtn {
width: 30px; height: 30px;
background: var(--surface);
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
color: var(--black-alpha-72);
cursor: pointer; display: grid; place-items: center;
transition: border-color var(--t-base), color var(--t-base);
}
.model-demo.dm-b .dm-tb .icbtn:hover { border-color: var(--heat-20); color: var(--heat); }
.model-demo.dm-b .dm-tb .icbtn svg { width: 13px; height: 13px; }
.model-demo.dm-b .dm-tb .chip {
display: inline-flex; align-items: center; gap: 6px;
height: 30px; padding: 0 10px;
background: var(--surface);
border: 1px solid var(--border-faint);
border-radius: var(--r-sm);
font-size: 12px; color: var(--black-alpha-72);
font-family: inherit; cursor: pointer;
transition: border-color var(--t-base), color var(--t-base);
}
.model-demo.dm-b .dm-tb .chip:hover { border-color: var(--heat-20); color: var(--heat); }
.model-demo.dm-b .dm-tb .chip svg { width: 10px; height: 10px; opacity: .6; }
.model-demo.dm-b .dm-stream {
flex: 1; min-height: 0;
overflow-y: auto;
padding: 22px 28px 200px;
background: var(--background-base);
}
.model-demo.dm-b .dm-day-h {
display: flex; align-items: baseline; gap: 8px;
margin: 6px 0 10px;
font-family: var(--font-mono); font-size: 10.5px;
color: var(--black-alpha-48); letter-spacing: .06em; text-transform: uppercase;
}
.model-demo.dm-b .dm-day-h::before {
content: ''; width: 14px; height: 1px;
background: var(--black-alpha-24);
display: inline-block; margin-right: 2px;
}
.model-demo.dm-b .dm-day-h .ct {
color: var(--black-alpha-72); font-weight: 500;
margin-left: auto; text-transform: none; letter-spacing: 0;
}
.model-demo.dm-b .dm-batch-h { display: flex; align-items: center; gap: 10px; margin-bottom: 12px; }
.model-demo.dm-b .dm-batch-h .nm { font-size: 13.5px; }
.model-demo.dm-b .dm-batch-h .info { display: flex; flex-wrap: wrap; gap: 4px; }
/* 状态 pill 紧贴标题右侧(局部尺寸变体 · 不改全局 .pill) */
.model-demo.dm-b .stat-pill {
margin-left: 8px;
padding: 2px 7px;
font-size: 10px;
}
.model-demo.dm-b .stat-pill .dot { width: 4px; height: 4px; }
.model-demo.dm-b .dm-cell.gen .ph { animation: dm-pulse 1.4s ease-in-out infinite; }
@keyframes dm-pulse { 0%, 100% { opacity: 1; } 50% { opacity: .55; } }
.model-demo.dm-b .dm-cell.err { border-color: var(--accent-crimson); }
.model-demo.dm-b .dm-cell.err .ph { color: var(--accent-crimson); background: var(--crimson-bg); }
/* 底部 fixed 参数面板 */
.model-demo.dm-b .dm-param-wrap {
position: absolute; left: 0; right: 0; bottom: 0;
padding: 14px 28px 22px;
background: linear-gradient(to bottom, transparent 0, var(--background-base) 24px);
z-index: 5;
}
.model-demo.dm-b .dm-param {
max-width: 1180px; margin: 0 auto;
background: var(--surface);
border: 1px solid var(--border-faint);
border-radius: var(--r-md);
padding: 10px 14px;
display: flex; align-items: center; gap: 10px;
box-shadow: var(--shadow-floating);
}
.model-demo.dm-b .dm-param .pchip {
display: inline-flex; align-items: center; gap: 6px;
height: 30px; padding: 0 12px;
background: var(--background-lighter);
border: 1px solid transparent;
border-radius: var(--r-pill);
font-size: 12px; color: var(--black-alpha-72);
cursor: pointer; font-family: inherit;
transition: border-color var(--t-base), background var(--t-base), color var(--t-base);
}
.model-demo.dm-b .dm-param .pchip:hover { background: var(--surface); border-color: var(--border-faint); }
.model-demo.dm-b .dm-param .pchip.active { background: var(--heat-12); color: var(--heat); }
.model-demo.dm-b .dm-param .pchip svg { width: 10px; height: 10px; opacity: .6; }
.model-demo.dm-b .dm-param .pchip .lbl-mono {
font-family: var(--font-mono); font-size: 10px;
color: var(--black-alpha-48); letter-spacing: .02em;
}
.model-demo.dm-b .dm-param .pchip.active .lbl-mono { color: var(--heat); }
.model-demo.dm-b .dm-param .pchip .muted { color: var(--black-alpha-48); }
.model-demo.dm-b .dm-param .spacer { flex: 1; }
.model-demo.dm-b .dm-param .meta-right {
font-family: var(--font-mono); font-size: 10.5px;
color: var(--black-alpha-48); letter-spacing: .02em;
margin-right: 6px; text-align: right;
}
.model-demo.dm-b .dm-param .meta-right .v { color: var(--accent-black); font-weight: 600; }
.model-demo.dm-b .dm-param .gen-btn {
height: 38px; padding: 0 20px;
background: var(--heat); color: var(--accent-white);
border: 0; border-radius: var(--r-md);
font-size: 13.5px; font-weight: 600; font-family: inherit;
cursor: pointer;
display: inline-flex; align-items: center; gap: 8px;
box-shadow: var(--shadow-cta);
}
.model-demo.dm-b .dm-param .gen-btn svg { width: 14px; height: 14px; }

View File

@ -2,11 +2,13 @@ import { useEffect, useMemo, useState } from "react";
import {
ArrowLeft,
ArrowRight,
Bookmark,
Check,
ChevronDown,
Download,
Grid2X2,
ImagePlus,
LayoutGrid,
List,
MoreHorizontal,
Plus,
@ -14,7 +16,9 @@ import {
RefreshCw,
Search,
Sparkles,
WandSparkles
Trash2,
WandSparkles,
X
} from "lucide-react";
import type { AITask, Asset, ModelConfig, Product } from "../types";
import type { Page } from "./route-config";
@ -845,6 +849,474 @@ function Pill({
);
}
/*
Demo( · )
variant="A" public/exact/model-photo-demo-a.html
variant="B" public/exact/model-photo-demo-b.html
= ( + 6 + ), products ,退线
(A:参数面板 + ;B:任务流 + fixed )
*/
/* 基线左栏占位商品(无真 products 时兜底) */
const DEMO_SIDE_PRODUCTS = [
{ id: "d1", title: "透真补水面膜", category: "美妆个护", batches: 6 },
{ id: "d2", title: "透真清透防晒霜", category: "美妆个护", batches: 3 },
{ id: "d3", title: "南卡 Lite Pro 蓝牙耳机", category: "数码 3C", batches: 2 },
{ id: "d4", title: "滋啦速食牛肉面", category: "食品饮料", batches: 1 },
{ id: "d5", title: "三顿半同款冻干咖啡", category: "食品饮料", batches: 1 },
{ id: "d6", title: "小熊 4L 可视空气炸锅", category: "家居家电", batches: 0 }
];
export function ModelPhotoDemoPage({ variant, products, onBack }: { variant: "A" | "B"; products: Product[]; onBack: () => void }) {
return <><div className="page-head"><div><h1> {variant}</h1><div className="sub"><span className="mono">// model-photo-demo-{variant.toLowerCase()}</span> · 原型对比页补齐</div></div><div className="actions"><button className="btn" type="button" onClick={onBack}><ArrowLeft size={13} />返回模特图</button></div></div><div className={`demo-layout demo-${variant.toLowerCase()}`}><aside className="tool-rail"><div className="rail-h">商品空间</div>{products.slice(0, 6).map((product) => <button className="rail-product" type="button" key={product.id}><div className="placeholder"><span className="ph-frame">{product.title.slice(0, 4)}</span></div><span>{product.title}</span></button>)}</aside><section className="pane"><div className="pane-h"><strong>方案 {variant}</strong><span className="spacer" /><span className="mono muted-2">DEMO</span></div><div className="result-board dense">{Array.from({ length: 6 }).map((_, index) => <div className="result-tile" key={index}><div className="placeholder"><span className="ph-frame">MODEL {index + 1}</span></div></div>)}</div></section></div></>;
// 左栏商品空间:优先真 products(最近 6 条),空则回退基线占位
const sideProducts =
products.length > 0
? products.slice(0, 6).map((item, index) => ({
id: item.id,
title: item.title,
category: item.category || "未分类",
batches: DEMO_SIDE_PRODUCTS[index % DEMO_SIDE_PRODUCTS.length].batches
}))
: DEMO_SIDE_PRODUCTS;
const [activeId, setActiveId] = useState(sideProducts[0].id);
const active = sideProducts.find((item) => item.id === activeId) || sideProducts[0];
const totalCount = products.length > 0 ? products.length : 24;
// 共享左栏(两版一致)
const side = (
<aside className="dm-side">
<div className="dm-side-h">
<div className="ti-row">
<span className="ti"></span>
<button className="add" type="button" title="新建商品">
<Plus size={11} />
</button>
</div>
<div className="dm-search">
<Search size={13} />
<input type="text" placeholder="搜索商品 / 分类" />
</div>
</div>
<div className="dm-prod-list">
{sideProducts.map((item) => (
<button
type="button"
key={item.id}
className={`dm-prod ${item.id === active.id ? "active" : ""}`}
onClick={() => setActiveId(item.id)}
>
<div className="thumb"></div>
<div className="body">
<div className="nm">{item.title}</div>
<div className="sub">// {item.category} · {item.batches} 批</div>
</div>
</button>
))}
</div>
<button className="dm-all" type="button">
<LayoutGrid size={12} />
<span className="ct">{totalCount} </span>
<ArrowRight className="arrow" size={12} />
</button>
</aside>
);
// ── 返回(基线无,挂在主区头部,保留 onBack)──
const backBtn = (
<button className="dm-back" type="button" onClick={onBack}>
<ArrowLeft size={14} />
</button>
);
if (variant === "A") {
return (
<div className="model-demo dm-a">
<div className="dm-banner">
// DEMO · 方案 A · <b>商品 = 项目空间</b>。左栏仅商品空间:搜索 + 最近 6 条 + <b>全部商品</b>兜底入口;
主区:模特卡 + + + ,
</div>
<div className="dm-grid">
{side}
<section className="dm-main">
{/* 顶部 · 商品名 + 统计 */}
<div className="dm-main-h">
<div className="cur">
<div className="crumb">// 商品空间</div>
<h2>{active.title}</h2>
</div>
<div className="stats">
<span> <b>6</b> </span>
<span className="sep">·</span>
<span> <b>22</b> </span>
<span className="sep">·</span>
<span> <b>3 </b></span>
</div>
{backBtn}
</div>
{/* 参数面板 + 结果双栏 */}
<div className="dm-body">
{/* 左 · 参数面板 */}
<div className="dm-form">
<div className="dm-form-scroll">
<div className="dm-field">
<div className="dm-field-h">
<span className="opt">( · {active.title})</span>
</div>
<div className="dm-models">
{["Ava", "Zoe", "Ben", "Lin", "Mia"].map((name, index) => (
<div className={`dm-model ${index === 0 ? "selected" : ""}`} key={name}>
<div className="ph">{name} · 3:4</div>
<div className="nm">{name}</div>
</div>
))}
<div className="dm-model add">
<Plus size={18} />
</div>
</div>
</div>
<div className="dm-field">
<div className="dm-field-h"></div>
<div className="dm-chip-row">
{["1 张", "2 张", "4 张", "8 张"].map((label) => (
<button type="button" key={label} className={`dm-chip ${label === "4 张" ? "active" : ""}`}>
{label}
</button>
))}
</div>
</div>
<div className="dm-field">
<div className="dm-field-h"></div>
<div className="dm-chip-row">
{["1:1", "3:4", "9:16", "16:9"].map((label) => (
<button type="button" key={label} className={`dm-chip ${label === "3:4" ? "active" : ""}`}>
{label}
</button>
))}
</div>
</div>
<div className="dm-field">
<div className="dm-field-h"><span className="opt">()</span></div>
<textarea className="dm-textarea" placeholder="例:户外阳光、敷面膜的特写、白底产品摄影" />
</div>
</div>
<div className="dm-form-cta">
<div className="dm-cost">
<span> <span className="v"> ¥1.20</span></span>
<span> ¥327.40</span>
</div>
<button className="dm-gen" type="button">
<Sparkles size={15} />
· {active.title} × Ava
</button>
</div>
</div>
{/* 右 · 结果(当前商品全部批次) */}
<div className="dm-result">
<div className="dm-result-h">
<span className="ti"> · Ava × 4 </span>
<span className="sub">// 3 分钟前 · 已完成</span>
</div>
{/* 批次 1 */}
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">4×</div>
<div className="meta">
<div className="nm">Ava × 4 </div>
<div className="info">
{active.title} <span className="sep">·</span> 3:4 <span className="sep">·</span> 3 {" "}
<span className="sep">·</span> ¥1.20
</div>
</div>
<div className="ops">
<button type="button" title="全部重跑"><RefreshCw size={13} /></button>
<button type="button" title="全部下载"><Download size={13} /></button>
<button type="button" title="加入资产库"><Bookmark size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
{[1, 2, 3, 4].map((n) => (
<div className="dm-cell" key={n}>
<div className="ph">Ava · #{n}</div>
<span className="tag">3:4</span>
</div>
))}
</div>
</div>
{/* 批次 2 */}
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">4×</div>
<div className="meta">
<div className="nm">Zoe × 4 </div>
<div className="info">
{active.title} <span className="sep">·</span> 3:4 <span className="sep">·</span> 12 {" "}
<span className="sep">·</span> ¥1.20
</div>
</div>
<div className="ops">
<button type="button" title="全部重跑"><RefreshCw size={13} /></button>
<button type="button" title="全部下载"><Download size={13} /></button>
<button type="button" title="加入资产库"><Bookmark size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
{[1, 2, 3, 4].map((n) => (
<div className="dm-cell" key={n}>
<div className="ph">Zoe · #{n}</div>
<span className="tag">3:4</span>
</div>
))}
</div>
</div>
{/* 批次 3 · 生成中 */}
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">2×</div>
<div className="meta">
<div className="nm">Ben × 2 </div>
<div className="info">
{active.title} <span className="sep">·</span> 3:4 <span className="sep">·</span> {" "}
<span className="sep">·</span>
</div>
</div>
<div className="ops">
<button type="button" title="取消"><X size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
<div className="dm-cell"><div className="ph"></div></div>
<div className="dm-cell"><div className="ph"></div></div>
</div>
</div>
</div>
</div>
</section>
</div>
</div>
);
}
// ── variant === "B" · v2:任务流主区 + 底部 fixed 参数栏 ──
return (
<div className="model-demo dm-b">
<div className="dm-banner">
// DEMO v2 · 方案 A · <b>商品空间 + 任务流主区</b>。左栏只保留商品空间(搜索+最近6条+全部入口),
, toolbar, fixed ( image-optimize)
</div>
<div className="dm-grid">
{side}
<section className="dm-main">
{/* 顶部 标题 + stats + toolbar */}
<div className="dm-main-h">
<div className="crumb">// 商品空间 · 模特上身图</div>
<div className="title-row">
<h2>{active.title}</h2>
{backBtn}
</div>
<div className="row">
<div className="stats">
<span>{active.category}</span><span className="sep">·</span>
<span> <b>6</b> </span><span className="sep">·</span>
<span> <b>22</b> </span><span className="sep">·</span>
<span> <b>3 </b></span>
</div>
<span className="spacer" />
<div className="dm-tb">
<button className="icbtn" type="button" title="搜索批次"><Search size={13} /></button>
<button className="chip" type="button"> <ChevronDown size={10} /></button>
<button className="chip" type="button"> <ChevronDown size={10} /></button>
<button className="chip" type="button"> <ChevronDown size={10} /></button>
</div>
</div>
</div>
{/* 任务流 */}
<div className="dm-stream">
<div className="dm-day-h">
<span></span>
<span className="ct">3 · 10 </span>
</div>
{/* 批次 1 */}
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">4×</div>
<div className="meta">
<div className="nm">Ava × 4 <span className="pill ok stat-pill"><span className="dot" /></span></div>
<div className="info">
<span>3:4</span><span className="sep">·</span>
<span>3 </span><span className="sep">·</span>
<span>¥1.20</span>
</div>
</div>
<div className="ops">
<button type="button" title="全部重跑"><RefreshCw size={13} /></button>
<button type="button" title="全部下载"><Download size={13} /></button>
<button type="button" title="加入资产库"><Bookmark size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
{[1, 2, 3, 4].map((n) => (
<div className="dm-cell" key={n}><div className="ph">Ava · #{n}</div><span className="tag">3:4</span></div>
))}
</div>
</div>
{/* 批次 2 */}
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">4×</div>
<div className="meta">
<div className="nm">Zoe × 4 <span className="pill ok stat-pill"><span className="dot" /></span></div>
<div className="info">
<span>3:4</span><span className="sep">·</span>
<span>12 </span><span className="sep">·</span>
<span>¥1.20</span>
</div>
</div>
<div className="ops">
<button type="button" title="全部重跑"><RefreshCw size={13} /></button>
<button type="button" title="全部下载"><Download size={13} /></button>
<button type="button" title="加入资产库"><Bookmark size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
{[1, 2, 3, 4].map((n) => (
<div className="dm-cell" key={n}><div className="ph">Zoe · #{n}</div><span className="tag">3:4</span></div>
))}
</div>
</div>
{/* 批次 3 · 生成中 */}
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">2×</div>
<div className="meta">
<div className="nm">Ben × 2 <span className="pill info stat-pill"><span className="dot" /></span></div>
<div className="info">
<span>3:4</span><span className="sep">·</span>
<span></span><span className="sep">·</span>
<span>¥0.60</span>
</div>
</div>
<div className="ops">
<button type="button" title="取消"><X size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
<div className="dm-cell gen"><div className="ph"></div></div>
<div className="dm-cell gen"><div className="ph"></div></div>
</div>
</div>
{/* 昨天 */}
<div className="dm-day-h">
<span></span>
<span className="ct">2 · 8 </span>
</div>
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">4×</div>
<div className="meta">
<div className="nm">Lin × 4 <span className="pill ok stat-pill"><span className="dot" /></span></div>
<div className="info">
<span>3:4</span><span className="sep">·</span>
<span> 18:24</span><span className="sep">·</span>
<span>¥1.20</span>
</div>
</div>
<div className="ops">
<button type="button" title="全部重跑"><RefreshCw size={13} /></button>
<button type="button" title="全部下载"><Download size={13} /></button>
<button type="button" title="加入资产库"><Bookmark size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
{[1, 2, 3, 4].map((n) => (
<div className="dm-cell" key={n}><div className="ph">Lin · #{n}</div><span className="tag">3:4</span></div>
))}
</div>
</div>
{/* 更早 */}
<div className="dm-day-h">
<span></span>
<span className="ct">1 · 2 · 1 </span>
</div>
<div className="dm-batch">
<div className="dm-batch-h">
<div className="pic">2×</div>
<div className="meta">
<div className="nm">Ava × 2 <span className="pill err stat-pill"><span className="dot" /></span></div>
<div className="info">
<span>3:4</span><span className="sep">·</span>
<span>2 </span>
</div>
</div>
<div className="ops">
<button type="button" title="全部重跑"><RefreshCw size={13} /></button>
<button type="button" title="删除"><Trash2 size={13} /></button>
</div>
</div>
<div className="dm-batch-grid">
<div className="dm-cell err"><div className="ph"> · </div></div>
<div className="dm-cell err"><div className="ph"> · </div></div>
</div>
</div>
</div>
{/* 底部 fixed 参数面板 */}
<div className="dm-param-wrap">
<div className="dm-param">
<button className="pchip active" type="button">
<span className="lbl-mono"></span>
<span>Ava</span>
<ChevronDown size={10} />
</button>
<button className="pchip" type="button">
<span className="lbl-mono"></span>
<span>4</span>
<ChevronDown size={10} />
</button>
<button className="pchip" type="button">
<span className="lbl-mono"></span>
<span>3:4</span>
<ChevronDown size={10} />
</button>
<button className="pchip" type="button">
<span className="lbl-mono"></span>
<span className="muted">+ </span>
</button>
<span className="spacer" />
<span className="meta-right"> <span className="v">¥1.20</span> · <span className="v">¥327.40</span></span>
<button className="gen-btn" type="button">
<Sparkles size={14} />
· {active.title} × Ava
</button>
</div>
</div>
</section>
</div>
</div>
);
}

View File

@ -0,0 +1,19 @@
import { chromium } from "playwright";
import { mkdirSync } from "node:fs";
const BASE = "http://127.0.0.1:5180", API = "http://127.0.0.1:8010";
const OUT = process.argv[2] || "shots-aitools";
mkdirSync(OUT, { recursive: true });
const tok = (await (await fetch(`${API}/api/auth/login/`, { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ username: "airshelf", password: "Restraint2026" }) })).json()).token;
const pages = [["model-photo", "/model-photo"], ["platform-cover", "/platform-cover"], ["demo-a", "/model-photo/demo-a"], ["demo-b", "/model-photo/demo-b"]];
const browser = await chromium.launch();
const ctx = await browser.newContext({ viewport: { width: 1440, height: 900 }, deviceScaleFactor: 1 });
await ctx.addInitScript((t) => localStorage.setItem("airshelf_token", t), tok);
const page = await ctx.newPage();
for (const [name, route] of pages) {
await page.goto(BASE + route, { waitUntil: "networkidle", timeout: 30000 });
await page.waitForTimeout(1500);
await page.screenshot({ path: `${OUT}/${name}.png`, fullPage: true });
console.log("shot", name);
}
await browser.close();
console.log("DONE");