Compare commits

...

6 Commits

Author SHA1 Message Date
OmniAI Developer 56ed94bf43 fix: remove duplicate workspace activation flag 2026-06-10 10:39:14 +08:00
stringadmin 2509925644 Merge pull request 'feat: 首页响应式视觉升级与全局UI细节打磨' (#30) from feat/home-responsive-polish-and-ui-refinements into master
Reviewed-on: #30
2026-06-10 02:22:18 +00:00
stringadmin 4562243fd7 Merge branch 'master' into feat/home-responsive-polish-and-ui-refinements 2026-06-10 02:22:13 +00:00
ludan 52677e33f1 feat: 首页响应式视觉升级与全局UI细节打磨
本次提交包含以下改进:

## 1. 首页轮播卡片响应式重构 (HomePage.tsx + home.css)
- 将旋转木马卡片偏移量从固定px值改为clamp()流式单位,随视口宽度自适应缩放
- 使用calc(0px - ...)替代乘法计算方向偏移,兼容CSS变量传递
- 轮播舞台新增mask-image渐变遮罩,边缘卡片自然淡出
- 非激活卡片增加saturate/brightness滤镜,强化主次视觉层级
- 激活卡与非激活卡分别设置图片filter效果
- 移除旧carousel-card-label样式
- 多断点适配:1200px/980px/720px/480px逐级调整卡片尺寸和舞台高度

## 2. 首页入口按钮重设计 (HomePage.tsx + home.css)
- 按钮文案从'新手/老手/电商'改为'快速生成/专业创作/电商出图'
- 每个按钮新增small副标题('新手友好'/'画布工作流'/'商品视觉')
- 主按钮(专业创作)使用渐变绿色背景+发光阴影,新建--primary small样式
- 普通按钮玻璃态背景+内阴影,hover绿色边框高亮
- 720px以下单列全宽布局,按钮居中

## 3. 首页全页视觉强化 (home.css)
- Scrim层三重渐变叠加+radial光晕
- Hero区域文字text-shadow + text-wrap: balance排版
- Feature页面::before叠加渐变遮罩
- Feature Visual卡片增加边框/阴影/背景三层嵌套
- Experience区域斜向分割线装饰背景
- Cookie Consent弹窗玻璃态重设计,移动端自适应

## 4. 首页工具盒区域打磨 (toolbox.css)
- 全新CSS变量(--toolbox-radius-card/inner)
- 工具盒整体深色渐变背景+radial光晕
- Shell容器max-width + clamp流式padding
- 左侧品牌区域标题/brand-icon/subtitle重设计
- 工具列表项、工作流卡片统一玻璃态风格
- 工具卡片hover上浮4px+绿色边框+阴影增强
- @media: 1160px/980px/680px/420px四断点响应式

## 5. 工具盒卡片布局简化 (MorePage.tsx + more.css)
- 核心工具卡片移除独立icon区域,改为单列网格布局
- 普通工具卡片隐藏.more-card__icon(近期记录除外)
- 预览图aspect-ratio从16/9改为4/3,内边距优化
- 移动端移除featured-icon相关样式

## 6. 脚本评审Showcase响应式改造 (script-review-showcase.css)
- 主容器从@media切换为@container查询,跟随父容器自适应
- 新增880px/720px/560px三档container断点
- 图表列在720px以下改为水平进度条布局(bar从垂直改水平)
- 图表列增加卡片边框/圆角/背景
- 品牌区域、评分标签、流程卡片逐级压缩
- @media保留外层padding控制

## 7. 通知中心UI修复 (dark-green.css)
- notification-center改为inline-flex定位锚点
- 面板改为absolute+flex列布局,修复定位偏移
- 列表flex自适应高度+overscroll-behavior: contain
- 移动端面板右偏移clamp适配,箭头位置同步
- 高度单位从vh改为dvh,避免移动浏览器地址栏干扰
2026-06-09 14:22:37 +08:00
stringadmin d535d0d74a Merge pull request 'feat: 新增引导式新手指引 (OnboardingTour) 组件,全站页面接入' (#29) from feat/add-onboarding-tour into master
Reviewed-on: #29
2026-06-08 14:31:17 +00:00
OmniAI Developer 6ed65ca3ee feat: 新增引导式新手指引 (OnboardingTour) 组件,全站页面接入 2026-06-08 21:32:17 +08:00
30 changed files with 2427 additions and 249 deletions
Binary file not shown.

After

Width:  |  Height:  |  Size: 729 KiB

+15
View File
@@ -375,6 +375,7 @@ function App() {
const [ecommerceEverMounted, setEcommerceEverMounted] = useState(false);
const [workbenchResetToken, setWorkbenchResetToken] = useState(0);
const [onboardingActive, setOnboardingActive] = useState(false);
const isEcommerceActive = activeView === "ecommerce" || activeView === "ecommerceHub";
useEffect(() => {
if (isEcommerceActive && !ecommerceEverMounted) setEcommerceEverMounted(true);
@@ -473,6 +474,17 @@ function App() {
}
}, [session, setView, setWorkspaceExpanded]);
const handleStartOnboarding = useCallback(() => {
setOnboardingActive(true);
try { window.localStorage.setItem("omniai:onboarding", "1"); } catch {}
handleSetView("workbench");
}, [handleSetView]);
const handleEndOnboarding = useCallback(() => {
setOnboardingActive(false);
try { window.localStorage.removeItem("omniai:onboarding"); } catch {}
}, []);
const clearAuthenticatedState = useCallback((options?: { resetView?: boolean }) => {
clearAllUserStorage();
clearSessionState();
@@ -1320,6 +1332,8 @@ function App() {
key={`workbench-${workbenchResetToken}`}
isAuthenticated={Boolean(session)}
session={session}
onboarding={onboardingActive}
onEndOnboarding={handleEndOnboarding}
onRequireLogin={handleRequireTaskLogin}
onOpenResultInCanvas={handleOpenResultInCanvas}
onRefreshUsage={refreshUsage}
@@ -1330,6 +1344,7 @@ function App() {
return (
<HomePage
onOpenGenerate={() => handleSetView("workbench")}
onStartOnboarding={handleStartOnboarding}
onOpenCanvas={() => handleSetView("canvas")}
onOpenEcommerce={() => handleSetView("ecommerce")}
onOpenScriptReview={() => handleSetView("scriptTokens")}
+500
View File
@@ -0,0 +1,500 @@
import { useCallback, useEffect, useMemo, useRef, useState, type CSSProperties } from "react";
import { createPortal } from "react-dom";
import { CloseOutlined, LeftOutlined, RightOutlined } from "@ant-design/icons";
import "../styles/components/onboarding.css";
// ─── Types ───────────────────────────────────────────────────
export type TourPhaseId = "chat" | "image" | "video";
interface TooltipStep {
target: string;
title: string;
description: string;
/** Which side of the target to place the tooltip on (preferred). */
placement?: "top" | "bottom" | "left" | "right";
/** If true, this step requires the user to interact with the element to proceed. */
interactive?: boolean;
/** Shown as hint text when interactive. */
actionHint?: string;
}
interface TourPhase {
id: TourPhaseId;
label: string;
steps: TooltipStep[];
}
interface OnboardingTourProps {
active: boolean;
phase: TourPhaseId;
stepIndex: number;
onNext: (phase: TourPhaseId, stepIndex: number) => void;
onSkip: (phase: TourPhaseId) => void;
onDone: () => void;
}
// ─── Tour definitions ────────────────────────────────────────
const PHASES: Record<TourPhaseId, TourPhase> = {
chat: {
id: "chat",
label: "对话模式",
steps: [
{
target: "onboarding-chat-upload",
title: "参考素材上传",
description: "点击或拖拽上传图片、视频、音频等参考素材,帮助 AI 更好地理解你的需求。",
placement: "right",
},
{
target: "onboarding-chat-model",
title: "AI 模型选择",
description: "在这里选择对话使用的 AI 模型,不同模型有不同的擅长领域和风格。",
placement: "bottom",
},
{
target: "onboarding-chat-speed",
title: "思考速度",
description: "「思考速度:高」回复更迅速简洁;「思考速度:急速」适合快速问答场景。",
placement: "bottom",
},
{
target: "onboarding-chat-depth",
title: "推理深度",
description: "「推理深度:强」进行更深层逻辑推理;「推理深度:极限」适合复杂多步骤问题。",
placement: "bottom",
},
{
target: "onboarding-chat-input",
title: "提示词输入框",
description: "在这里输入你的问题或创作需求,按 Enter 发送,Shift + Enter 换行。",
placement: "top",
},
{
target: "onboarding-mode-selector",
title: "切换到图像生成模式",
description: "点击「下一步」自动切换,或点击这个按钮手动选择「图像生成」进入下一阶段。",
placement: "bottom",
},
],
},
image: {
id: "image",
label: "图像生成",
steps: [
{
target: "onboarding-image-upload",
title: "参考图上传",
description: "上传参考图片,AI 将基于参考图的风格和内容生成新图像。支持 PNG / JPG / WebP。",
placement: "right",
},
{
target: "onboarding-image-model",
title: "图像模型选择",
description: "选择用于图像生成的 AI 模型,不同模型在风格、精度和速度上有所侧重。",
placement: "bottom",
},
{
target: "onboarding-image-settings",
title: "比例与分辨率",
description: "设置生成图像的宽高比(如 16:9、1:1)和清晰度(1K/2K),根据使用场景选择。",
placement: "bottom",
},
{
target: "onboarding-image-grid",
title: "单图 / 多宫格模式",
description: "「单图」生成一张完整图像;「多宫格」一次生成多张变体供你挑选最佳方案。",
placement: "bottom",
},
{
target: "onboarding-image-input",
title: "图像提示词",
description: "描述你想要的图像内容、风格和细节,越具体效果越好。",
placement: "top",
},
{
target: "onboarding-mode-selector",
title: "切换到视频生成模式",
description: "点击「下一步」自动切换,或点击这个按钮手动选择「视频生成」进入下一阶段。",
placement: "bottom",
},
],
},
video: {
id: "video",
label: "视频生成",
steps: [
{
target: "onboarding-video-upload",
title: "参考素材上传",
description: "上传参考图片或视频片段,帮助 AI 确定视频的风格、色调和内容方向。",
placement: "right",
},
{
target: "onboarding-video-model",
title: "视频模型选择",
description: "选择视频生成模型。不同模型在画质、时长、运动流畅度上各有优势。",
placement: "bottom",
},
{
target: "onboarding-video-frame",
title: "生成方式:全能 / 首尾帧",
description: "「全能参考」根据描述直接生成;「首尾帧」通过设定起始和结束画面精确控制转场。",
placement: "bottom",
},
{
target: "onboarding-video-ratio",
title: "视频画面比例",
description: "选择画面比例。9:16 适合手机短视频(抖音/Reels),16:9 适合横屏展示。",
placement: "bottom",
},
{
target: "onboarding-video-duration",
title: "视频时长设置",
description: "设置生成视频的秒数。时长越长,生成时间越久,建议从 5 秒开始尝试。",
placement: "bottom",
},
{
target: "onboarding-video-quality",
title: "分辨率与画质",
description: "选择视频清晰度。720P 生成更快适合预览,1080P 画质更高适合最终成品。",
placement: "bottom",
},
{
target: "onboarding-video-generate",
title: "一切就绪,开始创作!",
description: "设置完毕后,点击发送按钮(或按 Enter)即可开始你的首次视频生成。祝你创作愉快!",
placement: "top",
},
],
},
};
// ─── Connector line calculation ──────────────────────────────
interface ConnectorPoints {
x1: number; y1: number; // tooltip edge center
x2: number; y2: number; // target edge center
}
function calcConnector(
tooltipRect: DOMRect,
targetRect: DOMRect,
placement: TooltipStep["placement"],
): ConnectorPoints {
const tx = targetRect.left + targetRect.width / 2;
const ty = targetRect.top + targetRect.height / 2;
const tcx = tooltipRect.left + tooltipRect.width / 2;
const tcy = tooltipRect.top + tooltipRect.height / 2;
switch (placement) {
case "top":
return { x1: tcx, y1: tooltipRect.bottom, x2: tx, y2: targetRect.top };
case "bottom":
return { x1: tcx, y1: tooltipRect.top, x2: tx, y2: targetRect.bottom };
case "left":
return { x1: tooltipRect.right, y1: tcy, x2: targetRect.left, y2: ty };
case "right":
return { x1: tooltipRect.left, y1: tcy, x2: targetRect.right, y2: ty };
default:
return { x1: tcx, y1: tooltipRect.top, x2: tx, y2: targetRect.bottom };
}
}
// ─── Placement engine ─────────────────────────────────────────
interface PlacementResult {
left: number;
top: number;
actualPlacement: TooltipStep["placement"];
}
/** Score a candidate — lower is better. Penalises covering the target or overflow. */
function scorePlacement(
left: number, top: number, tw: number, th: number,
targetRect: DOMRect, vw: number, vh: number,
): number {
let score = 0;
// Overflow penalty
if (left < 0) score += Math.abs(left);
if (top < 0) score += Math.abs(top);
if (left + tw > vw) score += (left + tw - vw);
if (top + th > vh) score += (top + th - vh);
// Overlap with target penalty (avoid covering the highlighted element)
const overlapX = Math.max(0, Math.min(left + tw, targetRect.right) - Math.max(left, targetRect.left));
const overlapY = Math.max(0, Math.min(top + th, targetRect.bottom) - Math.max(top, targetRect.top));
if (overlapX > 0 && overlapY > 0) score += overlapX * overlapY * 0.01;
return score;
}
function findBestPlacement(
targetRect: DOMRect, tw: number, th: number,
preferred: TooltipStep["placement"],
): PlacementResult {
const gap = 144;
const vw = window.innerWidth;
const vh = window.innerHeight;
const all: Array<TooltipStep["placement"]> = [
preferred ?? "bottom",
...(["bottom", "top", "right", "left"] as const).filter((p) => p !== (preferred ?? "bottom")),
];
let best: PlacementResult = { left: 0, top: 0, actualPlacement: "bottom" };
let bestScore = Infinity;
for (const p of all) {
let left = 0, top = 0;
switch (p) {
case "bottom":
left = targetRect.left + targetRect.width / 2 - tw / 2;
top = targetRect.bottom + gap;
break;
case "top":
left = targetRect.left + targetRect.width / 2 - tw / 2;
top = targetRect.top - th - gap;
break;
case "right":
left = targetRect.right + gap;
top = targetRect.top + targetRect.height / 2 - th / 2;
break;
case "left":
left = targetRect.left - tw - gap;
top = targetRect.top + targetRect.height / 2 - th / 2;
break;
}
left = Math.max(12, Math.min(left, vw - tw - 12));
top = Math.max(12, Math.min(top, vh - th - 12));
const s = scorePlacement(left, top, tw, th, targetRect, vw, vh);
if (s < bestScore) {
bestScore = s;
best = { left, top, actualPlacement: p };
}
if (s === 0) break; // perfect
}
return best;
}
// ─── Component ────────────────────────────────────────────────
export default function OnboardingTour({
active, phase, stepIndex, onNext, onSkip, onDone,
}: OnboardingTourProps) {
const [pos, setPos] = useState<PlacementResult>({ left: 0, top: 0, actualPlacement: "bottom" });
const [targetRect, setTargetRect] = useState<DOMRect | null>(null);
const [visible, setVisible] = useState(false);
const [connector, setConnector] = useState<ConnectorPoints | null>(null);
const tooltipRef = useRef<HTMLDivElement>(null);
const prevPhaseRef = useRef(phase);
const prevStepRef = useRef(stepIndex);
const phaseDef = PHASES[phase];
const currentStep = phaseDef?.steps[stepIndex];
const totalSteps = phaseDef?.steps.length ?? 0;
const isLastStep = stepIndex >= totalSteps - 1;
const isVideoLastStep = phase === "video" && isLastStep;
const stepChanged = prevPhaseRef.current !== phase || prevStepRef.current !== stepIndex;
prevPhaseRef.current = phase;
prevStepRef.current = stepIndex;
const recalc = useCallback(() => {
if (!currentStep) return;
const el = document.querySelector(`[data-onboarding="${currentStep.target}"]`) as HTMLElement | null;
if (!el) return; // Will be retried by the polling loop
const rect = el.getBoundingClientRect();
setTargetRect(rect);
const tooltip = tooltipRef.current;
if (!tooltip) return;
const tr = tooltip.getBoundingClientRect();
const best = findBestPlacement(rect, tr.width, tr.height, currentStep.placement);
setPos(best);
// Recalculate tooltip rect after position update (use the same best pos)
const virtualTooltipRect = new DOMRect(best.left, best.top, tr.width, tr.height);
setConnector(calcConnector(virtualTooltipRect, rect, best.actualPlacement));
}, [currentStep]);
useEffect(() => {
if (!active) { setVisible(false); return; }
const t = setTimeout(() => { setVisible(true); recalc(); }, 120);
return () => clearTimeout(t);
}, [active, phase, stepIndex, recalc]);
// Reposition and retry when elements aren't ready
useEffect(() => {
if (!active || !visible) return;
const h = () => recalc();
window.addEventListener("resize", h);
window.addEventListener("scroll", h, true);
const obs = new MutationObserver(h);
obs.observe(document.body, { childList: true, subtree: true, attributes: true });
// Polling retry: keep looking for the target element if not found yet
let retryId: number | null = null;
let attempts = 0;
const poll = () => {
recalc();
attempts += 1;
if (attempts < 40) retryId = requestAnimationFrame(poll);
};
// Start polling after a short delay
const startTimer = setTimeout(() => { poll(); }, 200);
return () => {
window.removeEventListener("resize", h);
window.removeEventListener("scroll", h, true);
obs.disconnect();
clearTimeout(startTimer);
if (retryId !== null) cancelAnimationFrame(retryId);
};
}, [active, visible, recalc]);
// Animate in on step change
useEffect(() => {
if (!active || !visible || !stepChanged) return;
const el = tooltipRef.current;
if (!el) return;
el.classList.remove("onboarding-tooltip--pop");
void el.offsetWidth; // force reflow
el.classList.add("onboarding-tooltip--pop");
}, [active, visible, stepChanged, phase, stepIndex]);
if (!active || !currentStep) return null;
const connectorPath = connector
? `M ${connector.x1} ${connector.y1} L ${connector.x2} ${connector.y2}`
: "";
const arrowAngle = connector
? Math.atan2(connector.y2 - connector.y1, connector.x2 - connector.x1) * (180 / Math.PI)
: 0;
const clipPath = targetRect
? `polygon(0% 0%, 0% 100%, ${targetRect.left - 6}px 100%, ${targetRect.left - 6}px ${targetRect.top - 6}px, ${targetRect.right + 6}px ${targetRect.top - 6}px, ${targetRect.right + 6}px ${targetRect.bottom + 6}px, ${targetRect.left - 6}px ${targetRect.bottom + 6}px, ${targetRect.left - 6}px 100%, 100% 100%, 100% 0%)`
: "";
return createPortal(
<div className={`onboarding-root${visible ? " is-visible" : ""}`} aria-label="新手引导教程">
{/* Overlay */}
<div className="onboarding-overlay" style={{ clipPath, WebkitClipPath: clipPath }} />
{/* Spotlight ring */}
{targetRect && (
<div
className="onboarding-spotlight"
style={{
left: targetRect.left - 8,
top: targetRect.top - 8,
width: targetRect.width + 16,
height: targetRect.height + 16,
}}
>
{/* Animated pulse ring */}
<div className="onboarding-spotlight__pulse" />
<div className="onboarding-spotlight__pulse onboarding-spotlight__pulse--delay" />
</div>
)}
{/* Connector SVG line */}
{connector && (
<svg className="onboarding-connector" aria-hidden="true">
<defs>
<linearGradient id="ob-conn-grad" x1="0%" y1="0%" x2="100%" y2="0%">
<stop offset="0%" stopColor="var(--accent, #00ff88)" stopOpacity="0.2" />
<stop offset="100%" stopColor="var(--accent, #00ff88)" stopOpacity="0.9" />
</linearGradient>
</defs>
{/* Animated dash line */}
<path
d={connectorPath}
fill="none"
stroke="var(--accent, #00ff88)"
strokeWidth="2"
strokeDasharray="8 4"
strokeLinecap="round"
opacity="0.7"
className="onboarding-connector__path"
/>
{/* Arrow at target end */}
<circle
cx={connector.x2}
cy={connector.y2}
r="5"
fill="var(--accent, #00ff88)"
className="onboarding-connector__dot"
/>
</svg>
)}
{/* Tooltip card */}
<div
ref={tooltipRef}
className={`onboarding-tooltip onboarding-tooltip--${pos.actualPlacement}`}
style={{ left: pos.left, top: pos.top }}
role="dialog"
aria-label={currentStep.title}
>
{/* Arrow pointing toward target */}
<div
className={`onboarding-tooltip__arrow onboarding-tooltip__arrow--${pos.actualPlacement}`}
style={{ transform: `rotate(${arrowAngle}deg)` }}
/>
<div className="onboarding-tooltip__head">
<span className="onboarding-tooltip__phase-badge">{phaseDef.label}</span>
<span className="onboarding-tooltip__counter">
{stepIndex + 1} / {totalSteps}
</span>
</div>
<strong className="onboarding-tooltip__title">{currentStep.title}</strong>
<p className="onboarding-tooltip__desc">{currentStep.description}</p>
<div className="onboarding-tooltip__actions">
<button type="button" className="onboarding-tooltip__btn onboarding-tooltip__btn--ghost" onClick={onDone}>
<CloseOutlined />
</button>
{stepIndex > 0 && (
<button
type="button"
className="onboarding-tooltip__btn onboarding-tooltip__btn--ghost"
onClick={() => onNext(phase, stepIndex - 1)}
>
<LeftOutlined />
</button>
)}
{isVideoLastStep ? (
<button type="button" className="onboarding-tooltip__btn onboarding-tooltip__btn--primary" onClick={onDone}>
使 <RightOutlined />
</button>
) : isLastStep && phase !== "video" ? (
<button type="button" className="onboarding-tooltip__btn onboarding-tooltip__btn--primary" onClick={() => onSkip(phase)}>
{phase === "chat" ? "进入图像生成" : "进入视频生成"} <RightOutlined />
</button>
) : (
<button type="button" className="onboarding-tooltip__btn onboarding-tooltip__btn--primary" onClick={() => onNext(phase, stepIndex + 1)}>
<RightOutlined />
</button>
)}
</div>
</div>
{/* Bottom progress bar */}
<div className="onboarding-progress" aria-hidden="true">
{(["chat", "image", "video"] as TourPhaseId[]).map((p) => (
<div key={p} className="onboarding-progress__phase">
<div
className={`onboarding-progress__dot${p === phase ? " is-active" : ""}${
(["chat", "image", "video"].indexOf(p) < ["chat", "image", "video"].indexOf(phase)) ? " is-done" : ""
}`}
/>
<span>{PHASES[p].label}</span>
</div>
))}
</div>
</div>,
document.body,
);
}
+21 -2
View File
@@ -10,7 +10,7 @@ import {
SearchOutlined,
UserOutlined,
} from "@ant-design/icons";
import { useCallback, useEffect, useMemo, useRef, useState, type JSX } from "react";
import { useCallback, useEffect, useMemo, useRef, useState, type ChangeEvent, type DragEvent, type JSX } from "react";
import "../../styles/pages/assets.css";
import { assetClient, type ServerAssetItem } from "../../api/assetClient";
import { aiGenerationClient } from "../../api/aiGenerationClient";
@@ -95,6 +95,17 @@ function AssetsPage({ isAuthenticated, onOpenLogin }: AssetsPageProps) {
const [contextMenu, setContextMenu] = useState<{ x: number; y: number; asset: LibraryAssetItem } | null>(null);
const contextMenuRef = useRef<HTMLDivElement>(null);
const uploadInputRef = useRef<HTMLInputElement>(null);
const [isUploadDragging, setIsUploadDragging] = useState(false);
const handleUploadDragOver = (e: DragEvent) => { e.preventDefault(); if (e.dataTransfer.types.includes("Files")) setIsUploadDragging(true); };
const handleUploadDragLeave = (e: DragEvent) => { e.preventDefault(); if (!e.currentTarget.contains(e.relatedTarget as Node)) setIsUploadDragging(false); };
const handleUploadDrop = (e: DragEvent) => {
e.preventDefault();
setIsUploadDragging(false);
if (e.dataTransfer.files.length) {
void handleUploadFiles(e.dataTransfer.files);
}
};
const handleContextMenu = useCallback((e: React.MouseEvent, asset: LibraryAssetItem) => {
e.preventDefault();
@@ -270,7 +281,15 @@ function AssetsPage({ isAuthenticated, onOpenLogin }: AssetsPageProps) {
placeholder="搜索资产..."
/>
</label>
<button type="button" className="studio-generate-btn studio-generate-btn--compact" onClick={() => uploadInputRef.current?.click()} disabled={isUploading}>
<button
type="button"
className={`studio-generate-btn studio-generate-btn--compact${isUploadDragging ? " is-dragging" : ""}`}
onClick={() => uploadInputRef.current?.click()}
onDragOver={handleUploadDragOver}
onDragLeave={handleUploadDragLeave}
onDrop={handleUploadDrop}
disabled={isUploading}
>
{isUploading ? <LoadingOutlined /> : <PlusOutlined />}
{isUploading ? "上传中..." : "添加"}
</button>
@@ -61,6 +61,9 @@ function CharacterMixPage({
const abortRef = useRef(false);
const taskIdRef = useRef<string | null>(null);
const [isDragging, setIsDragging] = useState(false);
const [isCanvasDragging, setIsCanvasDragging] = useState(false);
const characterInputRef = useRef<HTMLInputElement | null>(null);
const videoInputRef = useRef<HTMLInputElement | null>(null);
useEffect(() => {
return () => {
@@ -262,6 +265,23 @@ function CharacterMixPage({
}
};
const handleCanvasDragOver = (e: DragEvent) => { e.preventDefault(); e.stopPropagation(); if (e.dataTransfer.types.includes("Files")) setIsCanvasDragging(true); };
const handleCanvasDragLeave = (e: DragEvent) => { e.preventDefault(); e.stopPropagation(); if (e.currentTarget === e.target || !e.currentTarget.contains(e.relatedTarget as Node)) setIsCanvasDragging(false); };
const handleCanvasDrop = (e: DragEvent) => {
e.preventDefault();
e.stopPropagation();
setIsCanvasDragging(false);
handleDrop(e);
};
const handleCanvasClick = () => {
if (!characterPreview) {
characterInputRef.current?.click();
} else if (!videoPreview) {
videoInputRef.current?.click();
}
};
return (
<section className="image-workbench-page character-mix-page" aria-label="角色迁移">
<header className="image-workbench-topbar">
@@ -342,6 +362,7 @@ function CharacterMixPage({
<div className="studio-panel__section-body">
<label className={characterFile ? "studio-upload-slot--filled" : "studio-upload-slot--empty"}>
<input
ref={characterInputRef}
type="file"
accept="image/*"
onChange={(event) => {
@@ -383,6 +404,7 @@ function CharacterMixPage({
<div className="studio-panel__section-body">
<label className={videoFile ? "studio-upload-slot--filled" : "studio-upload-slot--empty"}>
<input
ref={videoInputRef}
type="file"
accept="video/*"
onChange={(event) => {
@@ -441,12 +463,21 @@ function CharacterMixPage({
) : null}
</div>
) : (
<div className="studio-canvas-ghost">
<div
className={`studio-canvas-ghost${isCanvasDragging ? " is-dragging" : ""}`}
onClick={handleCanvasClick}
onDragOver={handleCanvasDragOver}
onDragLeave={handleCanvasDragLeave}
onDrop={handleCanvasDrop}
role="button"
tabIndex={0}
onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") handleCanvasClick(); }}
>
<div className="studio-canvas-ghost__icon">
<SwapOutlined />
</div>
<div className="studio-canvas-ghost__title"></div>
<div className="studio-canvas-ghost__hint"></div>
<div className="studio-canvas-ghost__hint"> (PNG/JPG) (MP4/MOV/AVI)</div>
</div>
)
}
@@ -6,7 +6,7 @@ import {
PictureOutlined,
UploadOutlined,
} from "@ant-design/icons";
import { useEffect, useMemo, useRef, useState, type ChangeEvent } from "react";
import { useEffect, useMemo, useRef, useState, type ChangeEvent, type DragEvent } from "react";
import { aiGenerationClient } from "../../api/aiGenerationClient";
import { communityClient } from "../../api/communityClient";
import WorkspacePageShell from "../../components/WorkspacePageShell";
@@ -73,6 +73,29 @@ export default function CommunityCaseAddPage({ session, onOpenLogin, onOpenRevie
const allowed = canManageCommunityCases(session);
const imageInputRef = useRef<HTMLInputElement | null>(null);
const workflowInputRef = useRef<HTMLInputElement | null>(null);
const [isImageDragging, setIsImageDragging] = useState(false);
const [isWorkflowDragging, setIsWorkflowDragging] = useState(false);
const handleImageDragOver = (e: DragEvent) => { e.preventDefault(); if (e.dataTransfer.types.includes("Files")) setIsImageDragging(true); };
const handleImageDragLeave = (e: DragEvent) => { e.preventDefault(); if (!e.currentTarget.contains(e.relatedTarget as Node)) setIsImageDragging(false); };
const handleImageDrop = (e: DragEvent) => {
e.preventDefault();
setIsImageDragging(false);
if (e.dataTransfer.files.length) {
void handleImageChange({ target: { files: e.dataTransfer.files } } as ChangeEvent<HTMLInputElement>);
}
};
const handleWorkflowDragOver = (e: DragEvent) => { e.preventDefault(); if (e.dataTransfer.types.includes("Files")) setIsWorkflowDragging(true); };
const handleWorkflowDragLeave = (e: DragEvent) => { e.preventDefault(); if (!e.currentTarget.contains(e.relatedTarget as Node)) setIsWorkflowDragging(false); };
const handleWorkflowDrop = (e: DragEvent) => {
e.preventDefault();
setIsWorkflowDragging(false);
if (e.dataTransfer.files.length) {
void handleWorkflowChange({ target: { files: e.dataTransfer.files } } as ChangeEvent<HTMLInputElement>);
}
};
const [target, setTarget] = useState<CaseTarget>("generation");
const [title, setTitle] = useState("");
const [description, setDescription] = useState("");
@@ -331,7 +354,14 @@ export default function CommunityCaseAddPage({ session, onOpenLogin, onOpenRevie
</label>
<div className="community-case-add-upload-row">
<input ref={imageInputRef} type="file" accept="image/*" hidden onChange={handleImageChange} />
<button type="button" onClick={() => imageInputRef.current?.click()}>
<button
type="button"
className={isImageDragging ? "is-dragging" : ""}
onClick={() => imageInputRef.current?.click()}
onDragOver={handleImageDragOver}
onDragLeave={handleImageDragLeave}
onDrop={handleImageDrop}
>
<UploadOutlined />
</button>
@@ -345,7 +375,14 @@ export default function CommunityCaseAddPage({ session, onOpenLogin, onOpenRevie
<>
<div className="community-case-add-upload-row">
<input ref={workflowInputRef} type="file" accept="application/json,.json" hidden onChange={handleWorkflowChange} />
<button type="button" onClick={() => workflowInputRef.current?.click()}>
<button
type="button"
className={isWorkflowDragging ? "is-dragging" : ""}
onClick={() => workflowInputRef.current?.click()}
onDragOver={handleWorkflowDragOver}
onDragLeave={handleWorkflowDragLeave}
onDrop={handleWorkflowDrop}
>
<UploadOutlined />
JSON
</button>
@@ -98,6 +98,10 @@ function DigitalHumanPage({
activeTaskIdRef.current = activeTaskId;
const keepaliveRestoredRef = useRef(false);
const [isDragging, setIsDragging] = useState(false);
const imageInputRef = useRef<HTMLInputElement | null>(null);
const audioInputRef = useRef<HTMLInputElement | null>(null);
const canvasDragCounterRef = useRef(0);
const [isCanvasDragging, setIsCanvasDragging] = useState(false);
useEffect(() => {
return () => {
@@ -171,6 +175,39 @@ function DigitalHumanPage({
}
};
const handleCanvasDragOver = (e: DragEvent) => { e.preventDefault(); e.stopPropagation(); if (e.dataTransfer.types.includes("Files")) setIsCanvasDragging(true); };
const handleCanvasDragLeave = (e: DragEvent) => { e.preventDefault(); e.stopPropagation(); if (e.currentTarget === e.target || !e.currentTarget.contains(e.relatedTarget as Node)) setIsCanvasDragging(false); };
const handleCanvasDrop = (e: DragEvent) => {
e.preventDefault();
e.stopPropagation();
setIsCanvasDragging(false);
const file = e.dataTransfer.files[0];
if (!file) return;
if (file.type.startsWith("image/")) {
if (imagePreview) URL.revokeObjectURL(imagePreview);
setImageName(file.name);
setImageFile(file);
setImagePreview(URL.createObjectURL(file));
pushDebugEntry("选择图片", `${file.name} / ${file.type || "unknown"} / ${formatFileSize(file.size)}`);
setNotice(`已拖放参考图 ${file.name}`);
} else if (file.type.startsWith("audio/")) {
if (audioPreview) URL.revokeObjectURL(audioPreview);
setAudioName(file.name);
setAudioFile(file);
setAudioPreview(URL.createObjectURL(file));
pushDebugEntry("选择音频", `${file.name} / ${file.type || "unknown"} / ${formatFileSize(file.size)}`);
setNotice(`已拖放音频 ${file.name}`);
}
};
const handleCanvasClick = () => {
if (!imagePreview) {
imageInputRef.current?.click();
} else if (!audioPreview) {
audioInputRef.current?.click();
}
};
const handleDownloadResult = async () => {
if (!resultVideoUrl || isDownloadingResult) return;
setIsDownloadingResult(true);
@@ -463,6 +500,7 @@ function DigitalHumanPage({
<div className="studio-panel__section-body">
<label className={imageName ? "studio-upload-slot--filled" : "studio-upload-slot--empty"}>
<input
ref={imageInputRef}
type="file"
accept="image/*"
onChange={(event) => {
@@ -501,6 +539,7 @@ function DigitalHumanPage({
<div className="studio-panel__section-body">
<label className={audioName ? "studio-upload-slot--filled" : "studio-upload-slot--empty"}>
<input
ref={audioInputRef}
type="file"
accept="audio/*"
onChange={(event) => {
@@ -541,12 +580,21 @@ function DigitalHumanPage({
<img src={imagePreview} alt="参考人像" />
</div>
) : (
<div className="studio-canvas-ghost">
<div
className={`studio-canvas-ghost${isCanvasDragging ? " is-dragging" : ""}`}
onClick={handleCanvasClick}
onDragOver={handleCanvasDragOver}
onDragLeave={handleCanvasDragLeave}
onDrop={handleCanvasDrop}
role="button"
tabIndex={0}
onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") handleCanvasClick(); }}
>
<div className="studio-canvas-ghost__icon">
<CustomerServiceOutlined />
</div>
<div className="studio-canvas-ghost__title"></div>
<div className="studio-canvas-ghost__hint"></div>
<div className="studio-canvas-ghost__hint"> (PNG/JPG/WEBP) (MP3/WAV/M4A)</div>
</div>
)
}
+5 -4
View File
@@ -988,6 +988,10 @@ function ProductClonePage(_props: ProductClonePageProps = {}) {
const selectedProductSetOutput =
productSetOutputOptions.find((option) => option.key === productSetOutput) ?? productSetOutputOptions[0]!;
const selectedCloneOutput = cloneOutputOptions.find((option) => option.key === cloneOutput) ?? cloneOutputOptions[1]!;
const cloneRequirementPlaceholder =
cloneOutput === "model"
? "建议包含以下信息:产品名称、核心卖点、期望场景、模特外貌描写(如小麦色皮肤、齐刘海、眼角有泪痣)、具体参数"
: "建议包含以下信息,产品名称,核心卖点,期望场景,具体参数";
const productSetPreviewReady = productSetStatus === "done";
const cloneSetTotal = useMemo(
() => Object.values(cloneSetCounts).reduce((sum, value) => sum + value, 0),
@@ -1934,7 +1938,6 @@ function ProductClonePage(_props: ProductClonePageProps = {}) {
age: cloneModelAge,
ethnicity: cloneModelEthnicity,
body: cloneModelBody,
appearance: cloneModelAppearance,
scenes: selectedCloneModelScenes,
customScene: cloneModelCustomScene,
}
@@ -2225,7 +2228,6 @@ function ProductClonePage(_props: ProductClonePageProps = {}) {
cloneModelSelects={cloneModelSelects}
openCloneModelSelect={openCloneModelSelect}
cloneModelSelectDropUp={cloneModelSelectDropUp}
cloneModelAppearance={cloneModelAppearance}
cloneVideoQuality={cloneVideoQuality}
cloneVideoQualityOptions={cloneVideoQualityOptions}
cloneVideoDuration={cloneVideoDuration}
@@ -2257,7 +2259,6 @@ function ProductClonePage(_props: ProductClonePageProps = {}) {
setCloneModelCustomScene={setCloneModelCustomScene}
setOpenCloneModelSelect={setOpenCloneModelSelect}
setCloneModelSelectDropUp={setCloneModelSelectDropUp}
setCloneModelAppearance={setCloneModelAppearance}
setCloneVideoQuality={setCloneVideoQuality}
setCloneVideoDuration={setCloneVideoDuration}
clampCloneVideoDuration={clampCloneVideoDuration}
@@ -2620,7 +2621,7 @@ function ProductClonePage(_props: ProductClonePageProps = {}) {
if (event.key === "Escape") setRequirementImageMentionQuery(null);
}}
maxLength={500}
placeholder="建议包含以下信息,产品名称,核心卖点,期望场景,具体参数"
placeholder={cloneRequirementPlaceholder}
/>
{requirementImageMentionQuery !== null && ecommerceMentionImages.length ? (
<ImageMentionMenu images={ecommerceMentionImages} query={requirementImageMentionQuery} onSelect={insertRequirementImageMention} />
@@ -100,7 +100,6 @@ interface EcommerceClonePanelProps {
cloneModelSelects: CloneModelSelectItem[];
openCloneModelSelect: CloneModelSelectKey | null;
cloneModelSelectDropUp: boolean;
cloneModelAppearance: string;
cloneVideoQuality: CloneVideoQualityKey;
cloneVideoQualityOptions: CloneVideoQualityOption[];
cloneVideoDuration: number;
@@ -132,7 +131,6 @@ interface EcommerceClonePanelProps {
setCloneModelCustomScene: (value: string) => void;
setOpenCloneModelSelect: (value: CloneModelSelectKey | null) => void;
setCloneModelSelectDropUp: (value: boolean) => void;
setCloneModelAppearance: (value: string) => void;
setCloneVideoQuality: (value: CloneVideoQualityKey) => void;
setCloneVideoDuration: (value: number) => void;
clampCloneVideoDuration: (value: number) => number;
@@ -172,7 +170,6 @@ export default function EcommerceClonePanel({
cloneModelSelects,
openCloneModelSelect,
cloneModelSelectDropUp,
cloneModelAppearance,
cloneVideoQuality,
cloneVideoQualityOptions,
cloneVideoDuration,
@@ -204,7 +201,6 @@ export default function EcommerceClonePanel({
setCloneModelCustomScene,
setOpenCloneModelSelect,
setCloneModelSelectDropUp,
setCloneModelAppearance,
setCloneVideoQuality,
setCloneVideoDuration,
clampCloneVideoDuration,
@@ -668,14 +664,6 @@ export default function EcommerceClonePanel({
);
})}
</div>
<label className="clone-ai-model-textarea">
<strong></strong>
<textarea
value={cloneModelAppearance}
onChange={(event) => setCloneModelAppearance(event.target.value)}
placeholder="例如:小麦色皮肤、齐刘海、眼角有泪痣..."
/>
</label>
</div>
)}
</div>
@@ -758,7 +746,7 @@ export default function EcommerceClonePanel({
style={{ display: "none" }}
/>
<button type="button" className="clone-ai-video-outfit-upload-btn" onClick={() => videoOutfitVideoRef.current?.click()}>
{videoOutfitVideoUrl ? "重新选择视频" : "选择视频文件"}
{videoOutfitVideoUrl ? "重新上传视频" : "点击上传视频"}
</button>
{videoOutfitVideoUrl ? <span className="clone-ai-video-outfit-info"></span> : null}
</div>
@@ -774,7 +762,7 @@ export default function EcommerceClonePanel({
style={{ display: "none" }}
/>
<button type="button" className="clone-ai-video-outfit-upload-btn" onClick={() => videoOutfitRefRef.current?.click()}>
{videoOutfitRefUrl ? "重新选择参考图" : "选择参考图"}
{videoOutfitRefUrl ? "重新上传参考图" : "点击上传参考图"}
</button>
{videoOutfitRefUrl ? <span className="clone-ai-video-outfit-info"></span> : null}
</div>
@@ -1,5 +1,5 @@
import { CloudUploadOutlined, LoadingOutlined, QuestionCircleOutlined } from "@ant-design/icons";
import type { ChangeEvent, RefObject } from "react";
import { useState, type ChangeEvent, type DragEvent, type RefObject } from "react";
import { EcommerceProgressBar } from "../EcommerceProgressBar";
interface EcommerceDetailPanelProps {
@@ -59,6 +59,31 @@ export default function EcommerceDetailPanel({
handleDetailGenerate,
onCancelGenerate,
}: EcommerceDetailPanelProps) {
const [isDragging, setIsDragging] = useState(false);
const handleDragOver = (e: DragEvent<HTMLButtonElement>) => {
e.preventDefault();
e.stopPropagation();
if (e.dataTransfer.types.includes("Files")) setIsDragging(true);
};
const handleDragLeave = (e: DragEvent<HTMLButtonElement>) => {
e.preventDefault();
e.stopPropagation();
if (e.currentTarget === e.target || !e.currentTarget.contains(e.relatedTarget as Node)) {
setIsDragging(false);
}
};
const handleDrop = (e: DragEvent<HTMLButtonElement>) => {
e.preventDefault();
e.stopPropagation();
setIsDragging(false);
if (e.dataTransfer.files.length) {
handleDetailUpload({ target: { files: e.dataTransfer.files } } as ChangeEvent<HTMLInputElement>);
}
};
return (
<>
<div className="product-clone-panel__scroll">
@@ -67,7 +92,14 @@ export default function EcommerceDetailPanel({
<QuestionCircleOutlined />
</h2>
<button type="button" className="product-clone-upload-zone product-detail-upload" onClick={() => detailInputRef.current?.click()}>
<button
type="button"
className={`product-clone-upload-zone product-detail-upload${isDragging ? " is-dragging" : ""}`}
onClick={() => detailInputRef.current?.click()}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleDrop}
>
<strong>
<CloudUploadOutlined />
@@ -1,5 +1,5 @@
import { CloudUploadOutlined, LoadingOutlined, QuestionCircleOutlined } from "@ant-design/icons";
import type { ChangeEvent, RefObject } from "react";
import { useState, type ChangeEvent, type DragEvent, type RefObject } from "react";
import { EcommerceProgressBar } from "../EcommerceProgressBar";
interface EcommerceTryOnPanelProps {
@@ -73,12 +73,44 @@ export default function EcommerceTryOnPanel({
handleTryOnGenerate,
onCancelGenerate,
}: EcommerceTryOnPanelProps) {
const [isDragging, setIsDragging] = useState(false);
const handleDragOver = (e: DragEvent<HTMLButtonElement>) => {
e.preventDefault();
e.stopPropagation();
if (e.dataTransfer.types.includes("Files")) setIsDragging(true);
};
const handleDragLeave = (e: DragEvent<HTMLButtonElement>) => {
e.preventDefault();
e.stopPropagation();
if (e.currentTarget === e.target || !e.currentTarget.contains(e.relatedTarget as Node)) {
setIsDragging(false);
}
};
const handleDrop = (e: DragEvent<HTMLButtonElement>) => {
e.preventDefault();
e.stopPropagation();
setIsDragging(false);
if (e.dataTransfer.files.length) {
handleGarmentUpload({ target: { files: e.dataTransfer.files } } as ChangeEvent<HTMLInputElement>);
}
};
return (
<>
<div className="product-clone-panel__scroll">
<section className="product-clone-field">
<h2></h2>
<button type="button" className="product-clone-upload-zone product-try-on-upload" onClick={() => garmentInputRef.current?.click()}>
<button
type="button"
className={`product-clone-upload-zone product-try-on-upload${isDragging ? " is-dragging" : ""}`}
onClick={() => garmentInputRef.current?.click()}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleDrop}
>
<strong>
<CloudUploadOutlined />
+26 -8
View File
@@ -35,6 +35,7 @@ const {
interface HomePageProps {
onOpenGenerate: () => void;
onStartOnboarding?: () => void;
onOpenCanvas?: () => void;
onOpenEcommerce: () => void;
onOpenScriptReview?: () => void;
@@ -194,10 +195,18 @@ function getHomeCarouselCardStyle(offset: number): CSSProperties {
const depth = Math.abs(offset);
const direction = Math.sign(offset);
const isActive = depth === 0;
const xByDepth = [0, 190, 320, 430, 520, 590];
const xByDepth = [
"0px",
"clamp(52px, 13.5vw, 198px)",
"clamp(90px, 22.5vw, 334px)",
"clamp(122px, 30.5vw, 448px)",
"clamp(148px, 37vw, 542px)",
"clamp(170px, 42vw, 614px)",
];
const yByDepth = [8, -2, -8, -13, -18, -24];
const scaleByDepth = [1, 1, 1, 1, 1, 1];
const x = direction * (xByDepth[depth] ?? xByDepth[xByDepth.length - 1]!);
const xDistance = xByDepth[depth] ?? xByDepth[xByDepth.length - 1]!;
const x = direction < 0 ? `calc(0px - ${xDistance})` : xDistance;
const y = yByDepth[depth] ?? yByDepth[yByDepth.length - 1]!;
const z = isActive ? 90 : 28 - depth;
const scale = scaleByDepth[depth] ?? scaleByDepth[scaleByDepth.length - 1]!;
@@ -206,7 +215,7 @@ function getHomeCarouselCardStyle(offset: number): CSSProperties {
"--apple-card-offset": offset,
"--apple-card-depth": depth,
"--apple-card-z": 80 - depth,
"--apple-card-x": `${x}px`,
"--apple-card-x": x,
"--apple-card-y": `${y}px`,
"--apple-card-z-offset": `${z}px`,
"--apple-card-rotate-y": "0deg",
@@ -468,7 +477,7 @@ function EcommerceFeatureShowcase() {
);
}
function HomePage({ onOpenGenerate, onOpenCanvas, onOpenEcommerce, onOpenScriptReview, onOpenTokenMonitor, onSelectView, onOpenImageTool }: HomePageProps) {
function HomePage({ onOpenGenerate, onStartOnboarding, onOpenCanvas, onOpenEcommerce, onOpenScriptReview, onOpenTokenMonitor, onSelectView, onOpenImageTool }: HomePageProps) {
const [splashDismissed, setSplashDismissed] = useState(() => sessionStorage.getItem("omniai:splash-seen") === "1");
const [activeSlideIndex, setActiveSlideIndex] = useState(0);
const [carouselMotion, setCarouselMotion] = useState<HomeCarouselMotion | null>(null);
@@ -620,17 +629,26 @@ function HomePage({ onOpenGenerate, onOpenCanvas, onOpenEcommerce, onOpenScriptR
</div>
<div className="omni-home__actions" aria-label="首页入口">
<button type="button" className="omni-home__entry" onClick={onOpenGenerate}>
<button type="button" className="omni-home__entry" onClick={onStartOnboarding || onOpenGenerate}>
<PlusOutlined />
<span></span>
<span>
<small></small>
</span>
</button>
<button type="button" className="omni-home__entry omni-home__entry--primary" onClick={onOpenCanvas || onOpenGenerate}>
<PlayCircleOutlined />
<span></span>
<span>
<small></small>
</span>
</button>
<button type="button" className="omni-home__entry" onClick={onOpenEcommerce}>
<ShoppingOutlined />
<span></span>
<span>
<small></small>
</span>
</button>
</div>
</section>
@@ -947,19 +947,22 @@ function ImageWorkbenchPage({ initialTool = "workbench", onOpenMore, onSelectVie
</div>
</div>
) : (
<button
type="button"
className={`image-workbench-empty image-workbench-empty--button${isInpaintDragging ? " is-dragging" : ""}`}
<div
className={`studio-canvas-ghost${isInpaintDragging ? " is-dragging" : ""}`}
onClick={() => inpaintFileInputRef.current?.click()}
onDragOver={handleInpaintDragOver}
onDragLeave={handleInpaintDragLeave}
onDrop={handleInpaintDrop}
role="button"
tabIndex={0}
onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") inpaintFileInputRef.current?.click(); }}
>
{isInpaintDragging ? <span className="image-workbench-upload-drop-overlay" style={{ borderRadius: "var(--radius-sm)" }}><span></span></span> : null}
<FileImageOutlined />
<strong></strong>
<span> PNG / JPG / WebP</span>
</button>
<div className="studio-canvas-ghost__icon">
<FileImageOutlined />
</div>
<div className="studio-canvas-ghost__title"></div>
<div className="studio-canvas-ghost__hint"> PNG / JPG / WebP使</div>
</div>
)}
</section>
@@ -1389,12 +1392,21 @@ function ImageWorkbenchPage({ initialTool = "workbench", onOpenMore, onSelectVie
<img src={referenceImage} alt="参考图预览" />
</div>
) : (
<div className="studio-canvas-ghost">
<div
className={`studio-canvas-ghost${isDragging ? " is-dragging" : ""}`}
onClick={() => fileInputRef.current?.click()}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleDrop}
role="button"
tabIndex={0}
onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") fileInputRef.current?.click(); }}
>
<div className="studio-canvas-ghost__icon">
<PictureOutlined />
</div>
<div className="studio-canvas-ghost__title"></div>
<div className="studio-canvas-ghost__hint"></div>
<div className="studio-canvas-ghost__hint"> (PNG / JPG / WebP)</div>
</div>
)}
</section>
+3 -3
View File
@@ -345,11 +345,12 @@ function MorePage({ onSelectView, onOpenImageTool }: MorePageProps) {
key={tool.id}
type="button"
className={`more-card more-card--featured${getPreviewClassName(tool.id)}`}
style={{ "--card-gradient": coreToolGradients[tool.id] ?? "linear-gradient(135deg, rgba(var(--accent-rgb), 0.12), rgba(var(--accent-rgb), 0.04))" } as CSSProperties}
style={{
"--card-gradient": coreToolGradients[tool.id] ?? "linear-gradient(135deg, rgba(var(--accent-rgb), 0.12), rgba(var(--accent-rgb), 0.04))",
} as CSSProperties}
aria-label={`打开核心工具:${tool.title}${tool.text}`}
onClick={() => openTool(tool)}
>
<span className="more-card__featured-icon">{tool.icon}</span>
<div className="more-card__featured-body">
<span className="more-card__featured-kicker">{categoryLabels[tool.category]}</span>
<strong>{tool.title}</strong>
@@ -388,7 +389,6 @@ function MorePage({ onSelectView, onOpenImageTool }: MorePageProps) {
onClick={() => openTool(tool)}
disabled={!tool.ready}
>
<span className="more-card__icon">{tool.icon}</span>
<span className="more-card__topline">
{tool.tags.slice(0, 2).map((tag) => (
<span key={tag}>{tag}</span>
+37 -4
View File
@@ -18,7 +18,7 @@ import {
ShareAltOutlined,
UserOutlined,
} from "@ant-design/icons";
import { useEffect, useMemo, useRef, useState, type ChangeEvent, type FormEvent, type KeyboardEvent } from "react";
import { useEffect, useMemo, useRef, useState, type ChangeEvent, type DragEvent, type FormEvent, type KeyboardEvent } from "react";
import { createPortal } from "react-dom";
import "../../styles/pages/profile.css";
import { aiGenerationClient } from "../../api/aiGenerationClient";
@@ -228,6 +228,28 @@ function ProfilePage({
const avatarLabel = displayName.trim().slice(0, 1).toUpperCase() || "访";
const avatarInputRef = useRef<HTMLInputElement | null>(null);
const bannerInputRef = useRef<HTMLInputElement | null>(null);
const [isBannerDragging, setIsBannerDragging] = useState(false);
const [isAvatarDragging, setIsAvatarDragging] = useState(false);
const handleBannerDragOver = (e: DragEvent) => { e.preventDefault(); if (e.dataTransfer.types.includes("Files")) setIsBannerDragging(true); };
const handleBannerDragLeave = (e: DragEvent) => { e.preventDefault(); if (!e.currentTarget.contains(e.relatedTarget as Node)) setIsBannerDragging(false); };
const handleBannerDrop = (e: DragEvent) => {
e.preventDefault();
setIsBannerDragging(false);
if (e.dataTransfer.files.length) {
handleBannerUpload({ target: { files: e.dataTransfer.files } } as ChangeEvent<HTMLInputElement>);
}
};
const handleAvatarDragOver = (e: DragEvent) => { e.preventDefault(); if (e.dataTransfer.types.includes("Files")) setIsAvatarDragging(true); };
const handleAvatarDragLeave = (e: DragEvent) => { e.preventDefault(); if (!e.currentTarget.contains(e.relatedTarget as Node)) setIsAvatarDragging(false); };
const handleAvatarDrop = (e: DragEvent) => {
e.preventDefault();
setIsAvatarDragging(false);
if (e.dataTransfer.files.length) {
handleAvatarUpload({ target: { files: e.dataTransfer.files } } as ChangeEvent<HTMLInputElement>);
}
};
const [mode, setMode] = useState<WebAuthMode>("login");
const [authTab, setAuthTab] = useState<AuthTab>("password");
@@ -1047,8 +1069,11 @@ function ProfilePage({
<input ref={avatarInputRef} type="file" accept="image/*" hidden onChange={(event) => void handleAvatarUpload(event)} />
<input ref={bannerInputRef} type="file" accept="image/*" hidden onChange={(event) => void handleBannerUpload(event)} />
<header
className={`profile-page__banner${bannerUrl ? " has-image" : ""}`}
className={`profile-page__banner${bannerUrl ? " has-image" : ""}${isBannerDragging ? " is-dragging" : ""}`}
style={bannerUrl ? { backgroundImage: `url(${bannerUrl})` } : undefined}
onDragOver={handleBannerDragOver}
onDragLeave={handleBannerDragLeave}
onDrop={handleBannerDrop}
>
<button type="button" className="profile-page__banner-btn" onClick={() => bannerInputRef.current?.click()} aria-label="更换背景">
<CameraOutlined />
@@ -1060,13 +1085,21 @@ function ProfilePage({
<div className="profile-page__body">
<aside className="profile-page__sidebar">
<div className="profile-page__sidebar-head">
<div className="profile-page__avatar-ring">
<div className={`profile-page__avatar-ring${isAvatarDragging ? " is-dragging" : ""}`}>
{avatarUrl ? (
<img className="profile-page__avatar" src={avatarUrl} alt="" />
) : (
<span className="profile-page__avatar">{avatarLabel}</span>
)}
<button type="button" className="profile-page__avatar-edit" onClick={() => avatarInputRef.current?.click()} aria-label="更换头像">
<button
type="button"
className="profile-page__avatar-edit"
onClick={() => avatarInputRef.current?.click()}
onDragOver={handleAvatarDragOver}
onDragLeave={handleAvatarDragLeave}
onDrop={handleAvatarDrop}
aria-label="更换头像"
>
<CameraOutlined />
</button>
<span className="profile-page__avatar-badge">
@@ -601,11 +601,20 @@ function ResolutionUpscalePage({
</div>
)
) : (
<div className="studio-canvas-ghost">
<div
className={`studio-canvas-ghost${isDragging ? " is-dragging" : ""}`}
onClick={() => fileInputRef.current?.click()}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleDrop}
role="button"
tabIndex={0}
onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") fileInputRef.current?.click(); }}
>
<div className="studio-canvas-ghost__icon">
<ThunderboltOutlined />
</div>
<div className="studio-canvas-ghost__title">{mode === "image" ? "拖拽或选择图片" : "拖拽或选择视频"}</div>
<div className="studio-canvas-ghost__title">{mode === "image" ? "点击或拖拽上传图片" : "点击或拖拽上传视频"}</div>
<div className="studio-canvas-ghost__hint">{mode === "image" ? "支持 PNG / JPG / WebP" : "支持 MP4 / MOV / WebM"}</div>
</div>
)}
@@ -679,14 +679,23 @@ function ScriptTokensPage() {
</div>
) : !result && (
<div className="script-eval-v5-input-section">
<div className="script-eval-v5-illustration" aria-label="上传剧本并开始评测">
<div className={`script-eval-v5-illustration${isDragging ? " is-dragging" : ""}`} aria-label="上传剧本并开始评测">
<div
className="script-eval-v5-illustration-hit"
role="button"
tabIndex={0}
onClick={() => fileInputRef.current?.click()}
onKeyDown={uploadKeyDown}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleDrop}
>
{isDragging && (
<div className="script-eval-v5-upload-drop-overlay">
<UploadOutlined />
<span></span>
</div>
)}
<div className="script-eval-v5-upload-card-icon">
<ShellIcon name="file-text" />
</div>
@@ -447,15 +447,19 @@ function SubtitleRemovalPage({
</div>
) : (
<div
className="studio-canvas-ghost"
className={`studio-canvas-ghost${isDragging ? " is-dragging" : ""}`}
onClick={() => fileInputRef.current?.click()}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleFileDrop}
role="button"
tabIndex={0}
onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") fileInputRef.current?.click(); }}
>
<div className="studio-canvas-ghost__icon">
<VideoCameraOutlined />
</div>
<div className="studio-canvas-ghost__title"></div>
<div className="studio-canvas-ghost__title"></div>
<div className="studio-canvas-ghost__hint"> MP4 1GB 1080P</div>
</div>
)}
@@ -12,7 +12,7 @@ import {
ScissorOutlined,
SwapOutlined,
} from "@ant-design/icons";
import { useCallback, useEffect, useRef, useState } from "react";
import { useCallback, useEffect, useRef, useState, type DragEvent } from "react";
import "../../styles/pages/more-tools.css";
import "../../styles/pages/image-workbench.css";
import { aiGenerationClient } from "../../api/aiGenerationClient";
@@ -48,6 +48,7 @@ function WatermarkRemovalPage({
const [status, setStatus] = useState("上传含水印的图片,点击开始去水印");
const [activeTaskId, setActiveTaskId] = useState("");
const [taskProgress, setTaskProgress] = useState(0);
const [isDragging, setIsDragging] = useState(false);
const [isProcessing, setIsProcessing] = useState(false);
const [isSavingAsset, setIsSavingAsset] = useState(false);
const [isDownloading, setIsDownloading] = useState(false);
@@ -124,6 +125,10 @@ function WatermarkRemovalPage({
setStatus(`已导入 ${file.name}`);
};
const handleDragOver = (e: DragEvent) => { e.preventDefault(); e.stopPropagation(); if (e.dataTransfer.types.includes("Files")) setIsDragging(true); };
const handleDragLeave = (e: DragEvent) => { e.preventDefault(); e.stopPropagation(); if (e.currentTarget === e.target || !e.currentTarget.contains(e.relatedTarget as Node)) setIsDragging(false); };
const handleCanvasDrop = (e: DragEvent) => { e.preventDefault(); e.stopPropagation(); setIsDragging(false); handleFileDrop(e); };
const handleImportUrl = () => {
const normalizedUrl = sourceUrl.trim();
if (!/^https?:\/\//i.test(normalizedUrl)) {
@@ -403,17 +408,22 @@ function WatermarkRemovalPage({
</div>
</div>
) : (
<button
type="button"
className="image-workbench-empty image-workbench-empty--button"
<div
className={`studio-canvas-ghost${isDragging ? " is-dragging" : ""}`}
onClick={() => fileInputRef.current?.click()}
onDragOver={(e) => e.preventDefault()}
onDrop={handleFileDrop}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleCanvasDrop}
role="button"
tabIndex={0}
onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") fileInputRef.current?.click(); }}
>
<DeleteOutlined />
<strong></strong>
<span> PNG / JPG / WebP</span>
</button>
<div className="studio-canvas-ghost__icon">
<DeleteOutlined />
</div>
<div className="studio-canvas-ghost__title"></div>
<div className="studio-canvas-ghost__hint"> PNG / JPG / WebP"开始去水印"</div>
</div>
)}
</section>
</main>
+148 -2
View File
@@ -193,12 +193,15 @@ import {
PromptPreviewLayer,
} from "./WorkbenchPromptPreview";
import { SelectChip, CompoundSelectChip, InlineOptionChip } from "./WorkbenchSelectChips";
import OnboardingTour, { type TourPhaseId } from "../../components/OnboardingTour";
export type { WorkbenchResultActionPayload } from "./workbenchConstants";
interface WorkbenchPageProps {
isAuthenticated: boolean;
session: WebUserSession | null;
onboarding?: boolean;
onEndOnboarding?: () => void;
onRequireLogin: (input: CreatePreviewTaskInput) => void;
onOpenResultInCanvas?: (payload: import("./workbenchConstants").WorkbenchResultActionPayload) => void;
onRefreshUsage?: () => void;
@@ -231,6 +234,8 @@ function formatCreditValue(value: number): string {
function WorkbenchPage({
isAuthenticated,
session,
onboarding,
onEndOnboarding,
onRequireLogin,
onOpenResultInCanvas,
onRefreshUsage,
@@ -264,7 +269,41 @@ function WorkbenchPage({
const renderedMessageIdsRef = useRef<string[]>([]);
const hasHandledInitialMessagesRef = useRef(false);
const [activeMode, setActiveMode] = useState<WorkbenchMode>("video");
// Onboarding signal — init from prop or localStorage
const [effectiveOnboarding, setEffectiveOnboarding] = useState(
() => onboarding || (() => { try { return window.localStorage.getItem("omniai:onboarding") === "1"; } catch { return false; } })(),
);
// Track whether onboarding prop was ever true, to avoid overwriting localStorage-initiated true
const obWasActiveRef = useRef(onboarding);
useEffect(() => {
if (onboarding) {
obWasActiveRef.current = true;
setEffectiveOnboarding(true);
} else if (obWasActiveRef.current) {
// Only deactivate when prop transitions true→false (user dismissed)
setEffectiveOnboarding(false);
obWasActiveRef.current = false;
}
// If prop was never true, don't touch effectiveOnboarding (preserves localStorage init)
}, [onboarding]);
// Poll localStorage as a fallback (handles cases where prop isn't propagated)
useEffect(() => {
if (effectiveOnboarding) return;
const check = () => {
try {
if (window.localStorage.getItem("omniai:onboarding") === "1") {
setEffectiveOnboarding(true);
}
} catch {}
};
check();
const interval = setInterval(check, 200);
return () => clearInterval(interval);
}, [effectiveOnboarding]);
const [activeMode, setActiveMode] = useState<WorkbenchMode>(() => effectiveOnboarding ? "chat" : "video");
const [inputValue, setInputValue] = useState("");
const [messages, setMessages] = useState<ChatMessage[]>(() => (resetToken ? [] : readStoredMessages()));
const [promptHistory, setPromptHistory] = useState<string[]>(() => readStoredPromptHistory());
@@ -294,6 +333,34 @@ function WorkbenchPage({
const [sidebarCollapsed, setSidebarCollapsed] = useState(true);
const [deleteDialog, setDeleteDialog] = useState<DeleteDialogState | null>(null);
const [deleteSubmitting, setDeleteSubmitting] = useState(false);
// ── Onboarding tour state ──────────────────────────────
const [tourPhase, setTourPhase] = useState<TourPhaseId>("chat");
const [tourStep, setTourStep] = useState(0);
// Sync activeMode with tour phase and keep home view during onboarding
useEffect(() => {
if (!effectiveOnboarding) return;
// Reset tour state for repeat runs
setTourPhase("chat");
setTourStep(0);
// Force "今天想生成什么?" home view — prevent conversation auto-select
skipConversationAutoSelectRef.current = true;
setWorkspaceStarted(false);
setActiveConversationId(null);
activeConversationIdRef.current = null;
persistActiveConversationId(null);
messagesRef.current = [];
setMessages([]);
}, [effectiveOnboarding]);
useEffect(() => {
if (effectiveOnboarding) {
if (tourPhase === "chat") setActiveMode("chat");
else if (tourPhase === "image") setActiveMode("image");
else if (tourPhase === "video") setActiveMode("video");
}
}, [effectiveOnboarding, tourPhase]);
// ───────────────────────────────────────────────────────
const [, setGenerationProgress] = useState(0);
const [cursorIndex, setCursorIndex] = useState(0);
const [promptSelectionRange, setPromptSelectionRange] = useState({ start: 0, end: 0 });
@@ -427,6 +494,7 @@ function WorkbenchPage({
const toolTheme = MODE_META[activeMode];
const workbenchAccent = "#00ff88";
const hasConversationRecords = activeConversationId !== null || messages.length > 0;
const hasActivatedWorkspace = !effectiveOnboarding && (workspaceStarted || isGenerating || hasConversationRecords);
const referenceCount = referenceItems.length;
const activeVideoModelValue = toHappyHorseDisplayModel(videoModel);
const activeModelValue =
@@ -454,7 +522,6 @@ function WorkbenchPage({
[conversations],
);
const hasSidebarRecords = conversationRecords.length > 0;
const hasActivatedWorkspace = workspaceStarted || isGenerating || hasConversationRecords;
const activeConversationTitle = useMemo(() => {
if (!activeConversationId) return "";
@@ -1572,7 +1639,41 @@ function WorkbenchPage({
setToolbarMenuId((current) => (current === menuId ? null : menuId));
};
// ── Onboarding tour helpers ────────────────────────────
const obTarget = (map: Partial<Record<TourPhaseId, string>>): string | undefined =>
effectiveOnboarding ? map[tourPhase] : undefined;
const handleTourNext = useCallback((_phase: TourPhaseId, stepIndex: number) => {
setTourStep(stepIndex);
}, []);
const handleTourSkip = useCallback((phase: TourPhaseId) => {
const next: Record<TourPhaseId, TourPhaseId> = { chat: "image", image: "video", video: "video" };
const nextPhase = next[phase];
if (nextPhase === phase) {
onEndOnboarding?.();
} else {
setTourPhase(nextPhase);
setTourStep(0);
if (nextPhase === "image") setActiveMode("image");
else if (nextPhase === "video") setActiveMode("video");
}
}, [onEndOnboarding, setActiveMode]);
const handleTourDone = useCallback(() => {
setEffectiveOnboarding(false);
onEndOnboarding?.();
}, [onEndOnboarding]);
// Advance tour phase when user switches mode during onboarding
const handleModeChange = (mode: WorkbenchMode) => {
if (effectiveOnboarding) {
// Advance tour phase when switching to the next mode
if (tourPhase === "chat" && mode === "image") { setTourPhase("image"); setTourStep(0); }
else if (tourPhase === "image" && mode === "video") { setTourPhase("video"); setTourStep(0); }
// Block switching to other modes during guided tour
else if (mode !== tourPhase) return;
}
setActiveMode(mode);
setToolbarMenuId(null);
setReferencePreviewOpen(false);
@@ -2795,6 +2896,7 @@ function WorkbenchPage({
className="wb-composer__ref-upload"
onClick={handleReferenceUploadClick}
disabled={disabled}
data-onboarding={obTarget({ chat: "onboarding-chat-upload", image: "onboarding-image-upload", video: "onboarding-video-upload" })}
aria-label={`上传${referenceUploadLabel}`}
aria-expanded={referenceItems.length > 0 ? referencePreviewOpen : undefined}
aria-controls={referenceItems.length > 0 ? "workbench-reference-stack" : undefined}
@@ -2854,6 +2956,7 @@ function WorkbenchPage({
const renderComposerToolbar = (disabled = false, showStop = false) => (
<div className="wb-composer__toolbar">
<div className="wb-composer__toolbar-left">
<span data-onboarding={obTarget({ chat: "onboarding-mode-selector", image: "onboarding-mode-selector" })}>
<SelectChip
chipId="studio-mode"
value={activeMode}
@@ -2866,8 +2969,10 @@ function WorkbenchPage({
ariaLabel="工作台模式"
direction={dropdownDirection}
/>
</span>
{activeMode === "chat" && (
<>
<span data-onboarding={obTarget({ chat: "onboarding-chat-model" })}>
<SelectChip
chipId="chat-model"
value={chatModel}
@@ -2880,6 +2985,8 @@ function WorkbenchPage({
ariaLabel="对话模型"
direction={dropdownDirection}
/>
</span>
<span data-onboarding={obTarget({ chat: "onboarding-chat-speed" })}>
<SelectChip
chipId="chat-speed"
value={thinkingSpeed}
@@ -2892,6 +2999,8 @@ function WorkbenchPage({
ariaLabel="思考速度"
direction={dropdownDirection}
/>
</span>
<span data-onboarding={obTarget({ chat: "onboarding-chat-depth" })}>
<SelectChip
chipId="chat-depth"
value={thinkingDepth}
@@ -2904,10 +3013,12 @@ function WorkbenchPage({
ariaLabel="思考深度"
direction={dropdownDirection}
/>
</span>
</>
)}
{activeMode === "image" && (
<>
<span data-onboarding={obTarget({ image: "onboarding-image-model" })}>
<SelectChip
chipId="image-model"
value={imageModel}
@@ -2919,6 +3030,8 @@ function WorkbenchPage({
onChange={(v) => { setImageModel(v); if (!GRID_SUPPORTED_MODELS.has(v)) setImageGridMode("single"); }}
direction={dropdownDirection}
/>
</span>
<span data-onboarding={obTarget({ image: "onboarding-image-settings" })}>
<CompoundSelectChip
chipId="image-settings"
summary={imageSettingsSummary}
@@ -2928,7 +3041,9 @@ function WorkbenchPage({
onToggle={() => toggleToolbarMenu("image-settings")}
direction={dropdownDirection}
/>
</span>
{GRID_SUPPORTED_MODELS.has(imageModel) && (
<span data-onboarding={obTarget({ image: "onboarding-image-grid" })}>
<SelectChip
chipId="image-grid-mode"
value={imageGridMode}
@@ -2940,11 +3055,13 @@ function WorkbenchPage({
onChange={setImageGridMode}
direction={dropdownDirection}
/>
</span>
)}
</>
)}
{activeMode === "video" && (
<>
<span data-onboarding={obTarget({ video: "onboarding-video-model" })}>
<SelectChip
chipId="video-model"
value={videoModel}
@@ -2956,6 +3073,8 @@ function WorkbenchPage({
onChange={setVideoModel}
direction={dropdownDirection}
/>
</span>
<span data-onboarding={obTarget({ video: "onboarding-video-frame" })}>
<SelectChip
chipId="video-mode"
value={videoFrameMode}
@@ -2967,6 +3086,8 @@ function WorkbenchPage({
onChange={setVideoFrameMode}
direction={dropdownDirection}
/>
</span>
<span data-onboarding={obTarget({ video: "onboarding-video-ratio" })}>
<CompoundSelectChip
chipId="video-ratio"
summary={videoRatio}
@@ -2976,6 +3097,8 @@ function WorkbenchPage({
onToggle={() => toggleToolbarMenu("video-ratio")}
direction={dropdownDirection}
/>
</span>
<span data-onboarding={obTarget({ video: "onboarding-video-duration" })}>
<InlineOptionChip
chipId="video-duration"
value={videoDuration}
@@ -2988,6 +3111,8 @@ function WorkbenchPage({
onChange={setVideoDuration}
direction={dropdownDirection}
/>
</span>
<span data-onboarding={obTarget({ video: "onboarding-video-quality" })}>
<InlineOptionChip
chipId="video-quality"
value={videoQuality}
@@ -3000,6 +3125,7 @@ function WorkbenchPage({
onChange={setVideoQuality}
direction={dropdownDirection}
/>
</span>
</>
)}
</div>
@@ -3013,6 +3139,7 @@ function WorkbenchPage({
disabled={sendDisabled || isGenerating}
title={isGenerating ? "任务处理中" : sendButtonTitle}
aria-label={isGenerating ? "任务处理中" : sendButtonTitle}
data-onboarding={obTarget({ video: "onboarding-video-generate" })}
onClick={() => {
if (getCachedRole() === "admin") console.log("[ai/workbench-send-click]", {
mode: activeMode,
@@ -3171,6 +3298,7 @@ function WorkbenchPage({
className={`wb-composer__textarea${showPromptPreview ? " wb-composer__textarea--overlay-mode" : ""}`}
placeholder={composerPlaceholder}
value={inputValue}
data-onboarding={obTarget({ chat: "onboarding-chat-input", image: "onboarding-image-input" })}
onChange={handlePromptChange}
onSelect={handlePromptSelectionChange}
onKeyUp={handlePromptSelectionChange}
@@ -3236,6 +3364,14 @@ function WorkbenchPage({
{renderMessagePreviewOverlay()}
{renderPromptCaseOverlay()}
{renderDeleteDialog()}
<OnboardingTour
active={Boolean(effectiveOnboarding)}
phase={tourPhase}
stepIndex={tourStep}
onNext={handleTourNext}
onSkip={handleTourSkip}
onDone={handleTourDone}
/>
</section>
);
}
@@ -3357,6 +3493,7 @@ function WorkbenchPage({
placeholder={composerPlaceholder}
value={inputValue}
disabled={false}
data-onboarding={obTarget({ chat: "onboarding-chat-input", image: "onboarding-image-input" })}
onChange={handlePromptChange}
onSelect={handlePromptSelectionChange}
onKeyUp={handlePromptSelectionChange}
@@ -3404,6 +3541,15 @@ function WorkbenchPage({
{showRechargeModal && RechargeModal ? (
<RechargeModal open={showRechargeModal} onClose={() => setShowRechargeModal(false)} currentBalance={session?.user?.balanceCents} />
) : null}
<OnboardingTour
active={Boolean(effectiveOnboarding)}
phase={tourPhase}
stepIndex={tourStep}
onNext={handleTourNext}
onSkip={handleTourSkip}
onDone={handleTourDone}
/>
</section>
);
}
+4 -4
View File
@@ -149,14 +149,14 @@ export const CHAT_MODEL_OPTIONS: WorkbenchOption[] = [
export const THINKING_SPEED_OPTIONS: WorkbenchOption[] = [
{ value: "default", label: "默认" },
{ value: "high", label: "高" },
{ value: "ultra", label: "急速" },
{ value: "high", label: "思考速度:高" },
{ value: "ultra", label: "思考速度:急速" },
];
export const THINKING_DEPTH_OPTIONS: WorkbenchOption[] = [
{ value: "default", label: "默认" },
{ value: "strong", label: "强" },
{ value: "extreme", label: "极限" },
{ value: "strong", label: "推理深度:强" },
{ value: "extreme", label: "推理深度:极限" },
];
export const CHAT_NATURAL_SYSTEM_PROMPT = [
+317
View File
@@ -0,0 +1,317 @@
/* ─── Onboarding Tour ──────────────────────────────────────── */
.onboarding-root {
position: fixed;
inset: 0;
z-index: 10000;
pointer-events: none;
opacity: 0;
transition: opacity 0.35s ease;
font-family: Inter, "SF Pro Text", -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;
}
.onboarding-root.is-visible {
opacity: 1;
}
/* ─── Overlay ──────────────────────────────────────────────── */
.onboarding-overlay {
position: fixed;
inset: 0;
background: rgba(0, 0, 0, 0.64);
pointer-events: auto;
transition: clip-path 0.32s cubic-bezier(0.4, 0, 0.2, 1);
will-change: clip-path;
}
/* During interactive steps, let clicks pass through to dropdowns etc. */
.onboarding-overlay--passive {
pointer-events: none;
}
/* Tooltip also lets clicks through during interactive steps, except for buttons */
.onboarding-tooltip--passive {
pointer-events: none;
}
.onboarding-tooltip--passive .onboarding-tooltip__btn {
pointer-events: auto;
}
/* ─── Spotlight ring ───────────────────────────────────────── */
.onboarding-spotlight {
position: fixed;
z-index: 10001;
border-radius: 10px;
border: 2px solid rgba(0, 255, 136, 0.5);
box-shadow:
0 0 22px rgba(0, 255, 136, 0.18),
0 0 50px rgba(0, 255, 136, 0.06),
inset 0 0 0 1px rgba(0, 255, 136, 0.05);
pointer-events: none;
transition: left 0.32s cubic-bezier(0.4, 0, 0.2, 1),
top 0.32s cubic-bezier(0.4, 0, 0.2, 1),
width 0.32s cubic-bezier(0.4, 0, 0.2, 1),
height 0.32s cubic-bezier(0.4, 0, 0.2, 1);
}
.onboarding-spotlight__pulse {
position: absolute;
inset: -4px;
border-radius: 14px;
border: 1px solid rgba(0, 255, 136, 0.3);
animation: ob-pulse 2.4s ease-out infinite;
}
.onboarding-spotlight__pulse--delay {
animation-delay: 1.2s;
}
@keyframes ob-pulse {
0% { transform: scale(1); opacity: 0.6; }
100% { transform: scale(1.08); opacity: 0; }
}
/* ─── Connector SVG ────────────────────────────────────────── */
.onboarding-connector {
position: fixed;
inset: 0;
z-index: 10002;
width: 100%;
height: 100%;
pointer-events: none;
}
.onboarding-connector__path {
animation: ob-dash 1.6s linear infinite;
stroke-dashoffset: 0;
}
@keyframes ob-dash {
to { stroke-dashoffset: -24; }
}
.onboarding-connector__dot {
animation: ob-dot-pulse 1.4s ease-in-out infinite;
}
@keyframes ob-dot-pulse {
0%, 100% { r: 4; opacity: 0.7; }
50% { r: 7; opacity: 1; }
}
/* ─── Tooltip card ─────────────────────────────────────────── */
.onboarding-tooltip {
position: fixed;
z-index: 10003;
width: min(92vw, 360px);
padding: 20px 22px 18px;
border-radius: 16px;
background: var(--bg-elevated, #1e1e1e);
border: 1px solid var(--border-subtle, #333);
box-shadow:
0 12px 40px rgba(0, 0, 0, 0.5),
0 0 0 1px rgba(255, 255, 255, 0.03) inset,
0 0 60px rgba(0, 255, 136, 0.04);
pointer-events: auto;
transition: left 0.32s cubic-bezier(0.4, 0, 0.2, 1),
top 0.32s cubic-bezier(0.4, 0, 0.2, 1);
}
.onboarding-tooltip--pop {
animation: ob-pop-in 0.35s cubic-bezier(0.34, 1.56, 0.64, 1);
}
@keyframes ob-pop-in {
from { opacity: 0; transform: scale(0.92) translateY(6px); }
to { opacity: 1; transform: scale(1) translateY(0); }
}
/* Tooltip arrow — CSS triangle pointing toward target */
.onboarding-tooltip__arrow {
position: absolute;
width: 0;
height: 0;
display: none; /* replaced by SVG connector; fallback for simple cases */
}
/* ─── Tooltip head ─────────────────────────────────────────── */
.onboarding-tooltip__head {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 10px;
}
.onboarding-tooltip__phase-badge {
font-size: 10px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.08em;
padding: 2px 10px;
border-radius: 999px;
background: rgba(var(--accent-rgb, 0, 255, 136), 0.12);
color: var(--accent, #00ff88);
}
.onboarding-tooltip__counter {
font-size: 11px;
font-weight: 600;
color: var(--fg-muted, #777);
font-variant-numeric: tabular-nums;
}
/* ─── Tooltip body ─────────────────────────────────────────── */
.onboarding-tooltip__title {
display: block;
font-size: 17px;
font-weight: 700;
color: var(--fg-body, #e5e5e5);
margin-bottom: 6px;
line-height: 1.3;
}
.onboarding-tooltip__desc {
font-size: 13px;
line-height: 1.6;
color: var(--fg-muted, #999);
margin: 0 0 4px;
}
.onboarding-tooltip__action-hint {
font-size: 13px;
line-height: 1.5;
color: var(--accent, #00ff88);
margin: 8px 0 4px;
font-weight: 600;
animation: ob-hint-blink 1.8s ease-in-out infinite;
}
@keyframes ob-hint-blink {
0%, 100% { opacity: 1; }
50% { opacity: 0.5; }
}
/* ─── Tooltip actions ──────────────────────────────────────── */
.onboarding-tooltip__actions {
display: flex;
align-items: center;
gap: 10px;
flex-wrap: wrap;
margin-top: 14px;
}
.onboarding-tooltip__btn {
display: inline-flex;
align-items: center;
gap: 5px;
padding: 7px 16px;
border-radius: 8px;
border: none;
font-size: 13px;
font-weight: 600;
cursor: pointer;
transition: background 0.15s ease, opacity 0.15s ease, transform 0.12s ease;
font-family: inherit;
white-space: nowrap;
}
.onboarding-tooltip__btn:active {
transform: scale(0.97);
}
.onboarding-tooltip__btn--primary {
background: var(--accent, #00ff88);
color: #000;
}
.onboarding-tooltip__btn--primary:hover {
opacity: 0.85;
}
.onboarding-tooltip__btn--ghost {
background: transparent;
color: var(--fg-muted, #888);
border: 1px solid var(--border-subtle, #444);
}
.onboarding-tooltip__btn--ghost:hover {
color: var(--fg-body, #e5e5e5);
border-color: var(--fg-muted, #888);
}
.onboarding-tooltip__wait-hint {
font-size: 11px;
color: var(--fg-muted, #777);
font-style: italic;
animation: ob-hint-blink 2s ease-in-out infinite;
}
/* ─── Progress bar (bottom-right) ──────────────────────────── */
.onboarding-progress {
position: fixed;
z-index: 10002;
bottom: 24px;
right: 32px;
display: flex;
align-items: center;
gap: 40px;
pointer-events: auto;
}
.onboarding-progress__phase {
display: flex;
flex-direction: column;
align-items: center;
gap: 5px;
}
.onboarding-progress__dot {
width: 10px;
height: 10px;
border-radius: 50%;
background: var(--border-subtle, #444);
transition: background 0.3s ease, transform 0.3s ease, box-shadow 0.3s ease;
}
.onboarding-progress__dot.is-active {
background: var(--accent, #00ff88);
transform: scale(1.4);
box-shadow: 0 0 12px rgba(0, 255, 136, 0.4);
}
.onboarding-progress__dot.is-done {
background: rgba(var(--accent-rgb, 0, 255, 136), 0.5);
}
.onboarding-progress__phase span {
font-size: 10px;
color: var(--fg-muted, #666);
white-space: nowrap;
}
.onboarding-progress__phase .is-active + span,
.onboarding-progress__phase .is-done + span {
color: var(--fg-body, #ccc);
}
/* ─── Responsive ───────────────────────────────────────────── */
@media (max-width: 640px) {
.onboarding-tooltip {
width: calc(100vw - 20px);
left: 10px !important;
}
.onboarding-progress {
gap: 20px;
}
}
+72 -97
View File
@@ -1542,9 +1542,7 @@
.product-clone-page[data-tool="clone"] .clone-ai-replicate-panel {
display: grid;
flex: 0 0 auto;
grid-template-rows: auto auto minmax(0, 1fr);
gap: 9px;
min-height: 0;
overflow: visible;
border: 1px solid #303540;
border-radius: 14px;
@@ -1874,11 +1872,8 @@
.product-clone-page[data-tool="clone"] .clone-ai-module-panel {
display: grid;
flex: 0 0 272px;
grid-template-rows: auto minmax(0, 1fr);
flex: 0 0 auto;
gap: 10px;
height: 272px;
min-height: 0;
border: 1px solid #303540;
border-radius: 14px;
background: #1c1f26;
@@ -1906,25 +1901,6 @@
grid-template-columns: repeat(2, minmax(0, 1fr));
align-content: start;
gap: 8px;
min-height: 0;
overflow-y: auto;
padding-right: 4px;
scrollbar-width: thin;
scrollbar-color: #3d4552 #171a20;
}
.product-clone-page[data-tool="clone"] .clone-ai-module-list::-webkit-scrollbar {
width: 6px;
}
.product-clone-page[data-tool="clone"] .clone-ai-module-list::-webkit-scrollbar-track {
border-radius: 999px;
background: #171a20;
}
.product-clone-page[data-tool="clone"] .clone-ai-module-list::-webkit-scrollbar-thumb {
border-radius: 999px;
background: #3d4552;
}
.product-clone-page[data-tool="clone"] .clone-ai-module-list button {
@@ -1981,11 +1957,8 @@
.product-clone-page[data-tool="clone"] .clone-ai-model-panel {
display: grid;
flex: 0 0 272px;
grid-template-rows: auto minmax(0, 1fr);
flex: 0 0 auto;
gap: 10px;
height: 272px;
min-height: 0;
border: 1px solid #303540;
border-radius: 14px;
background: #1c1f26;
@@ -2032,25 +2005,7 @@
}
.product-clone-page[data-tool="clone"] .clone-ai-model-scroll {
min-height: 0;
overflow-y: auto;
padding-right: 4px;
scrollbar-width: thin;
scrollbar-color: #3d4552 #171a20;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-scroll::-webkit-scrollbar {
width: 6px;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-scroll::-webkit-scrollbar-track {
border-radius: 999px;
background: #171a20;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-scroll::-webkit-scrollbar-thumb {
border-radius: 999px;
background: #3d4552;
overflow: visible;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-scenes,
@@ -2223,16 +2178,12 @@
z-index: 30;
display: grid;
gap: 4px;
max-height: 150px;
overflow-y: auto;
border: 1px solid #303540;
border-radius: 8px;
background: #22252d;
background-image: none;
padding: 5px;
box-shadow: none;
scrollbar-width: thin;
scrollbar-color: #3d4552 #171a20;
transform-origin: top center;
animation: clone-ai-model-select-pop 160ms cubic-bezier(0.2, 0.82, 0.2, 1) both;
}
@@ -2244,20 +2195,6 @@
animation-name: clone-ai-model-select-pop-up;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-select__menu::-webkit-scrollbar {
width: 6px;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-select__menu::-webkit-scrollbar-track {
border-radius: 999px;
background: #171a20;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-select__menu::-webkit-scrollbar-thumb {
border-radius: 999px;
background: #3d4552;
}
.product-clone-page[data-tool="clone"] .clone-ai-model-select__menu button {
display: flex;
align-items: center;
@@ -2351,31 +2288,14 @@
.product-clone-page[data-tool="clone"] .clone-ai-video-panel {
display: grid;
flex: 1 1 auto;
flex: 0 0 auto;
align-content: start;
gap: 12px;
min-height: 0;
overflow-y: auto;
overflow: visible;
border: 1px solid #303540;
border-radius: 14px;
background: #1c1f26;
padding: 12px;
scrollbar-width: thin;
scrollbar-color: #3d4552 #171a20;
}
.product-clone-page[data-tool="clone"] .clone-ai-video-panel::-webkit-scrollbar {
width: 6px;
}
.product-clone-page[data-tool="clone"] .clone-ai-video-panel::-webkit-scrollbar-track {
border-radius: 999px;
background: #171a20;
}
.product-clone-page[data-tool="clone"] .clone-ai-video-panel::-webkit-scrollbar-thumb {
border-radius: 999px;
background: #3d4552;
}
.product-clone-page[data-tool="clone"] .clone-ai-video-section {
@@ -8611,31 +8531,86 @@
transition: none;
}
.clone-ai-video-outfit-upload {
display: flex;
align-items: center;
gap: 10px;
position: relative;
display: grid;
gap: 8px;
margin-top: 6px;
}
.clone-ai-video-outfit-upload-btn {
padding: 7px 16px;
border: 1px solid var(--border-subtle);
border-radius: 8px;
background: var(--bg-inset);
color: var(--fg-body);
display: flex;
width: 100%;
min-height: 118px;
flex-direction: column;
align-items: center;
justify-content: center;
border: 1.5px dashed var(--ecm-line, var(--border-subtle));
border-radius: var(--ecm-radius-md, 14px);
background:
linear-gradient(180deg, rgba(255, 255, 255, 0.035), transparent 65%),
var(--ecm-inset, var(--bg-inset));
color: var(--ecm-text, var(--fg-body));
font-size: 13px;
font-weight: 850;
line-height: 1.35;
cursor: pointer;
transition: border-color 150ms, background 150ms;
text-align: center;
transition:
border-color 150ms,
background 150ms,
color 150ms,
transform 150ms;
}
.clone-ai-video-outfit-upload-btn::before {
display: grid;
width: 38px;
height: 38px;
place-items: center;
margin-bottom: 8px;
border: 1px solid rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.26);
border-radius: 999px;
background: rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.08);
color: var(--ecm-accent, var(--accent));
content: "+";
font-size: 24px;
font-weight: 800;
line-height: 1;
}
.clone-ai-video-outfit-upload-btn:hover {
border-color: var(--border-default);
background: var(--bg-hover);
border-color: rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.48);
background:
linear-gradient(180deg, rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.08), transparent 72%),
var(--ecm-inset-hover, var(--bg-hover));
color: var(--ecm-text, var(--fg-body));
}
.clone-ai-video-outfit-upload-btn:active {
transform: scale(0.99);
}
.clone-ai-video-outfit-info {
position: absolute;
top: 10px;
right: 10px;
max-width: calc(100% - 20px);
padding: 4px 8px;
border: 1px solid rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.28);
border-radius: 999px;
background: rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.1);
font-size: 12px;
color: var(--accent);
font-weight: 850;
color: var(--ecm-accent, var(--accent));
pointer-events: none;
}
.clone-ai-video-outfit-upload:has(.clone-ai-video-outfit-info) .clone-ai-video-outfit-upload-btn {
border-style: solid;
border-color: rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.38);
background:
linear-gradient(180deg, rgba(var(--ecm-accent-rgb, 0, 255, 136), 0.075), transparent 74%),
var(--ecm-inset, var(--bg-inset));
}
/* Ecommerce generation page SaaS polish: visual-only refinement for the product creation workspace. */
+533 -15
View File
@@ -402,21 +402,6 @@
transform: translateZ(20px) scale(1.02);
}
.omni-home__carousel-card-label {
position: absolute;
bottom: 12px;
left: 14px;
z-index: 2;
padding: 4px 12px;
border-radius: 999px;
background: rgba(var(--accent-rgb, 0, 255, 136), 0.16);
border: 1px solid rgba(var(--accent-rgb, 0, 255, 136), 0.24);
color: var(--fg-body, #f3f5f2);
font-size: 12px;
font-weight: 900;
white-space: nowrap;
}
.omni-home__carousel-card:hover {
box-shadow:
0 28px 58px rgb(0 0 0 / 34%),
@@ -2479,3 +2464,536 @@
font-size: 12px;
}
}
/* ===== Home page product polish and responsive hardening ===== */
.web-shell[data-view="home"] .omni-home {
--home-card-radius: 14px;
--home-panel-radius: 18px;
--home-safe-inline: clamp(16px, 4.6vw, 72px);
scroll-padding-top: 16px;
}
.web-shell[data-view="home"] .omni-home__scrim {
background:
linear-gradient(180deg, rgba(5, 8, 13, 0.82), rgba(5, 8, 13, 0.52) 38%, rgba(5, 8, 13, 0.9)),
radial-gradient(circle at 50% 24%, rgba(var(--accent-rgb), 0.1), transparent 36%),
linear-gradient(90deg, rgba(5, 8, 13, 0.86), rgba(5, 8, 13, 0.48) 48%, rgba(5, 8, 13, 0.86));
}
.web-shell[data-view="home"] .omni-home__shell {
padding: clamp(28px, 4.4vw, 58px) var(--home-safe-inline) clamp(34px, 4.8vw, 70px);
}
.web-shell[data-view="home"] .omni-home__hero {
width: min(100%, 1240px);
gap: clamp(16px, 2vw, 24px);
}
.web-shell[data-view="home"] .omni-home__copy {
gap: 14px;
}
.web-shell[data-view="home"] .omni-home__copy h1 {
max-width: min(100%, 920px);
white-space: normal;
text-wrap: balance;
font-size: clamp(34px, 4vw, 62px);
letter-spacing: 0;
text-shadow: 0 18px 54px rgb(0 0 0 / 34%);
}
.web-shell[data-view="home"] .omni-home__copy p {
max-width: 680px;
color: rgb(235 245 240 / 72%);
font-size: clamp(14px, 1.25vw, 18px);
line-height: 1.7;
}
.web-shell[data-view="home"] .omni-home__actions {
width: min(100%, 620px);
gap: 12px;
}
.web-shell[data-view="home"] .omni-home__entry {
min-height: 64px;
border-color: rgb(255 255 255 / 11%);
border-radius: 12px;
background:
linear-gradient(180deg, rgb(255 255 255 / 7%), rgb(255 255 255 / 3%)),
rgb(12 16 17 / 78%);
box-shadow:
inset 0 1px 0 rgb(255 255 255 / 8%),
0 16px 34px rgb(0 0 0 / 22%);
}
.web-shell[data-view="home"] .omni-home__entry > span:not(.anticon) {
display: grid;
gap: 3px;
justify-items: center;
min-width: 0;
line-height: 1.1;
}
.web-shell[data-view="home"] .omni-home__entry small {
color: rgb(235 245 240 / 48%);
font-size: 11px;
font-weight: 760;
line-height: 1;
white-space: nowrap;
}
.web-shell[data-view="home"] .omni-home__entry--primary small {
color: rgb(6 16 20 / 62%);
}
.web-shell[data-view="home"] .omni-home__entry:hover {
border-color: rgb(0 255 136 / 34%);
background:
linear-gradient(180deg, rgb(255 255 255 / 9%), rgb(255 255 255 / 4%)),
rgb(17 22 22 / 88%);
}
.web-shell[data-view="home"] .omni-home__entry--primary,
.web-shell[data-view="home"] .omni-home__entry--primary:hover {
border-color: rgb(0 255 136 / 82%);
background:
linear-gradient(180deg, rgb(88 255 172 / 100%), rgb(0 255 136 / 100%));
box-shadow:
0 18px 42px rgb(0 255 136 / 14%),
inset 0 1px 0 rgb(255 255 255 / 28%);
}
.web-shell[data-view="home"] .omni-home__carousel {
width: min(100%, 1180px);
min-height: clamp(380px, 36vw, 560px);
}
.web-shell[data-view="home"] .omni-home__carousel-stage {
width: min(100%, 1180px);
height: clamp(360px, 34vw, 530px);
overflow: hidden;
-webkit-mask-image: linear-gradient(90deg, transparent 0%, #000 5%, #000 95%, transparent 100%);
mask-image: linear-gradient(90deg, transparent 0%, #000 5%, #000 95%, transparent 100%);
}
.web-shell[data-view="home"] .omni-home__carousel-card {
width: clamp(500px, 45vw, 760px);
height: clamp(281px, 25.31vw, 428px);
border: 1px solid rgb(255 255 255 / 5%);
border-radius: var(--home-panel-radius);
background:
linear-gradient(180deg, rgb(255 255 255 / 4%), rgb(0 0 0 / 5%)),
#07100f;
box-shadow:
0 18px 46px rgb(0 0 0 / 30%),
inset 0 1px 0 rgb(255 255 255 / 8%),
inset 0 0 0 1px rgb(255 255 255 / 2%);
}
.web-shell[data-view="home"] .omni-home__carousel-card::before {
background:
linear-gradient(180deg, rgb(2 12 12 / 18%), transparent 34%, rgb(3 9 10 / 30%)),
linear-gradient(90deg, rgb(4 10 12 / 38%), transparent 24%, transparent 76%, rgb(4 10 12 / 38%)),
radial-gradient(circle at 50% 58%, rgb(0 255 136 / 8%), transparent 54%);
opacity: 0.92;
}
.web-shell[data-view="home"] .omni-home__carousel-card::after {
content: "";
position: absolute;
inset: 0;
z-index: 3;
border-radius: inherit;
box-shadow:
inset 0 0 0 1px rgb(255 255 255 / 4%),
inset 0 -42px 64px rgb(2 8 9 / 32%),
inset 0 34px 56px rgb(255 255 255 / 4%);
pointer-events: none;
}
.web-shell[data-view="home"] .omni-home__carousel-card.is-active {
border-color: rgb(255 255 255 / 7%);
box-shadow:
0 26px 70px rgb(0 0 0 / 38%),
0 0 0 1px rgb(255 255 255 / 4%),
0 0 46px rgb(0 0 0 / 18%),
inset 0 1px 0 rgb(255 255 255 / 10%);
}
.web-shell[data-view="home"] .omni-home__carousel-card:not(.is-active) {
filter: saturate(0.62) brightness(0.66) contrast(0.96);
}
.web-shell[data-view="home"] .omni-home__carousel-stage .omni-home__carousel-card img,
.web-shell[data-view="home"] .omni-home__carousel-stage .omni-home__carousel-card.is-active img {
object-fit: contain;
transform: translateZ(12px);
}
.web-shell[data-view="home"] .omni-home__carousel-stage .omni-home__carousel-card img {
filter:
saturate(0.72)
contrast(0.98)
brightness(0.72)
hue-rotate(8deg)
drop-shadow(0 18px 18px rgb(0 0 0 / 18%));
}
.web-shell[data-view="home"] .omni-home__carousel-stage .omni-home__carousel-card.is-active img {
filter:
saturate(0.9)
contrast(1.03)
brightness(0.88)
hue-rotate(6deg)
drop-shadow(0 22px 22px rgb(0 0 0 / 18%));
}
.web-shell[data-view="home"] .omni-home__feature-page {
padding-inline: var(--home-safe-inline);
}
.web-shell[data-view="home"] .omni-home__feature-page::before {
background:
linear-gradient(90deg, rgb(5 8 13 / 96%) 0%, rgb(5 8 13 / 80%) 40%, rgb(5 8 13 / 52%) 100%),
linear-gradient(180deg, rgb(255 255 255 / 4%), transparent 34%);
}
.web-shell[data-view="home"] .omni-home__feature-visual {
border-radius: var(--home-panel-radius);
border-color: rgb(255 255 255 / 10%);
background:
linear-gradient(180deg, rgb(255 255 255 / 5%), rgb(255 255 255 / 2%)),
#0d1111;
box-shadow:
0 24px 68px rgb(0 0 0 / 34%),
inset 0 1px 0 rgb(255 255 255 / 8%);
}
.web-shell[data-view="home"] .omni-home__feature-copy h2,
.web-shell[data-view="home"] .omni-home__experience-copy h2 {
text-wrap: balance;
}
.web-shell[data-view="home"] .omni-home__feature-copy p,
.web-shell[data-view="home"] .omni-home__experience-copy p {
color: rgb(232 238 236 / 74%);
}
.web-shell[data-view="home"] .omni-home__feature-page.is-script,
.web-shell[data-view="home"] .omni-home__feature-page.is-model,
.web-shell[data-view="home"] .omni-home__feature-page.is-ecommerce {
padding-inline: clamp(10px, 1.8vw, 28px);
}
.web-shell[data-view="home"] .omni-home__feature-page.is-script .omni-home__feature-visual,
.web-shell[data-view="home"] .omni-home__feature-page.is-model .omni-home__feature-visual,
.web-shell[data-view="home"] .omni-home__feature-page.is-ecommerce .omni-home__feature-visual {
border-radius: 12px;
}
.web-shell[data-view="home"] .omni-home__experience {
background:
linear-gradient(112deg, rgb(0 255 136 / 10%) 0 1px, transparent 1px 20%),
linear-gradient(68deg, rgb(0 255 136 / 7%) 0 1px, transparent 1px 22%),
linear-gradient(180deg, #070b10 0%, #05080d 100%);
}
.web-shell[data-view="home"] .omni-home__experience-route {
border-radius: var(--home-card-radius);
}
.web-shell[data-view="home"] :is(
.omni-home__shell,
.omni-home__feature-page,
.omni-home__toolbox-page,
.omni-home__experience
) {
scroll-margin-top: 64px;
}
.web-shell[data-view="home"] .cookie-consent {
right: clamp(14px, 2vw, 24px);
bottom: 12px;
width: min(560px, calc(100vw - 28px));
gap: 10px;
padding: 10px 12px;
border-color: rgb(0 255 136 / 20%);
border-radius: 13px;
background:
linear-gradient(180deg, rgb(255 255 255 / 6%), rgb(255 255 255 / 3%)),
rgb(13 17 18 / 92%);
box-shadow:
0 18px 46px rgb(0 0 0 / 30%),
inset 0 1px 0 rgb(255 255 255 / 7%);
}
.web-shell[data-view="home"] .cookie-consent strong {
font-size: 13px;
}
.web-shell[data-view="home"] .cookie-consent p {
margin-top: 4px;
overflow: hidden;
font-size: 12px;
line-height: 1.32;
text-overflow: ellipsis;
white-space: nowrap;
}
.web-shell[data-view="home"] .cookie-consent__actions {
gap: 8px;
}
.web-shell[data-view="home"] .cookie-consent__actions a,
.web-shell[data-view="home"] .cookie-consent__actions button {
min-height: 32px;
padding-inline: 12px;
font-size: 12px;
}
@media (max-width: 1200px) {
.web-shell[data-view="home"] .omni-home__carousel {
min-height: clamp(340px, 42vw, 500px);
}
.web-shell[data-view="home"] .omni-home__carousel-stage {
width: min(100%, 1020px);
height: clamp(320px, 39vw, 470px);
}
.web-shell[data-view="home"] .omni-home__carousel-card {
width: clamp(470px, 57vw, 660px);
height: clamp(264px, 32.06vw, 371px);
}
}
@media (max-width: 980px) {
.web-shell[data-view="home"] .omni-home {
scroll-snap-type: none;
overscroll-behavior-y: auto;
}
.web-shell[data-view="home"] .omni-home__shell {
min-height: auto;
padding-top: clamp(24px, 4vw, 38px);
padding-bottom: clamp(34px, 6vw, 58px);
}
.web-shell[data-view="home"] .omni-home__hero {
align-content: start;
}
.web-shell[data-view="home"] .omni-home__actions {
grid-template-columns: repeat(3, minmax(0, 1fr));
}
.web-shell[data-view="home"] .omni-home__entry {
min-height: 58px;
padding-inline: 18px;
font-size: 15px;
}
.web-shell[data-view="home"] .omni-home__entry small {
font-size: 10px;
}
.web-shell[data-view="home"] .omni-home__carousel {
width: min(100%, 820px);
min-height: clamp(290px, 54vw, 430px);
}
.web-shell[data-view="home"] .omni-home__carousel-stage {
width: min(100%, 820px);
height: clamp(280px, 51vw, 410px);
overflow: hidden;
}
.web-shell[data-view="home"] .omni-home__carousel-deck {
transform: scale(0.96);
}
.web-shell[data-view="home"] .omni-home__feature-page,
.web-shell[data-view="home"] .omni-home__experience {
min-height: auto;
padding-block: clamp(44px, 7vw, 72px);
}
.web-shell[data-view="home"] .omni-home__feature-copy h2,
.web-shell[data-view="home"] .omni-home__experience-copy h2 {
font-size: clamp(34px, 7vw, 58px);
}
.web-shell[data-view="home"] .omni-home__feature-copy p,
.web-shell[data-view="home"] .omni-home__experience-copy p {
font-size: clamp(15px, 2.4vw, 20px);
}
.web-shell[data-view="home"] .omni-home__feature-page.is-script .omni-home__feature-visual,
.web-shell[data-view="home"] .omni-home__feature-page.is-model .omni-home__feature-visual,
.web-shell[data-view="home"] .omni-home__feature-page.is-ecommerce .omni-home__feature-visual {
height: auto;
min-height: clamp(520px, 78vw, 720px);
}
.web-shell[data-view="home"] .omni-home__experience {
gap: 24px;
}
.web-shell[data-view="home"] .omni-home__experience-visual {
min-height: 260px;
}
}
@media (max-width: 720px) {
.web-shell[data-view="home"] .omni-home__copy h1 {
font-size: clamp(30px, 9vw, 44px);
line-height: 1.08;
}
.web-shell[data-view="home"] .omni-home__copy p {
max-width: 520px;
font-size: 14px;
}
.web-shell[data-view="home"] .omni-home__actions {
grid-template-columns: 1fr;
width: min(100%, 420px);
}
.web-shell[data-view="home"] .omni-home__entry {
justify-content: center;
min-height: 54px;
width: 100%;
}
.web-shell[data-view="home"] .omni-home__entry > span:not(.anticon) {
gap: 4px;
}
.web-shell[data-view="home"] .omni-home__carousel {
min-height: clamp(260px, 70vw, 360px);
}
.web-shell[data-view="home"] .omni-home__carousel-stage {
height: clamp(248px, 66vw, 340px);
}
.web-shell[data-view="home"] .omni-home__carousel-deck {
transform: scale(1);
}
.web-shell[data-view="home"] .omni-home__carousel-card {
width: min(94vw, 520px);
height: min(52.88vw, 292px);
min-height: 206px;
}
.web-shell[data-view="home"] .omni-home__feature-stats {
position: relative;
inset: auto;
margin-top: 18px;
max-width: none;
justify-content: flex-start;
}
.web-shell[data-view="home"] .omni-home__experience-routes {
grid-template-columns: 1fr 1fr;
gap: 10px;
}
.web-shell[data-view="home"] .omni-home__experience-route {
min-height: 84px;
padding: 13px 14px;
}
.web-shell[data-view="home"] .omni-home__experience-route b {
font-size: 22px;
}
}
@media (max-width: 480px) {
.web-shell[data-view="home"] .omni-home {
--home-safe-inline: 14px;
}
.web-shell[data-view="home"] .omni-home__shell {
padding-top: 20px;
padding-bottom: 32px;
}
.web-shell[data-view="home"] .omni-home__hero {
gap: 14px;
}
.web-shell[data-view="home"] .omni-home__copy {
gap: 10px;
}
.web-shell[data-view="home"] .omni-home__carousel {
min-height: clamp(228px, 68vw, 286px);
}
.web-shell[data-view="home"] .omni-home__carousel-stage {
height: clamp(218px, 64vw, 274px);
}
.web-shell[data-view="home"] .omni-home__carousel-deck {
transform: scale(1);
}
.web-shell[data-view="home"] .omni-home__carousel-card {
width: 94vw;
height: 52.88vw;
min-height: 188px;
}
.web-shell[data-view="home"] .omni-home__feature-page,
.web-shell[data-view="home"] .omni-home__experience {
padding-block: 34px 44px;
}
.web-shell[data-view="home"] :is(
.omni-home__shell,
.omni-home__feature-page,
.omni-home__toolbox-page,
.omni-home__experience
) {
scroll-margin-top: 60px;
}
.web-shell[data-view="home"] .omni-home__feature-page.is-script .omni-home__feature-visual,
.web-shell[data-view="home"] .omni-home__feature-page.is-model .omni-home__feature-visual,
.web-shell[data-view="home"] .omni-home__feature-page.is-ecommerce .omni-home__feature-visual {
min-height: 560px;
}
.web-shell[data-view="home"] .omni-home__experience-routes {
grid-template-columns: 1fr;
}
.web-shell[data-view="home"] .cookie-consent {
right: 10px;
bottom: max(10px, env(safe-area-inset-bottom));
grid-template-columns: 1fr;
width: calc(100vw - 20px);
padding: 12px;
}
.web-shell[data-view="home"] .cookie-consent p {
display: -webkit-box;
overflow: hidden;
text-overflow: initial;
white-space: normal;
-webkit-box-orient: vertical;
-webkit-line-clamp: 2;
}
.web-shell[data-view="home"] .cookie-consent__actions {
justify-content: stretch;
}
.web-shell[data-view="home"] .cookie-consent__actions a,
.web-shell[data-view="home"] .cookie-consent__actions button {
flex: 1 1 0;
}
}
+11 -32
View File
@@ -216,10 +216,10 @@
.more-card--featured {
display: grid;
grid-template-columns: 54px minmax(0, 1fr);
grid-template-columns: minmax(0, 1fr);
align-items: start;
justify-items: stretch;
gap: 16px;
gap: 12px;
min-height: 336px;
padding: 20px;
border-color: rgba(var(--accent-rgb), 0.2);
@@ -251,22 +251,6 @@
box-shadow: var(--more-card-shadow), 0 0 0 1px rgba(var(--accent-rgb), 0.12);
}
.more-card__featured-icon {
display: grid;
place-items: center;
width: 54px;
height: 54px;
border: 1px solid rgba(var(--accent-rgb), 0.24);
border-radius: var(--radius-xs, 8px);
background:
linear-gradient(180deg, rgba(var(--accent-rgb), 0.18), rgba(var(--accent-rgb), 0.08)),
var(--bg-inset);
color: var(--accent);
font-size: 24px;
flex-shrink: 0;
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.06);
}
.more-card__featured-body {
display: flex;
flex-direction: column;
@@ -281,7 +265,12 @@
.more-card--featured .more-card__preview {
width: 100%;
min-height: 0;
aspect-ratio: 16 / 9;
aspect-ratio: 4 / 3;
}
.more-card--featured .more-card__preview-frame img {
padding: 8px;
object-fit: contain;
}
.more-card--featured.more-card--no-preview {
@@ -449,7 +438,7 @@
}
.more-card__icon {
display: grid;
display: none;
place-items: center;
width: 38px;
height: 38px;
@@ -464,6 +453,7 @@
}
.more-card--recent .more-card__icon {
display: grid;
width: 30px;
height: 30px;
font-size: 14px;
@@ -1221,18 +1211,12 @@
}
.more-card--featured {
grid-template-columns: 44px minmax(0, 1fr);
grid-template-columns: minmax(0, 1fr);
min-height: 0;
padding: 16px;
gap: 12px;
}
.more-card__featured-icon {
width: 42px;
height: 42px;
font-size: 20px;
}
.more-card__featured-body strong {
font-size: 16px;
}
@@ -1312,11 +1296,6 @@
padding: 15px;
}
.more-card__featured-icon {
width: 40px;
height: 40px;
}
.more-card {
gap: 10px;
min-height: 0;
+165 -26
View File
@@ -684,12 +684,6 @@
}
@container (max-width: 1120px) {
.omni-script-review-showcase {
grid-template-columns: 1fr;
gap: 18px;
overflow-y: auto;
}
.srs-left-panel {
grid-template-rows: auto auto auto;
gap: 16px;
@@ -711,13 +705,32 @@
}
.srs-results-panel {
grid-template-rows: auto minmax(320px, auto) auto;
grid-template-rows: auto minmax(300px, auto) auto;
}
}
@media (max-width: 980px) {
.omni-script-review-showcase {
padding: 22px;
@container (max-width: 880px) {
.srs-brand-section {
gap: 12px;
}
.srs-brand-section h1 {
max-width: 100%;
font-size: clamp(28px, 8cqw, 44px);
}
.srs-brand-section p {
font-size: clamp(14px, 3.2cqw, 17px);
line-height: 1.65;
}
.srs-point-card {
min-height: 0;
border-radius: 10px;
}
.srs-results-panel {
gap: 14px;
}
.srs-score-hero {
@@ -729,33 +742,118 @@
display: none;
}
.srs-chart-body,
.srs-triple-section,
.srs-point-list {
grid-template-columns: 1fr;
}
.srs-chart-col {
grid-template-columns: minmax(70px, 92px) minmax(0, 1fr);
grid-template-rows: auto;
align-items: center;
justify-items: stretch;
}
.srs-chart-bar-wrap {
min-height: 80px;
}
.srs-chart-col-label {
text-align: left;
.srs-score-summary {
font-size: clamp(13px, 3cqw, 16px);
line-height: 1.55;
}
}
@media (max-width: 560px) {
.omni-script-review-showcase {
@container (max-width: 720px) {
.srs-chart-card {
padding: 16px;
}
.srs-chart-title {
letter-spacing: 0.04em;
}
.srs-chart-body {
grid-template-columns: 1fr;
gap: 12px;
padding-top: 14px;
}
.srs-chart-col {
grid-template-columns: minmax(94px, 0.36fr) minmax(0, 1fr);
grid-template-rows: auto;
align-items: center;
justify-items: stretch;
min-height: 46px;
border: 1px solid rgba(255, 255, 255, 0.055);
border-radius: 10px;
background: rgba(255, 255, 255, 0.03);
padding: 10px;
}
.srs-chart-bar-wrap {
grid-column: 2;
grid-row: 1;
width: 100%;
height: 16px;
min-height: 16px;
}
.srs-chart-bar-bg,
.srs-chart-bar-fill {
width: 100%;
height: 100% !important;
border-radius: 999px;
}
.srs-chart-bar-bg {
left: 0;
transform: none;
}
.srs-chart-bar-fill {
transform-origin: left center;
}
.srs-chart-col:nth-child(1) .srs-chart-bar-fill { width: 80%; }
.srs-chart-col:nth-child(2) .srs-chart-bar-fill { width: 80%; }
.srs-chart-col:nth-child(3) .srs-chart-bar-fill { width: 100%; }
.srs-chart-col:nth-child(4) .srs-chart-bar-fill { width: 80%; }
.srs-chart-col:nth-child(5) .srs-chart-bar-fill { width: 66%; }
.srs-chart-col:nth-child(6) .srs-chart-bar-fill { width: 53%; }
.srs-chart-bar-score {
top: 50%;
right: 8px;
left: auto;
font-size: 12px;
transform: translateY(-50%);
}
.srs-chart-bar-sub,
.srs-chart-bar-star {
font-size: 10px;
}
.srs-chart-col-label {
grid-column: 1;
grid-row: 1;
text-align: left;
}
.srs-chart-col-name {
font-size: 13px;
}
.srs-chart-col-desc {
display: none;
}
.srs-triple-section {
gap: 12px;
}
.srs-section-card {
overflow: visible;
}
.srs-section-item-text {
display: block;
overflow: visible;
-webkit-line-clamp: initial;
}
}
@container (max-width: 560px) {
.srs-flow-card {
grid-template-columns: 1fr;
justify-items: stretch;
@@ -769,4 +867,45 @@
align-items: flex-start;
flex-direction: column;
}
.srs-score-tags {
gap: 6px;
}
.srs-score-tag {
min-height: 24px;
padding-inline: 9px;
}
.srs-chart-col {
grid-template-columns: 1fr;
gap: 8px;
}
.srs-chart-bar-wrap,
.srs-chart-col-label {
grid-column: 1;
grid-row: auto;
}
}
@media (max-width: 1120px) {
.omni-script-review-showcase {
grid-template-columns: 1fr;
gap: clamp(16px, 2.8vw, 24px);
overflow-y: visible;
padding: clamp(20px, 3vw, 34px);
}
}
@media (max-width: 880px) {
.omni-script-review-showcase {
padding: clamp(16px, 3.8vw, 24px);
}
}
@media (max-width: 560px) {
.omni-script-review-showcase {
padding: 16px;
}
}
+11
View File
@@ -1620,12 +1620,23 @@
border: 0;
background: transparent;
cursor: default;
position: relative;
}
.script-eval-v5-illustration:hover {
background: transparent;
}
.script-eval-v5-illustration.is-dragging .script-eval-v5-illustration-hit {
background: var(--v5-green-deep);
outline: 2px dashed var(--v5-green);
transform: scale(1.02);
}
.script-eval-v5-illustration .script-eval-v5-upload-drop-overlay {
border-radius: 10px;
}
.script-eval-v5-illustration-hit {
display: inline-flex;
flex-direction: column;
+19
View File
@@ -299,6 +299,25 @@
gap: 10px;
padding: 32px;
text-align: center;
border-radius: 12px;
cursor: pointer;
transition: background 0.18s ease, outline 0.18s ease, transform 0.18s ease;
}
.studio-canvas-ghost:hover {
background: rgba(var(--accent-rgb), 0.05);
outline: 1px dashed rgba(var(--accent-rgb), 0.25);
}
.studio-canvas-ghost:focus-visible {
outline: 2px solid var(--accent);
outline-offset: 2px;
}
.studio-canvas-ghost.is-dragging {
background: rgba(var(--accent-rgb), 0.1);
outline: 2px dashed var(--accent);
transform: scale(1.02);
}
.studio-canvas-ghost__icon {
+249
View File
@@ -889,3 +889,252 @@
animation: none;
}
}
/* ===== Home toolbox polish and responsive hardening ===== */
.web-shell[data-view="home"] .omni-home__toolbox-page {
--toolbox-radius-card: 16px;
--toolbox-radius-inner: 12px;
background:
linear-gradient(180deg, #070b10 0%, #05080d 100%),
radial-gradient(ellipse 70% 48% at 58% 42%, rgba(0, 255, 136, 0.045) 0%, transparent 70%);
}
.web-shell[data-view="home"] .omni-home__toolbox-shell {
width: min(100%, 1440px);
margin-inline: auto;
padding: clamp(34px, 5vw, 64px) clamp(18px, 5vw, 72px);
align-items: center;
}
.web-shell[data-view="home"] .omni-home__toolbox-left {
width: clamp(300px, 28vw, 420px);
gap: 14px;
padding-top: 0;
}
.web-shell[data-view="home"] .omni-home__toolbox-brand-icon {
width: 48px;
height: 48px;
border-radius: 13px;
box-shadow: 0 16px 32px rgba(0, 255, 136, 0.12);
}
.web-shell[data-view="home"] .omni-home__toolbox-brand-icon .anticon {
font-size: 24px;
}
.web-shell[data-view="home"] .omni-home__toolbox-brand-text {
font-size: clamp(24px, 2.3vw, 32px);
}
.web-shell[data-view="home"] .omni-home__toolbox-title {
color: #f7fff9;
background: none;
-webkit-text-fill-color: currentColor;
font-size: clamp(32px, 3.4vw, 46px);
letter-spacing: 0;
}
.web-shell[data-view="home"] .omni-home__toolbox-subtitle {
color: rgba(232, 238, 236, 0.68);
font-size: clamp(15px, 1.18vw, 17px);
}
.web-shell[data-view="home"] .omni-home__toolbox-item,
.web-shell[data-view="home"] .omni-home__toolbox-workflow,
.web-shell[data-view="home"] .omni-home__toolbox-card {
border-color: rgba(255, 255, 255, 0.1);
background:
linear-gradient(180deg, rgba(255, 255, 255, 0.065), rgba(255, 255, 255, 0.028)),
rgba(10, 15, 16, 0.78);
box-shadow:
inset 0 1px 0 rgba(255, 255, 255, 0.075),
0 18px 42px rgba(0, 0, 0, 0.2);
}
.web-shell[data-view="home"] .omni-home__toolbox-item {
border-radius: 14px;
padding: 14px 16px;
}
.web-shell[data-view="home"] .omni-home__toolbox-item:hover {
border-color: rgba(0, 255, 136, 0.24);
transform: translateX(3px);
}
.web-shell[data-view="home"] .omni-home__toolbox-item-icon {
width: 42px;
height: 42px;
border-radius: 11px;
font-size: 22px;
}
.web-shell[data-view="home"] .omni-home__toolbox-item-name {
font-size: 16px;
}
.web-shell[data-view="home"] .omni-home__toolbox-item-desc {
font-size: 13px;
color: rgba(232, 238, 236, 0.48);
}
.web-shell[data-view="home"] .omni-home__toolbox-workflow {
margin-top: 4px;
border-radius: 14px;
padding: 15px 17px;
}
.web-shell[data-view="home"] .omni-home__toolbox-workflow-label {
margin-bottom: 10px;
font-size: 12px;
letter-spacing: 0.04em;
}
.web-shell[data-view="home"] .omni-home__toolbox-workflow-steps {
gap: 8px;
font-size: 13px;
}
.web-shell[data-view="home"] .omni-home__toolbox-grid {
gap: 14px;
min-height: clamp(500px, 48vw, 680px);
}
.web-shell[data-view="home"] .omni-home__toolbox-card {
border-radius: var(--toolbox-radius-card);
transform: translateZ(0);
}
.web-shell[data-view="home"] .omni-home__toolbox-card:hover {
transform: translateY(-4px);
border-color: rgba(0, 255, 136, 0.24);
box-shadow:
0 22px 54px rgba(0, 0, 0, 0.28),
0 0 0 1px rgba(0, 255, 136, 0.07);
}
.web-shell[data-view="home"] .omni-home__toolbox-card-header {
padding: 15px 16px 0;
}
.web-shell[data-view="home"] .omni-home__toolbox-card-content {
padding: 10px 16px;
min-height: 0;
}
.web-shell[data-view="home"] .omni-home__toolbox-card-footer {
padding: 7px 16px 14px;
}
.web-shell[data-view="home"] :is(.toolbox-card1-side, .toolbox-card3-side, .toolbox-card4-side),
.web-shell[data-view="home"] :is(.toolbox-card1-img, .toolbox-card3-portrait, .toolbox-card4-img),
.web-shell[data-view="home"] .toolbox-card2-frame {
border-radius: var(--toolbox-radius-inner);
}
@media (max-width: 1160px) {
.web-shell[data-view="home"] .omni-home__toolbox-shell {
gap: 22px;
}
.web-shell[data-view="home"] .omni-home__toolbox-left {
width: clamp(280px, 32vw, 360px);
}
.web-shell[data-view="home"] .omni-home__toolbox-grid {
min-height: clamp(460px, 58vw, 620px);
}
}
@media (max-width: 980px) {
.web-shell[data-view="home"] .omni-home__toolbox-page {
min-height: auto;
}
.web-shell[data-view="home"] .omni-home__toolbox-shell {
min-height: auto;
padding-block: clamp(42px, 7vw, 64px);
}
.web-shell[data-view="home"] .omni-home__toolbox-left {
width: min(100%, 760px);
}
.web-shell[data-view="home"] .omni-home__toolbox-list {
display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr));
}
.web-shell[data-view="home"] .omni-home__toolbox-workflow {
width: 100%;
}
.web-shell[data-view="home"] .omni-home__toolbox-grid {
grid-template-columns: repeat(2, minmax(0, 1fr));
min-height: auto;
}
.web-shell[data-view="home"] .omni-home__toolbox-card {
min-height: clamp(230px, 34vw, 300px);
}
}
@media (max-width: 680px) {
.web-shell[data-view="home"] .omni-home__toolbox-shell {
padding-inline: 14px;
gap: 18px;
}
.web-shell[data-view="home"] .omni-home__toolbox-brand-icon {
width: 42px;
height: 42px;
}
.web-shell[data-view="home"] .omni-home__toolbox-brand-text {
font-size: 22px;
}
.web-shell[data-view="home"] .omni-home__toolbox-title {
font-size: clamp(26px, 7vw, 34px);
}
.web-shell[data-view="home"] .omni-home__toolbox-list,
.web-shell[data-view="home"] .omni-home__toolbox-grid {
grid-template-columns: 1fr;
}
.web-shell[data-view="home"] .omni-home__toolbox-item {
align-items: center;
}
.web-shell[data-view="home"] .omni-home__toolbox-card {
min-height: 236px;
}
.web-shell[data-view="home"] .omni-home__toolbox-card-header {
align-items: flex-start;
gap: 10px;
}
.web-shell[data-view="home"] .omni-home__toolbox-card-tag {
flex-shrink: 0;
}
}
@media (max-width: 420px) {
.web-shell[data-view="home"] .omni-home__toolbox-workflow-steps {
align-items: flex-start;
}
.web-shell[data-view="home"] .omni-home__toolbox-workflow-arrow {
display: none;
}
.web-shell[data-view="home"] .omni-home__toolbox-card {
min-height: 220px;
}
.web-shell[data-view="home"] .omni-home__toolbox-card-footer {
gap: 5px;
}
}
+29 -2
View File
@@ -144,6 +144,10 @@
}
.web-shell[data-ui-theme="dark-green"] .notification-center {
position: relative;
display: inline-flex;
align-items: center;
flex: 0 0 auto;
isolation: isolate;
}
@@ -170,10 +174,15 @@
}
.web-shell[data-ui-theme="dark-green"] .notification-center__panel {
position: absolute;
display: flex;
flex-direction: column;
top: calc(100% + 12px);
right: -88px;
z-index: 1200;
width: min(420px, calc(100vw - 24px));
max-height: min(560px, calc(100vh - 92px));
height: auto;
max-height: min(460px, calc(100dvh - 84px));
border: 1px solid var(--dg-line);
border-radius: 16px;
background: #151719;
@@ -231,9 +240,12 @@
}
.web-shell[data-ui-theme="dark-green"] .notification-center__list {
max-height: min(486px, calc(100vh - 158px));
flex: 1 1 auto;
min-height: 0;
max-height: min(386px, calc(100dvh - 158px));
padding: 8px;
overflow-y: auto;
overscroll-behavior: contain;
}
.web-shell[data-ui-theme="dark-green"] .notification-center__item {
@@ -10466,6 +10478,21 @@
gap: 6px;
}
.web-shell[data-ui-theme="dark-green"] .notification-center__panel {
right: clamp(-112px, -24vw, -92px);
width: min(360px, calc(100vw - 20px));
max-height: min(420px, calc(100dvh - 76px));
border-radius: 14px;
}
.web-shell[data-ui-theme="dark-green"] .notification-center__panel::before {
right: clamp(104px, 25vw, 124px);
}
.web-shell[data-ui-theme="dark-green"] .notification-center__list {
max-height: min(344px, calc(100dvh - 150px));
}
.web-shell[data-ui-theme="dark-green"] :is(.creator-button, .member-button) {
height: 32px;
padding: 0 10px;