切换到 keyframe 模式时自动设为 adaptive,API 根据首帧图片比例 自动匹配最接近的输出比例,避免图片与视频比例不匹配。 用户仍可手动选择固定比例覆盖。 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
233 lines
7.6 KiB
TypeScript
233 lines
7.6 KiB
TypeScript
import { useEffect, useCallback } from 'react';
|
|
import { useInputBarStore } from '../store/inputBar';
|
|
import { useGenerationStore } from '../store/generation';
|
|
import { Dropdown } from './Dropdown';
|
|
import type { CreationMode, AspectRatio, Duration, GenerationType, ModelOption } from '../types';
|
|
import styles from './Toolbar.module.css';
|
|
|
|
const VideoIcon = () => (
|
|
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
|
|
<polygon points="23 7 16 12 23 17 23 7" />
|
|
<rect x="1" y="5" width="15" height="14" rx="2" ry="2" />
|
|
</svg>
|
|
);
|
|
|
|
const ImageIcon = () => (
|
|
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="1.5">
|
|
<rect x="3" y="3" width="18" height="18" rx="2" />
|
|
<circle cx="8.5" cy="8.5" r="1.5" />
|
|
<polyline points="21 15 16 10 5 21" />
|
|
</svg>
|
|
);
|
|
|
|
const DiamondIcon = () => (
|
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
|
|
<path d="M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5" />
|
|
</svg>
|
|
);
|
|
|
|
const LightningIcon = () => (
|
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="1.5">
|
|
<path d="M13 2L3 14h9l-1 8 10-12h-9l1-8z" />
|
|
</svg>
|
|
);
|
|
|
|
const StarIcon = () => (
|
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
|
|
<path d="M12 2l3.09 6.26L22 9.27l-5 4.87 1.18 6.88L12 17.77l-6.18 3.25L7 14.14 2 9.27l6.91-1.01L12 2z" />
|
|
</svg>
|
|
);
|
|
|
|
const SwapIcon = () => (
|
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round">
|
|
<path d="M7 12h10M17 12l-3-3M17 12l-3 3M7 12l3-3M7 12l3 3" />
|
|
</svg>
|
|
);
|
|
|
|
const MonitorIcon = () => (
|
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
|
|
<rect x="2" y="3" width="20" height="14" rx="2" />
|
|
<line x1="8" y1="21" x2="16" y2="21" />
|
|
<line x1="12" y1="17" x2="12" y2="21" />
|
|
</svg>
|
|
);
|
|
|
|
const ClockIcon = () => (
|
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
|
|
<circle cx="12" cy="12" r="10" />
|
|
<polyline points="12 6 12 12 16 14" />
|
|
</svg>
|
|
);
|
|
|
|
const ChevronDown = () => (
|
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
|
|
<polyline points="6 9 12 15 18 9" />
|
|
</svg>
|
|
);
|
|
|
|
const generationTypeItems = [
|
|
{ label: '视频生成', value: 'video' as GenerationType, icon: <VideoIcon /> },
|
|
];
|
|
|
|
const modelItems = [
|
|
{ label: 'AirDrama', value: 'seedance_2.0' as ModelOption, icon: <DiamondIcon /> },
|
|
{ label: 'AirDrama Fast', value: 'seedance_2.0_fast' as ModelOption, icon: <LightningIcon /> },
|
|
];
|
|
|
|
const modeItems = [
|
|
{ label: '全能参考', value: 'universal' as CreationMode, icon: <StarIcon /> },
|
|
{ label: '首尾帧', value: 'keyframe' as CreationMode, icon: <SwapIcon /> },
|
|
];
|
|
|
|
const ratioItems = [
|
|
{ label: '21:9', value: '21:9' as AspectRatio },
|
|
{ label: '16:9', value: '16:9' as AspectRatio },
|
|
{ label: '4:3', value: '4:3' as AspectRatio },
|
|
{ label: '1:1', value: '1:1' as AspectRatio },
|
|
{ label: '3:4', value: '3:4' as AspectRatio },
|
|
{ label: '9:16', value: '9:16' as AspectRatio },
|
|
];
|
|
|
|
const keyframeRatioItems = [
|
|
{ label: '自适应', value: 'adaptive' as AspectRatio },
|
|
...ratioItems,
|
|
];
|
|
|
|
const durationItems = Array.from({ length: 12 }, (_, i) => {
|
|
const v = i + 4;
|
|
return { label: `${v}s`, value: String(v) };
|
|
});
|
|
|
|
const modeLabels: Record<CreationMode, string> = {
|
|
universal: '全能参考',
|
|
keyframe: '首尾帧',
|
|
};
|
|
|
|
export function Toolbar() {
|
|
const generationType = useInputBarStore((s) => s.generationType);
|
|
const setGenerationType = useInputBarStore((s) => s.setGenerationType);
|
|
const model = useInputBarStore((s) => s.model);
|
|
const setModel = useInputBarStore((s) => s.setModel);
|
|
const mode = useInputBarStore((s) => s.mode);
|
|
const switchMode = useInputBarStore((s) => s.switchMode);
|
|
const aspectRatio = useInputBarStore((s) => s.aspectRatio);
|
|
const setAspectRatio = useInputBarStore((s) => s.setAspectRatio);
|
|
const duration = useInputBarStore((s) => s.duration);
|
|
const setDuration = useInputBarStore((s) => s.setDuration);
|
|
const isSubmittable = useInputBarStore((s) => s.canSubmit());
|
|
const triggerInsertAt = useInputBarStore((s) => s.triggerInsertAt);
|
|
|
|
const isKeyframe = mode === 'keyframe';
|
|
|
|
const addTask = useGenerationStore((s) => s.addTask);
|
|
|
|
const handleSend = useCallback(() => {
|
|
if (!isSubmittable) return;
|
|
addTask();
|
|
}, [isSubmittable, addTask]);
|
|
|
|
const handleInsertAt = useCallback(() => {
|
|
triggerInsertAt();
|
|
}, [triggerInsertAt]);
|
|
|
|
// Keyboard shortcut: Ctrl/Cmd + Enter
|
|
useEffect(() => {
|
|
const handler = (e: KeyboardEvent) => {
|
|
if ((e.ctrlKey || e.metaKey) && e.key === 'Enter') {
|
|
handleSend();
|
|
}
|
|
};
|
|
document.addEventListener('keydown', handler);
|
|
return () => document.removeEventListener('keydown', handler);
|
|
}, [handleSend]);
|
|
|
|
return (
|
|
<div className={styles.toolbar}>
|
|
{/* Generation type — fixed to video */}
|
|
<button className={`${styles.btn} ${styles.primary}`}>
|
|
<VideoIcon />
|
|
<span className={styles.label}>视频生成</span>
|
|
</button>
|
|
|
|
{/* Model selector */}
|
|
<Dropdown
|
|
items={modelItems}
|
|
value={model}
|
|
onSelect={(v) => setModel(v as ModelOption)}
|
|
minWidth={160}
|
|
trigger={
|
|
<button className={styles.btn}>
|
|
{model === 'seedance_2.0_fast' ? <LightningIcon /> : <DiamondIcon />}
|
|
<span className={styles.label}>{model === 'seedance_2.0_fast' ? 'AirDrama Fast' : 'AirDrama'}</span>
|
|
<ChevronDown />
|
|
</button>
|
|
}
|
|
/>
|
|
|
|
{/* Mode selector */}
|
|
<Dropdown
|
|
items={modeItems}
|
|
value={mode}
|
|
onSelect={(v) => switchMode(v as CreationMode)}
|
|
minWidth={150}
|
|
trigger={
|
|
<button className={styles.btn}>
|
|
{isKeyframe ? <SwapIcon /> : <StarIcon />}
|
|
<span className={styles.label}>{modeLabels[mode]}</span>
|
|
<ChevronDown />
|
|
</button>
|
|
}
|
|
/>
|
|
|
|
{/* Aspect ratio */}
|
|
<Dropdown
|
|
items={isKeyframe ? keyframeRatioItems : ratioItems}
|
|
value={aspectRatio}
|
|
onSelect={(v) => setAspectRatio(v as AspectRatio)}
|
|
minWidth={100}
|
|
trigger={
|
|
<button className={styles.btn}>
|
|
<MonitorIcon />
|
|
<span className={styles.label}>{aspectRatio === 'adaptive' ? '自适应' : aspectRatio}</span>
|
|
</button>
|
|
}
|
|
/>
|
|
|
|
{/* Duration */}
|
|
<Dropdown
|
|
items={durationItems}
|
|
value={String(duration)}
|
|
onSelect={(v) => setDuration(Number(v) as Duration)}
|
|
minWidth={100}
|
|
trigger={
|
|
<button className={styles.btn}>
|
|
<ClockIcon />
|
|
<span className={styles.label}>{duration}s</span>
|
|
</button>
|
|
}
|
|
/>
|
|
|
|
{/* @ button - universal mode only */}
|
|
{!isKeyframe && (
|
|
<button className={styles.btn} onClick={handleInsertAt}>
|
|
<span style={{ fontSize: 15, fontWeight: 600 }}>@</span>
|
|
</button>
|
|
)}
|
|
|
|
{/* Spacer */}
|
|
<div className={styles.spacer} />
|
|
|
|
{/* Send button */}
|
|
<button
|
|
className={`${styles.sendBtn} ${isSubmittable ? styles.sendEnabled : styles.sendDisabled}`}
|
|
onClick={handleSend}
|
|
>
|
|
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="#fff" strokeWidth="2.5" strokeLinecap="round" strokeLinejoin="round">
|
|
<line x1="12" y1="19" x2="12" y2="5" />
|
|
<polyline points="5 12 12 5 19 12" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
);
|
|
}
|