From 4cd9ff77d94e089d40a64d2c0a620f350db6eed8 Mon Sep 17 00:00:00 2001 From: kris Date: Tue, 7 Apr 2026 16:17:28 +0800 Subject: [PATCH] feat: surface intake entries inside production center --- CHANGELOG.md | 1 + web/storyforge-web-v4/assets/app.js | 18 +++++++++++++----- .../tests/workbench-pages.test.mjs | 14 ++++++++++++++ 3 files changed, 28 insertions(+), 5 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 779562d..900df2c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -688,3 +688,4 @@ - `integrations/health` 新增 `huobao` 视频配置摘要,能直接看出当前 `Huobao` 视频配置页是否已经录入视频引擎配置,以及对应的配置页路径,减少排查 `Seedance` 任务为什么只建单不出片的歧义。 - 首页 `1 主 2 次` 动作里把 `视频录制` 抬成了高频次级动作,当前项目有生产任务时能更快进入录制维护入口。 - `AI 视频` 表单开始直接显示“当前项目最近使用的视频引擎”,像 `Seedance 2.0 · seedance-2.0-pro` 这类信息会在打开表单时直接可见,并保留跳到火山配置状态的入口。 +- `生产中心` 现在把 `导入主页 / 导入作品 / 导入文本 / 上传视频` 这批接入入口统一接进了顶部动作区和生产队列工作流卡,不用离开生产中心也能开始接入素材。 diff --git a/web/storyforge-web-v4/assets/app.js b/web/storyforge-web-v4/assets/app.js index dd9b77e..51358d9 100644 --- a/web/storyforge-web-v4/assets/app.js +++ b/web/storyforge-web-v4/assets/app.js @@ -7719,9 +7719,10 @@ function renderProductionScreen() { { value: "outputs", label: "作品与产物" } ]; const activeTab = getActiveDetailTab("productionDetailTab", tabs); + const intakeEntryActionsHtml = `${button("导入主页", "open-import-homepage")} ${button("导入作品", "open-import-video-link")} ${button("导入文本", "open-import-text")} ${button("上传视频", "open-upload-video")}`; const productionActionsHtml = isMobileUi - ? `${renderPipelineButton("aiVideo")} ${renderPipelineButton("realCut")} ${button("视频录制", "focus-live-recorder-maintenance", "secondary")} ${button("交给主 Agent", "handoff-to-main-agent", "secondary", { attrs: productionHandoffAttrs })}` - : `${renderPipelineButton("aiVideo")} ${renderPipelineButton("realCut")} ${button("视频录制", "focus-live-recorder-maintenance", "secondary")} ${button("交给主 Agent", "handoff-to-main-agent", "secondary", { attrs: productionHandoffAttrs })} ${button("去复盘", "goto-review", "primary")} ${button("批量恢复", "batch-recover-jobs", "secondary", { disabledReason: recoverableCount ? "" : "当前没有可恢复的失败任务" })}`; + ? `${button("导入作品", "open-import-video-link", "primary")} ${button("导入主页", "open-import-homepage")} ${button("导入文本", "open-import-text")} ${button("上传视频", "open-upload-video")} ${renderPipelineButton("aiVideo")} ${renderPipelineButton("realCut")} ${button("视频录制", "focus-live-recorder-maintenance", "secondary")} ${button("交给主 Agent", "handoff-to-main-agent", "secondary", { attrs: productionHandoffAttrs })}` + : `${button("导入主页", "open-import-homepage")} ${button("导入作品", "open-import-video-link")} ${button("导入文本", "open-import-text")} ${button("上传视频", "open-upload-video")} ${renderPipelineButton("aiVideo")} ${renderPipelineButton("realCut")} ${button("视频录制", "focus-live-recorder-maintenance", "secondary")} ${button("交给主 Agent", "handoff-to-main-agent", "secondary", { attrs: productionHandoffAttrs })} ${button("去复盘", "goto-review", "primary")} ${button("批量恢复", "batch-recover-jobs", "secondary", { disabledReason: recoverableCount ? "" : "当前没有可恢复的失败任务" })}`; return screenShell( "生产中心", "这里已经接上真实任务、失败恢复和知识库文档,适合直接推进生产、恢复和复盘。", @@ -7762,7 +7763,7 @@ function renderProductionScreen() { ? "先确认录制服务和文件状态,再回到队列继续推进。" : activeTab === "outputs" ? "先看产物和作品,再决定是否回到复盘或继续生产。" - : "先看处理中任务,再把异常和产物安排到下一步。" + : "先接入素材,再看处理中任务和异常,把生产链真正跑起来。" )}

${activeTab === "recovery" @@ -7771,7 +7772,7 @@ function renderProductionScreen() { ? `${actionTag("去复盘", "goto-review")} ${actionTag("查看产物", "select-page-tab", `data-page-tab-key="productionDetailTab" data-page-tab-value="outputs"`)}` : activeTab === "recorder" ? `${actionTag("视频录制", "focus-live-recorder-maintenance")} ${actionTag("交给主 Agent", "handoff-to-main-agent", productionHandoffAttrs)}` - : `${actionTag("批量恢复", "batch-recover-jobs")} ${actionTag("交给主 Agent", "handoff-to-main-agent", productionHandoffAttrs)}` + : `${actionTag("导入作品", "open-import-video-link")} ${actionTag("上传视频", "open-upload-video")} ${actionTag("交给主 Agent", "handoff-to-main-agent", productionHandoffAttrs)}` }
@@ -7793,6 +7794,13 @@ function renderProductionScreen() { ${activeTab === "queue" ? `

当前任务

来自 recent_jobs
+
+

接入素材

+

把主页、作品链接、文本素材或本地视频先接进项目,再继续做 AI 视频、实拍剪辑和复盘。

+
+ ${intakeEntryActionsHtml} +
+
${(activeJobs.length ? activeJobs : jobs.slice(0, 4)).map((job) => `
@@ -7806,7 +7814,7 @@ function renderProductionScreen() { ${actionTag("看详情", "open-job-detail", `data-job-id="${escapeHtml(job.id)}"`)}
- `).join("") || `

还没有任务

先去找对标导入内容。

`} + `).join("") || `

还没有任务

先从导入主页、作品、文本或上传视频开始接入素材。

${intakeEntryActionsHtml}
`}
` : activeTab === "recovery" ? ` diff --git a/web/storyforge-web-v4/tests/workbench-pages.test.mjs b/web/storyforge-web-v4/tests/workbench-pages.test.mjs index 2fad29c..3bea86c 100644 --- a/web/storyforge-web-v4/tests/workbench-pages.test.mjs +++ b/web/storyforge-web-v4/tests/workbench-pages.test.mjs @@ -484,12 +484,26 @@ test("mobile discovery and production simplify duplicated top-level actions", () assert.match(discovery, /saveBenchmarkActionName = similarCandidates\.length \? "direct-save-benchmark-link" : "open-benchmark-link"/); assert.match(production, /const isMobileUi = isMobileViewport\(\);/); assert.match(production, /const productionActionsHtml = isMobileUi/); + assert.match(production, /button\("导入主页", "open-import-homepage"\)/); + assert.match(production, /button\("导入作品", "open-import-video-link"/); + assert.match(production, /button\("导入文本", "open-import-text"\)/); + assert.match(production, /button\("上传视频", "open-upload-video"\)/); assert.match(production, /button\("视频录制", "focus-live-recorder-maintenance", "secondary"\)/); assert.match(production, /button\("交给主 Agent", "handoff-to-main-agent"/); assert.match(production, /button\("去复盘", "goto-review", "primary"\)/); assert.match(clickActions, /name === "focus-live-recorder-maintenance"[\s\S]*captureMainAgentLandingContext\(action,\s*"goto-production"\);[\s\S]*focusLiveRecorderMaintenance\(\)/); }); +test("production queue promotes intake entrypoints so import flows are reachable without leaving production", () => { + const production = extractBetween(APP, "function renderProductionScreen()", "function renderReviewScreen()"); + assert.match(production, /const intakeEntryActionsHtml =/); + assert.match(production, /

接入素材<\/h4>/); + assert.match(production, /把主页、作品链接、文本素材或本地视频先接进项目/); + assert.match(production, /先从导入主页、作品、文本或上传视频开始接入素材/); + assert.match(production, /actionTag\("导入作品", "open-import-video-link"\)/); + assert.match(production, /actionTag\("上传视频", "open-upload-video"\)/); +}); + test("discovery page promotes selected-account actions into direct execute flows", () => { const discovery = extractBetween(APP, "function renderDiscoveryScreen()", "function renderTrackingScreen()"); const discoveryOverview = extractBetween(APP, "function renderDiscoveryOverviewSection(", "function renderDiscoveryRelationsSection(");