refactor: move timeline clip into its separate component and type into /types
This commit is contained in:
277
apps/web/src/components/editor/timeline-clip.tsx
Normal file
277
apps/web/src/components/editor/timeline-clip.tsx
Normal file
@ -0,0 +1,277 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { useState } from "react";
|
||||||
|
import { Button } from "../ui/button";
|
||||||
|
import { MoreVertical, Scissors, Trash2 } from "lucide-react";
|
||||||
|
import { useMediaStore } from "@/stores/media-store";
|
||||||
|
import { useTimelineStore } from "@/stores/timeline-store";
|
||||||
|
import { usePlaybackStore } from "@/stores/playback-store";
|
||||||
|
import { useDragClip } from "@/hooks/use-drag-clip";
|
||||||
|
import AudioWaveform from "./audio-waveform";
|
||||||
|
import { toast } from "sonner";
|
||||||
|
import { TimelineClipProps, ResizeState } from "@/types/timeline";
|
||||||
|
|
||||||
|
export function TimelineClip({
|
||||||
|
clip,
|
||||||
|
track,
|
||||||
|
zoomLevel,
|
||||||
|
isSelected,
|
||||||
|
onContextMenu,
|
||||||
|
onClipMouseDown,
|
||||||
|
onClipClick,
|
||||||
|
}: TimelineClipProps) {
|
||||||
|
const { mediaItems } = useMediaStore();
|
||||||
|
const { updateClipTrim, addClipToTrack, removeClipFromTrack } =
|
||||||
|
useTimelineStore();
|
||||||
|
const { currentTime } = usePlaybackStore();
|
||||||
|
const { draggedClipId, getDraggedClipPosition } =
|
||||||
|
useDragClip(zoomLevel);
|
||||||
|
|
||||||
|
const [resizing, setResizing] = useState<ResizeState | null>(null);
|
||||||
|
const [clipMenuOpen, setClipMenuOpen] = useState(false);
|
||||||
|
|
||||||
|
const effectiveDuration = clip.duration - clip.trimStart - clip.trimEnd;
|
||||||
|
const clipWidth = Math.max(80, effectiveDuration * 50 * zoomLevel);
|
||||||
|
|
||||||
|
// Use real-time position during drag, otherwise use stored position
|
||||||
|
const dragPosition = getDraggedClipPosition(clip.id);
|
||||||
|
const clipStartTime = dragPosition !== null ? dragPosition : clip.startTime;
|
||||||
|
const clipLeft = clipStartTime * 50 * zoomLevel;
|
||||||
|
|
||||||
|
const isBeingDragged = draggedClipId === clip.id;
|
||||||
|
|
||||||
|
const getTrackColor = (type: string) => {
|
||||||
|
switch (type) {
|
||||||
|
case "video":
|
||||||
|
return "bg-blue-500/20 border-blue-500/30";
|
||||||
|
case "audio":
|
||||||
|
return "bg-green-500/20 border-green-500/30";
|
||||||
|
case "effects":
|
||||||
|
return "bg-purple-500/20 border-purple-500/30";
|
||||||
|
default:
|
||||||
|
return "bg-gray-500/20 border-gray-500/30";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleResizeStart = (
|
||||||
|
e: React.MouseEvent,
|
||||||
|
clipId: string,
|
||||||
|
side: "left" | "right"
|
||||||
|
) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
setResizing({
|
||||||
|
clipId,
|
||||||
|
side,
|
||||||
|
startX: e.clientX,
|
||||||
|
initialTrimStart: clip.trimStart,
|
||||||
|
initialTrimEnd: clip.trimEnd,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const updateTrimFromMouseMove = (e: { clientX: number }) => {
|
||||||
|
if (!resizing) return;
|
||||||
|
|
||||||
|
const deltaX = e.clientX - resizing.startX;
|
||||||
|
const deltaTime = deltaX / (50 * zoomLevel);
|
||||||
|
|
||||||
|
if (resizing.side === "left") {
|
||||||
|
const newTrimStart = Math.max(
|
||||||
|
0,
|
||||||
|
Math.min(
|
||||||
|
clip.duration - clip.trimEnd - 0.1,
|
||||||
|
resizing.initialTrimStart + deltaTime
|
||||||
|
)
|
||||||
|
);
|
||||||
|
updateClipTrim(track.id, clip.id, newTrimStart, clip.trimEnd);
|
||||||
|
} else {
|
||||||
|
const newTrimEnd = Math.max(
|
||||||
|
0,
|
||||||
|
Math.min(
|
||||||
|
clip.duration - clip.trimStart - 0.1,
|
||||||
|
resizing.initialTrimEnd - deltaTime
|
||||||
|
)
|
||||||
|
);
|
||||||
|
updateClipTrim(track.id, clip.id, clip.trimStart, newTrimEnd);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleResizeMove = (e: React.MouseEvent) => {
|
||||||
|
updateTrimFromMouseMove(e);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleResizeEnd = () => {
|
||||||
|
setResizing(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteClip = () => {
|
||||||
|
removeClipFromTrack(track.id, clip.id);
|
||||||
|
setClipMenuOpen(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSplitClip = () => {
|
||||||
|
// Use current playback time as split point
|
||||||
|
const splitTime = currentTime;
|
||||||
|
// Only split if splitTime is within the clip's effective range
|
||||||
|
const effectiveStart = clip.startTime;
|
||||||
|
const effectiveEnd =
|
||||||
|
clip.startTime + (clip.duration - clip.trimStart - clip.trimEnd);
|
||||||
|
|
||||||
|
if (splitTime <= effectiveStart || splitTime >= effectiveEnd) {
|
||||||
|
toast.error("Playhead must be within clip to split");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const firstDuration = splitTime - effectiveStart;
|
||||||
|
const secondDuration = effectiveEnd - splitTime;
|
||||||
|
|
||||||
|
// First part: adjust original clip
|
||||||
|
updateClipTrim(
|
||||||
|
track.id,
|
||||||
|
clip.id,
|
||||||
|
clip.trimStart,
|
||||||
|
clip.trimEnd + secondDuration
|
||||||
|
);
|
||||||
|
|
||||||
|
// Second part: add new clip after split
|
||||||
|
addClipToTrack(track.id, {
|
||||||
|
mediaId: clip.mediaId,
|
||||||
|
name: clip.name + " (cut)",
|
||||||
|
duration: clip.duration,
|
||||||
|
startTime: splitTime,
|
||||||
|
trimStart: clip.trimStart + firstDuration,
|
||||||
|
trimEnd: clip.trimEnd,
|
||||||
|
});
|
||||||
|
|
||||||
|
setClipMenuOpen(false);
|
||||||
|
toast.success("Clip split successfully");
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderClipContent = () => {
|
||||||
|
const mediaItem = mediaItems.find((item) => item.id === clip.mediaId);
|
||||||
|
|
||||||
|
if (!mediaItem) {
|
||||||
|
return (
|
||||||
|
<span className="text-xs text-foreground/80 truncate">{clip.name}</span>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mediaItem.type === "image") {
|
||||||
|
return (
|
||||||
|
<div className="w-full h-full flex items-center justify-center">
|
||||||
|
<img
|
||||||
|
src={mediaItem.url}
|
||||||
|
alt={mediaItem.name}
|
||||||
|
className="w-full h-full object-cover"
|
||||||
|
draggable={false}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mediaItem.type === "video" && mediaItem.thumbnailUrl) {
|
||||||
|
return (
|
||||||
|
<div className="w-full h-full flex items-center gap-2">
|
||||||
|
<div className="w-8 h-8 flex-shrink-0">
|
||||||
|
<img
|
||||||
|
src={mediaItem.thumbnailUrl}
|
||||||
|
alt={mediaItem.name}
|
||||||
|
className="w-full h-full object-cover rounded-sm"
|
||||||
|
draggable={false}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="text-xs text-foreground/80 truncate flex-1">
|
||||||
|
{clip.name}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mediaItem.type === "audio") {
|
||||||
|
return (
|
||||||
|
<div className="w-full h-full flex items-center gap-2">
|
||||||
|
<div className="flex-1 min-w-0">
|
||||||
|
<AudioWaveform
|
||||||
|
audioUrl={mediaItem.url}
|
||||||
|
height={24}
|
||||||
|
className="w-full"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback for videos without thumbnails
|
||||||
|
return (
|
||||||
|
<span className="text-xs text-foreground/80 truncate">{clip.name}</span>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={`timeline-clip absolute h-full border ${getTrackColor(track.type)} flex items-center py-3 min-w-[80px] overflow-hidden group hover:shadow-lg ${isSelected ? "ring-2 ring-blue-500 z-10" : ""} ${isBeingDragged ? "shadow-lg z-20" : ""} ${isBeingDragged ? "cursor-grabbing" : "cursor-grab"}`}
|
||||||
|
style={{ width: `${clipWidth}px`, left: `${clipLeft}px` }}
|
||||||
|
onMouseDown={(e) => onClipMouseDown(e, clip)}
|
||||||
|
onClick={(e) => onClipClick(e, clip)}
|
||||||
|
onMouseMove={handleResizeMove}
|
||||||
|
onMouseUp={handleResizeEnd}
|
||||||
|
onMouseLeave={handleResizeEnd}
|
||||||
|
tabIndex={0}
|
||||||
|
onContextMenu={(e) => onContextMenu(e, clip.id)}
|
||||||
|
>
|
||||||
|
{/* Left trim handle */}
|
||||||
|
<div
|
||||||
|
className={`absolute left-0 top-0 bottom-0 w-2 cursor-w-resize transition-opacity bg-blue-500/50 hover:bg-blue-500 ${isSelected ? "opacity-100" : "opacity-0"}`}
|
||||||
|
onMouseDown={(e) => handleResizeStart(e, clip.id, "left")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{/* Clip content */}
|
||||||
|
<div className="flex-1 relative">
|
||||||
|
{renderClipContent()}
|
||||||
|
|
||||||
|
{/* Clip options menu */}
|
||||||
|
<div className="absolute top-1 right-1 z-10">
|
||||||
|
<Button
|
||||||
|
variant="text"
|
||||||
|
size="icon"
|
||||||
|
className="opacity-0 group-hover:opacity-100 transition-opacity"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
setClipMenuOpen(!clipMenuOpen);
|
||||||
|
}}
|
||||||
|
onMouseDown={(e) => e.stopPropagation()}
|
||||||
|
>
|
||||||
|
<MoreVertical className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
{clipMenuOpen && (
|
||||||
|
<div
|
||||||
|
className="absolute right-0 mt-2 w-32 bg-white border rounded shadow z-50"
|
||||||
|
onMouseDown={(e) => e.stopPropagation()}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
className="flex items-center w-full px-3 py-2 text-sm hover:bg-muted/30"
|
||||||
|
onClick={handleSplitClip}
|
||||||
|
>
|
||||||
|
<Scissors className="h-4 w-4 mr-2" /> Split
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
className="flex items-center w-full px-3 py-2 text-sm text-red-600 hover:bg-red-50"
|
||||||
|
onClick={handleDeleteClip}
|
||||||
|
>
|
||||||
|
<Trash2 className="h-4 w-4 mr-2" /> Delete
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Right trim handle */}
|
||||||
|
<div
|
||||||
|
className={`absolute right-0 top-0 bottom-0 w-2 cursor-e-resize transition-opacity bg-blue-500/50 hover:bg-blue-500 ${isSelected ? "opacity-100" : "opacity-0"}`}
|
||||||
|
onMouseDown={(e) => handleResizeStart(e, clip.id, "right")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
@ -10,7 +10,6 @@ import {
|
|||||||
Snowflake,
|
Snowflake,
|
||||||
Copy,
|
Copy,
|
||||||
SplitSquareHorizontal,
|
SplitSquareHorizontal,
|
||||||
MoreVertical,
|
|
||||||
Volume2,
|
Volume2,
|
||||||
VolumeX,
|
VolumeX,
|
||||||
Pause,
|
Pause,
|
||||||
@ -25,7 +24,7 @@ import {
|
|||||||
import {
|
import {
|
||||||
useTimelineStore,
|
useTimelineStore,
|
||||||
type TimelineTrack,
|
type TimelineTrack,
|
||||||
type TimelineClip,
|
type TimelineClip as TypeTimelineClip,
|
||||||
} from "@/stores/timeline-store";
|
} from "@/stores/timeline-store";
|
||||||
import { useMediaStore } from "@/stores/media-store";
|
import { useMediaStore } from "@/stores/media-store";
|
||||||
import { usePlaybackStore } from "@/stores/playback-store";
|
import { usePlaybackStore } from "@/stores/playback-store";
|
||||||
@ -40,7 +39,8 @@ import {
|
|||||||
SelectTrigger,
|
SelectTrigger,
|
||||||
SelectValue,
|
SelectValue,
|
||||||
} from "../ui/select";
|
} from "../ui/select";
|
||||||
import AudioWaveform from "./audio-waveform";
|
import { TimelineClip } from "./timeline-clip";
|
||||||
|
import { ContextMenuState } from "@/types/timeline";
|
||||||
|
|
||||||
export function Timeline() {
|
export function Timeline() {
|
||||||
// Timeline shows all tracks (video, audio, effects) and their clips.
|
// Timeline shows all tracks (video, audio, effects) and their clips.
|
||||||
@ -81,13 +81,7 @@ export function Timeline() {
|
|||||||
const [isInTimeline, setIsInTimeline] = useState(false);
|
const [isInTimeline, setIsInTimeline] = useState(false);
|
||||||
|
|
||||||
// Unified context menu state
|
// Unified context menu state
|
||||||
const [contextMenu, setContextMenu] = useState<{
|
const [contextMenu, setContextMenu] = useState<ContextMenuState | null>(null);
|
||||||
type: "track" | "clip";
|
|
||||||
trackId: string;
|
|
||||||
clipId?: string;
|
|
||||||
x: number;
|
|
||||||
y: number;
|
|
||||||
} | null>(null);
|
|
||||||
|
|
||||||
// Marquee selection state
|
// Marquee selection state
|
||||||
const [marquee, setMarquee] = useState<{
|
const [marquee, setMarquee] = useState<{
|
||||||
@ -1137,127 +1131,31 @@ function TimelineTrackContent({
|
|||||||
}: {
|
}: {
|
||||||
track: TimelineTrack;
|
track: TimelineTrack;
|
||||||
zoomLevel: number;
|
zoomLevel: number;
|
||||||
setContextMenu: (
|
setContextMenu: (menu: ContextMenuState | null) => void;
|
||||||
menu: {
|
contextMenu: ContextMenuState | null;
|
||||||
type: "track" | "clip";
|
|
||||||
trackId: string;
|
|
||||||
clipId?: string;
|
|
||||||
x: number;
|
|
||||||
y: number;
|
|
||||||
} | null
|
|
||||||
) => void;
|
|
||||||
contextMenu: {
|
|
||||||
type: "track" | "clip";
|
|
||||||
trackId: string;
|
|
||||||
clipId?: string;
|
|
||||||
x: number;
|
|
||||||
y: number;
|
|
||||||
} | null;
|
|
||||||
}) {
|
}) {
|
||||||
const { mediaItems } = useMediaStore();
|
const { mediaItems } = useMediaStore();
|
||||||
const {
|
const {
|
||||||
tracks,
|
tracks,
|
||||||
moveClipToTrack,
|
moveClipToTrack,
|
||||||
updateClipTrim,
|
|
||||||
updateClipStartTime,
|
updateClipStartTime,
|
||||||
addClipToTrack,
|
addClipToTrack,
|
||||||
removeClipFromTrack,
|
|
||||||
toggleTrackMute,
|
|
||||||
selectedClips,
|
selectedClips,
|
||||||
selectClip,
|
selectClip,
|
||||||
deselectClip,
|
deselectClip,
|
||||||
} = useTimelineStore();
|
} = useTimelineStore();
|
||||||
const { currentTime } = usePlaybackStore();
|
|
||||||
|
|
||||||
// Mouse-based drag hook
|
// Mouse-based drag hook
|
||||||
const {
|
const { isDragging, startDrag, endDrag, timelineRef } =
|
||||||
isDragging,
|
useDragClip(zoomLevel);
|
||||||
draggedClipId,
|
|
||||||
startDrag,
|
|
||||||
endDrag,
|
|
||||||
getDraggedClipPosition,
|
|
||||||
isValidDropTarget,
|
|
||||||
timelineRef,
|
|
||||||
} = useDragClip(zoomLevel);
|
|
||||||
const [isDropping, setIsDropping] = useState(false);
|
const [isDropping, setIsDropping] = useState(false);
|
||||||
const [dropPosition, setDropPosition] = useState<number | null>(null);
|
const [dropPosition, setDropPosition] = useState<number | null>(null);
|
||||||
const [wouldOverlap, setWouldOverlap] = useState(false);
|
const [wouldOverlap, setWouldOverlap] = useState(false);
|
||||||
const [resizing, setResizing] = useState<{
|
|
||||||
clipId: string;
|
|
||||||
side: "left" | "right";
|
|
||||||
startX: number;
|
|
||||||
initialTrimStart: number;
|
|
||||||
initialTrimEnd: number;
|
|
||||||
} | null>(null);
|
|
||||||
const dragCounterRef = useRef(0);
|
const dragCounterRef = useRef(0);
|
||||||
const [clipMenuOpen, setClipMenuOpen] = useState<string | null>(null);
|
|
||||||
|
|
||||||
// Handle clip deletion
|
|
||||||
const handleDeleteClip = (clipId: string) => {
|
|
||||||
removeClipFromTrack(track.id, clipId);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleResizeStart = (
|
|
||||||
e: React.MouseEvent,
|
|
||||||
clipId: string,
|
|
||||||
side: "left" | "right"
|
|
||||||
) => {
|
|
||||||
e.stopPropagation();
|
|
||||||
e.preventDefault();
|
|
||||||
|
|
||||||
const clip = track.clips.find((c) => c.id === clipId);
|
|
||||||
if (!clip) return;
|
|
||||||
|
|
||||||
setResizing({
|
|
||||||
clipId,
|
|
||||||
side,
|
|
||||||
startX: e.clientX,
|
|
||||||
initialTrimStart: clip.trimStart,
|
|
||||||
initialTrimEnd: clip.trimEnd,
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const updateTrimFromMouseMove = (e: { clientX: number }) => {
|
|
||||||
if (!resizing) return;
|
|
||||||
|
|
||||||
const clip = track.clips.find((c) => c.id === resizing.clipId);
|
|
||||||
if (!clip) return;
|
|
||||||
|
|
||||||
const deltaX = e.clientX - resizing.startX;
|
|
||||||
const deltaTime = deltaX / (50 * zoomLevel);
|
|
||||||
|
|
||||||
if (resizing.side === "left") {
|
|
||||||
const newTrimStart = Math.max(
|
|
||||||
0,
|
|
||||||
Math.min(
|
|
||||||
clip.duration - clip.trimEnd - 0.1,
|
|
||||||
resizing.initialTrimStart + deltaTime
|
|
||||||
)
|
|
||||||
);
|
|
||||||
updateClipTrim(track.id, clip.id, newTrimStart, clip.trimEnd);
|
|
||||||
} else {
|
|
||||||
const newTrimEnd = Math.max(
|
|
||||||
0,
|
|
||||||
Math.min(
|
|
||||||
clip.duration - clip.trimStart - 0.1,
|
|
||||||
resizing.initialTrimEnd - deltaTime
|
|
||||||
)
|
|
||||||
);
|
|
||||||
updateClipTrim(track.id, clip.id, clip.trimStart, newTrimEnd);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleResizeMove = (e: React.MouseEvent) => {
|
|
||||||
updateTrimFromMouseMove(e);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleResizeEnd = () => {
|
|
||||||
setResizing(null);
|
|
||||||
};
|
|
||||||
|
|
||||||
const [justFinishedDrag, setJustFinishedDrag] = useState(false);
|
const [justFinishedDrag, setJustFinishedDrag] = useState(false);
|
||||||
|
|
||||||
const handleClipMouseDown = (e: React.MouseEvent, clip: TimelineClip) => {
|
const handleClipMouseDown = (e: React.MouseEvent, clip: TypeTimelineClip) => {
|
||||||
// Handle selection first
|
// Handle selection first
|
||||||
if (!justFinishedDrag) {
|
if (!justFinishedDrag) {
|
||||||
const isSelected = selectedClips.some(
|
const isSelected = selectedClips.some(
|
||||||
@ -1283,6 +1181,43 @@ function TimelineTrackContent({
|
|||||||
startDrag(e, clip.id, track.id, clip.startTime, clickOffsetTime);
|
startDrag(e, clip.id, track.id, clip.startTime, clickOffsetTime);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleClipClick = (e: React.MouseEvent, clip: TypeTimelineClip) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
|
||||||
|
// Don't handle click if we just finished dragging
|
||||||
|
if (justFinishedDrag) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close context menu if it's open
|
||||||
|
if (contextMenu) {
|
||||||
|
setContextMenu(null);
|
||||||
|
return; // Don't handle selection when closing context menu
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only handle deselection here (selection is handled in mouseDown)
|
||||||
|
const isSelected = selectedClips.some(
|
||||||
|
(c) => c.trackId === track.id && c.clipId === clip.id
|
||||||
|
);
|
||||||
|
|
||||||
|
if (isSelected && !e.metaKey && !e.ctrlKey && !e.shiftKey) {
|
||||||
|
// If clip is already selected and no modifier keys, deselect it
|
||||||
|
deselectClip(track.id, clip.id);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleClipContextMenu = (e: React.MouseEvent, clipId: string) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
setContextMenu({
|
||||||
|
type: "clip",
|
||||||
|
trackId: track.id,
|
||||||
|
clipId: clipId,
|
||||||
|
x: e.clientX,
|
||||||
|
y: e.clientY,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
// Reset drag flag when drag ends
|
// Reset drag flag when drag ends
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!isDragging && justFinishedDrag) {
|
if (!isDragging && justFinishedDrag) {
|
||||||
@ -1378,7 +1313,7 @@ function TimelineTrackContent({
|
|||||||
(t: TimelineTrack) => t.id === fromTrackId
|
(t: TimelineTrack) => t.id === fromTrackId
|
||||||
);
|
);
|
||||||
const movingClip = sourceTrack?.clips.find(
|
const movingClip = sourceTrack?.clips.find(
|
||||||
(c: TimelineClip) => c.id === clipId
|
(c: TypeTimelineClip) => c.id === clipId
|
||||||
);
|
);
|
||||||
|
|
||||||
if (movingClip) {
|
if (movingClip) {
|
||||||
@ -1504,7 +1439,7 @@ function TimelineTrackContent({
|
|||||||
(t: TimelineTrack) => t.id === fromTrackId
|
(t: TimelineTrack) => t.id === fromTrackId
|
||||||
);
|
);
|
||||||
const movingClip = sourceTrack?.clips.find(
|
const movingClip = sourceTrack?.clips.find(
|
||||||
(c: TimelineClip) => c.id === clipId
|
(c: TypeTimelineClip) => c.id === clipId
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!movingClip) {
|
if (!movingClip) {
|
||||||
@ -1622,107 +1557,6 @@ function TimelineTrackContent({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const getTrackColor = (type: string) => {
|
|
||||||
switch (type) {
|
|
||||||
case "video":
|
|
||||||
return "bg-blue-500/20 border-blue-500/30";
|
|
||||||
case "audio":
|
|
||||||
return "bg-green-500/20 border-green-500/30";
|
|
||||||
case "effects":
|
|
||||||
return "bg-purple-500/20 border-purple-500/30";
|
|
||||||
default:
|
|
||||||
return "bg-gray-500/20 border-gray-500/30";
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const renderClipContent = (clip: TimelineClip) => {
|
|
||||||
const mediaItem = mediaItems.find((item) => item.id === clip.mediaId);
|
|
||||||
|
|
||||||
if (!mediaItem) {
|
|
||||||
return (
|
|
||||||
<span className="text-xs text-foreground/80 truncate">{clip.name}</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaItem.type === "image") {
|
|
||||||
return (
|
|
||||||
<div className="w-full h-full flex items-center justify-center">
|
|
||||||
<img
|
|
||||||
src={mediaItem.url}
|
|
||||||
alt={mediaItem.name}
|
|
||||||
className="w-full h-full object-cover"
|
|
||||||
draggable={false}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaItem.type === "video" && mediaItem.thumbnailUrl) {
|
|
||||||
return (
|
|
||||||
<div className="w-full h-full flex items-center gap-2">
|
|
||||||
<div className="w-8 h-8 flex-shrink-0">
|
|
||||||
<img
|
|
||||||
src={mediaItem.thumbnailUrl}
|
|
||||||
alt={mediaItem.name}
|
|
||||||
className="w-full h-full object-cover rounded-sm"
|
|
||||||
draggable={false}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<span className="text-xs text-foreground/80 truncate flex-1">
|
|
||||||
{clip.name}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaItem.type === "audio") {
|
|
||||||
return (
|
|
||||||
<div className="w-full h-full flex items-center gap-2">
|
|
||||||
<div className="flex-1 min-w-0">
|
|
||||||
<AudioWaveform
|
|
||||||
audioUrl={mediaItem.url}
|
|
||||||
height={24}
|
|
||||||
className="w-full"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fallback for videos without thumbnails
|
|
||||||
return (
|
|
||||||
<span className="text-xs text-foreground/80 truncate">{clip.name}</span>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSplitClip = (clip: TimelineClip) => {
|
|
||||||
// Use current playback time as split point
|
|
||||||
const splitTime = currentTime;
|
|
||||||
// Only split if splitTime is within the clip's effective range
|
|
||||||
const effectiveStart = clip.startTime;
|
|
||||||
const effectiveEnd =
|
|
||||||
clip.startTime + (clip.duration - clip.trimStart - clip.trimEnd);
|
|
||||||
if (splitTime <= effectiveStart || splitTime >= effectiveEnd) return;
|
|
||||||
const firstDuration = splitTime - effectiveStart;
|
|
||||||
const secondDuration = effectiveEnd - splitTime;
|
|
||||||
// First part: adjust original clip
|
|
||||||
updateClipTrim(
|
|
||||||
track.id,
|
|
||||||
clip.id,
|
|
||||||
clip.trimStart,
|
|
||||||
clip.trimEnd + secondDuration
|
|
||||||
);
|
|
||||||
// Second part: add new clip after split
|
|
||||||
addClipToTrack(track.id, {
|
|
||||||
mediaId: clip.mediaId,
|
|
||||||
name: clip.name + " (cut)",
|
|
||||||
duration: clip.duration,
|
|
||||||
startTime: splitTime,
|
|
||||||
trimStart: clip.trimStart + firstDuration,
|
|
||||||
trimEnd: clip.trimEnd,
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className="w-full h-full hover:bg-muted/20"
|
className="w-full h-full hover:bg-muted/20"
|
||||||
@ -1742,14 +1576,11 @@ function TimelineTrackContent({
|
|||||||
onDragEnter={handleTrackDragEnter}
|
onDragEnter={handleTrackDragEnter}
|
||||||
onDragLeave={handleTrackDragLeave}
|
onDragLeave={handleTrackDragLeave}
|
||||||
onDrop={handleTrackDrop}
|
onDrop={handleTrackDrop}
|
||||||
onMouseMove={handleResizeMove}
|
|
||||||
onMouseUp={(e) => {
|
onMouseUp={(e) => {
|
||||||
handleResizeEnd();
|
|
||||||
if (isDragging) {
|
if (isDragging) {
|
||||||
endDrag(track.id);
|
endDrag(track.id);
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
onMouseLeave={handleResizeEnd}
|
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
ref={timelineRef}
|
ref={timelineRef}
|
||||||
@ -1774,122 +1605,21 @@ function TimelineTrackContent({
|
|||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
{track.clips.map((clip) => {
|
{track.clips.map((clip) => {
|
||||||
const effectiveDuration =
|
|
||||||
clip.duration - clip.trimStart - clip.trimEnd;
|
|
||||||
const clipWidth = Math.max(
|
|
||||||
80,
|
|
||||||
effectiveDuration * 50 * zoomLevel
|
|
||||||
);
|
|
||||||
|
|
||||||
// Use real-time position during drag, otherwise use stored position
|
|
||||||
const dragPosition = getDraggedClipPosition(clip.id);
|
|
||||||
const clipStartTime =
|
|
||||||
dragPosition !== null ? dragPosition : clip.startTime;
|
|
||||||
const clipLeft = clipStartTime * 50 * zoomLevel;
|
|
||||||
|
|
||||||
const isSelected = selectedClips.some(
|
const isSelected = selectedClips.some(
|
||||||
(c) => c.trackId === track.id && c.clipId === clip.id
|
(c) => c.trackId === track.id && c.clipId === clip.id
|
||||||
);
|
);
|
||||||
|
|
||||||
const isBeingDragged = draggedClipId === clip.id;
|
|
||||||
return (
|
return (
|
||||||
<div
|
<TimelineClip
|
||||||
key={clip.id}
|
key={clip.id}
|
||||||
className={`timeline-clip absolute h-full border ${getTrackColor(track.type)} flex items-center py-3 min-w-[80px] overflow-hidden group hover:shadow-lg ${isSelected ? "ring-2 ring-blue-500 z-10" : ""} ${isBeingDragged ? "shadow-lg z-20" : ""} ${isBeingDragged ? "cursor-grabbing" : "cursor-grab"}`}
|
clip={clip}
|
||||||
style={{ width: `${clipWidth}px`, left: `${clipLeft}px` }}
|
track={track}
|
||||||
onMouseDown={(e) => handleClipMouseDown(e, clip)}
|
zoomLevel={zoomLevel}
|
||||||
onClick={(e) => {
|
isSelected={isSelected}
|
||||||
e.stopPropagation();
|
onContextMenu={handleClipContextMenu}
|
||||||
|
onClipMouseDown={handleClipMouseDown}
|
||||||
// Don't handle click if we just finished dragging
|
onClipClick={handleClipClick}
|
||||||
if (justFinishedDrag) {
|
/>
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close context menu if it's open
|
|
||||||
if (contextMenu) {
|
|
||||||
setContextMenu(null);
|
|
||||||
return; // Don't handle selection when closing context menu
|
|
||||||
}
|
|
||||||
|
|
||||||
// Only handle deselection here (selection is handled in mouseDown)
|
|
||||||
const isSelected = selectedClips.some(
|
|
||||||
(c) => c.trackId === track.id && c.clipId === clip.id
|
|
||||||
);
|
|
||||||
|
|
||||||
if (isSelected && !e.metaKey && !e.ctrlKey && !e.shiftKey) {
|
|
||||||
// If clip is already selected and no modifier keys, deselect it
|
|
||||||
deselectClip(track.id, clip.id);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
tabIndex={0}
|
|
||||||
onContextMenu={(e) => {
|
|
||||||
e.preventDefault();
|
|
||||||
e.stopPropagation();
|
|
||||||
setContextMenu({
|
|
||||||
type: "clip",
|
|
||||||
trackId: track.id,
|
|
||||||
clipId: clip.id,
|
|
||||||
x: e.clientX,
|
|
||||||
y: e.clientY,
|
|
||||||
});
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{/* Left trim handle */}
|
|
||||||
<div
|
|
||||||
className={`absolute left-0 top-0 bottom-0 w-2 cursor-w-resize transition-opacity bg-blue-500/50 hover:bg-blue-500 ${isSelected ? "opacity-100" : "opacity-0"}`}
|
|
||||||
onMouseDown={(e) => {
|
|
||||||
e.stopPropagation(); // Prevent triggering clip drag
|
|
||||||
handleResizeStart(e, clip.id, "left");
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
{/* Clip content */}
|
|
||||||
<div className="flex-1 relative">
|
|
||||||
{renderClipContent(clip)}
|
|
||||||
{/* Clip options menu */}
|
|
||||||
<div className="absolute top-1 right-1 z-10">
|
|
||||||
<Button
|
|
||||||
variant="text"
|
|
||||||
size="icon"
|
|
||||||
className="opacity-0 group-hover:opacity-100 transition-opacity"
|
|
||||||
onClick={() => setClipMenuOpen(clip.id)}
|
|
||||||
onMouseDown={(e) => e.stopPropagation()}
|
|
||||||
>
|
|
||||||
<MoreVertical className="h-4 w-4" />
|
|
||||||
</Button>
|
|
||||||
{clipMenuOpen === clip.id && (
|
|
||||||
<div
|
|
||||||
className="absolute right-0 mt-2 w-32 bg-white border rounded shadow z-50"
|
|
||||||
onMouseDown={(e) => e.stopPropagation()}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
className="flex items-center w-full px-3 py-2 text-sm hover:bg-muted/30"
|
|
||||||
onClick={() => {
|
|
||||||
handleSplitClip(clip);
|
|
||||||
setClipMenuOpen(null);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Scissors className="h-4 w-4 mr-2" /> Split
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
className="flex items-center w-full px-3 py-2 text-sm text-red-600 hover:bg-red-50"
|
|
||||||
onClick={() => handleDeleteClip(clip.id)}
|
|
||||||
>
|
|
||||||
<Trash2 className="h-4 w-4 mr-2" /> Delete
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{/* Right trim handle */}
|
|
||||||
<div
|
|
||||||
className={`absolute right-0 top-0 bottom-0 w-2 cursor-e-resize transition-opacity bg-blue-500/50 hover:bg-blue-500 ${isSelected ? "opacity-100" : "opacity-0"}`}
|
|
||||||
onMouseDown={(e) => {
|
|
||||||
e.stopPropagation(); // Prevent triggering clip drag
|
|
||||||
handleResizeStart(e, clip.id, "right");
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
</>
|
</>
|
||||||
|
27
apps/web/src/types/timeline.ts
Normal file
27
apps/web/src/types/timeline.ts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
import { TimelineTrack, TimelineClip } from "@/stores/timeline-store";
|
||||||
|
|
||||||
|
export interface TimelineClipProps {
|
||||||
|
clip: TimelineClip;
|
||||||
|
track: TimelineTrack;
|
||||||
|
zoomLevel: number;
|
||||||
|
isSelected: boolean;
|
||||||
|
onContextMenu: (e: React.MouseEvent, clipId: string) => void;
|
||||||
|
onClipMouseDown: (e: React.MouseEvent, clip: TimelineClip) => void;
|
||||||
|
onClipClick: (e: React.MouseEvent, clip: TimelineClip) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ResizeState {
|
||||||
|
clipId: string;
|
||||||
|
side: "left" | "right";
|
||||||
|
startX: number;
|
||||||
|
initialTrimStart: number;
|
||||||
|
initialTrimEnd: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ContextMenuState {
|
||||||
|
type: "track" | "clip";
|
||||||
|
trackId: string;
|
||||||
|
clipId?: string;
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
}
|
Reference in New Issue
Block a user