refactor: update timeline to use context menu component

This commit is contained in:
Maze Winther
2025-06-29 23:09:40 +02:00
parent ca29be23ff
commit 822323d883
2 changed files with 1812 additions and 1902 deletions

View File

@ -21,6 +21,13 @@ import {
TooltipTrigger, TooltipTrigger,
TooltipProvider, TooltipProvider,
} from "../ui/tooltip"; } from "../ui/tooltip";
import {
ContextMenu,
ContextMenuContent,
ContextMenuItem,
ContextMenuSeparator,
ContextMenuTrigger,
} from "../ui/context-menu";
import { import {
useTimelineStore, useTimelineStore,
type TimelineTrack, type TimelineTrack,
@ -40,7 +47,6 @@ import {
SelectValue, SelectValue,
} from "../ui/select"; } from "../ui/select";
import { TimelineClip } from "./timeline-clip"; 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.
@ -57,7 +63,6 @@ export function Timeline() {
selectedClips, selectedClips,
clearSelectedClips, clearSelectedClips,
setSelectedClips, setSelectedClips,
updateClipTrim,
splitClip, splitClip,
splitAndKeepLeft, splitAndKeepLeft,
splitAndKeepRight, splitAndKeepRight,
@ -84,9 +89,6 @@ export function Timeline() {
const timelineRef = useRef<HTMLDivElement>(null); const timelineRef = useRef<HTMLDivElement>(null);
const [isInTimeline, setIsInTimeline] = useState(false); const [isInTimeline, setIsInTimeline] = useState(false);
// Unified context menu state
const [contextMenu, setContextMenu] = useState<ContextMenuState | null>(null);
// Marquee selection state // Marquee selection state
const [marquee, setMarquee] = useState<{ const [marquee, setMarquee] = useState<{
startX: number; startX: number;
@ -129,15 +131,6 @@ export function Timeline() {
setDuration(Math.max(totalDuration, 10)); // Minimum 10 seconds for empty timeline setDuration(Math.max(totalDuration, 10)); // Minimum 10 seconds for empty timeline
}, [tracks, setDuration, getTotalDuration]); }, [tracks, setDuration, getTotalDuration]);
// Close context menu on click elsewhere
useEffect(() => {
const handleClick = () => setContextMenu(null);
if (contextMenu) {
window.addEventListener("click", handleClick);
return () => window.removeEventListener("click", handleClick);
}
}, [contextMenu]);
// Keyboard event for deleting selected clips // Keyboard event for deleting selected clips
useEffect(() => { useEffect(() => {
const handleKeyDown = (e: KeyboardEvent) => { const handleKeyDown = (e: KeyboardEvent) => {
@ -562,7 +555,8 @@ export function Timeline() {
const clip = track?.clips.find((c) => c.id === clipId); const clip = track?.clips.find((c) => c.id === clipId);
if (clip && track) { if (clip && track) {
const effectiveStart = clip.startTime; const effectiveStart = clip.startTime;
const effectiveEnd = clip.startTime + (clip.duration - clip.trimStart - clip.trimEnd); const effectiveEnd =
clip.startTime + (clip.duration - clip.trimStart - clip.trimEnd);
if (currentTime > effectiveStart && currentTime < effectiveEnd) { if (currentTime > effectiveStart && currentTime < effectiveEnd) {
splitAndKeepLeft(trackId, clipId, currentTime); splitAndKeepLeft(trackId, clipId, currentTime);
@ -590,7 +584,8 @@ export function Timeline() {
const clip = track?.clips.find((c) => c.id === clipId); const clip = track?.clips.find((c) => c.id === clipId);
if (clip && track) { if (clip && track) {
const effectiveStart = clip.startTime; const effectiveStart = clip.startTime;
const effectiveEnd = clip.startTime + (clip.duration - clip.trimStart - clip.trimEnd); const effectiveEnd =
clip.startTime + (clip.duration - clip.trimStart - clip.trimEnd);
if (currentTime > effectiveStart && currentTime < effectiveEnd) { if (currentTime > effectiveStart && currentTime < effectiveEnd) {
splitAndKeepRight(trackId, clipId, currentTime); splitAndKeepRight(trackId, clipId, currentTime);
@ -618,7 +613,12 @@ export function Timeline() {
const clip = track?.clips.find((c) => c.id === clipId); const clip = track?.clips.find((c) => c.id === clipId);
const mediaItem = mediaItems.find((item) => item.id === clip?.mediaId); const mediaItem = mediaItems.find((item) => item.id === clip?.mediaId);
if (clip && track && mediaItem?.type === "video" && track.type === "video") { if (
clip &&
track &&
mediaItem?.type === "video" &&
track.type === "video"
) {
const audioClipId = separateAudio(trackId, clipId); const audioClipId = separateAudio(trackId, clipId);
if (audioClipId) separatedCount++; if (audioClipId) separatedCount++;
} }
@ -664,8 +664,12 @@ export function Timeline() {
// --- Scroll synchronization effect --- // --- Scroll synchronization effect ---
useEffect(() => { useEffect(() => {
const rulerViewport = rulerScrollRef.current?.querySelector('[data-radix-scroll-area-viewport]') as HTMLElement; const rulerViewport = rulerScrollRef.current?.querySelector(
const tracksViewport = tracksScrollRef.current?.querySelector('[data-radix-scroll-area-viewport]') as HTMLElement; "[data-radix-scroll-area-viewport]"
) as HTMLElement;
const tracksViewport = tracksScrollRef.current?.querySelector(
"[data-radix-scroll-area-viewport]"
) as HTMLElement;
if (!rulerViewport || !tracksViewport) return; if (!rulerViewport || !tracksViewport) return;
const handleRulerScroll = () => { const handleRulerScroll = () => {
const now = Date.now(); const now = Date.now();
@ -683,18 +687,22 @@ export function Timeline() {
rulerViewport.scrollLeft = tracksViewport.scrollLeft; rulerViewport.scrollLeft = tracksViewport.scrollLeft;
isUpdatingRef.current = false; isUpdatingRef.current = false;
}; };
rulerViewport.addEventListener('scroll', handleRulerScroll); rulerViewport.addEventListener("scroll", handleRulerScroll);
tracksViewport.addEventListener('scroll', handleTracksScroll); tracksViewport.addEventListener("scroll", handleTracksScroll);
return () => { return () => {
rulerViewport.removeEventListener('scroll', handleRulerScroll); rulerViewport.removeEventListener("scroll", handleRulerScroll);
tracksViewport.removeEventListener('scroll', handleTracksScroll); tracksViewport.removeEventListener("scroll", handleTracksScroll);
}; };
}, []); }, []);
// --- Playhead auto-scroll effect --- // --- Playhead auto-scroll effect ---
useEffect(() => { useEffect(() => {
const rulerViewport = rulerScrollRef.current?.querySelector('[data-radix-scroll-area-viewport]') as HTMLElement; const rulerViewport = rulerScrollRef.current?.querySelector(
const tracksViewport = tracksScrollRef.current?.querySelector('[data-radix-scroll-area-viewport]') as HTMLElement; "[data-radix-scroll-area-viewport]"
) as HTMLElement;
const tracksViewport = tracksScrollRef.current?.querySelector(
"[data-radix-scroll-area-viewport]"
) as HTMLElement;
if (!rulerViewport || !tracksViewport) return; if (!rulerViewport || !tracksViewport) return;
const playheadPx = playheadPosition * 50 * zoomLevel; const playheadPx = playheadPosition * 50 * zoomLevel;
const viewportWidth = rulerViewport.clientWidth; const viewportWidth = rulerViewport.clientWidth;
@ -798,7 +806,11 @@ export function Timeline() {
<Tooltip> <Tooltip>
<TooltipTrigger asChild> <TooltipTrigger asChild>
<Button variant="text" size="icon" onClick={handleSplitAndKeepLeft}> <Button
variant="text"
size="icon"
onClick={handleSplitAndKeepLeft}
>
<ArrowLeftToLine className="h-4 w-4" /> <ArrowLeftToLine className="h-4 w-4" />
</Button> </Button>
</TooltipTrigger> </TooltipTrigger>
@ -807,7 +819,11 @@ export function Timeline() {
<Tooltip> <Tooltip>
<TooltipTrigger asChild> <TooltipTrigger asChild>
<Button variant="text" size="icon" onClick={handleSplitAndKeepRight}> <Button
variant="text"
size="icon"
onClick={handleSplitAndKeepRight}
>
<ArrowRightToLine className="h-4 w-4" /> <ArrowRightToLine className="h-4 w-4" />
</Button> </Button>
</TooltipTrigger> </TooltipTrigger>
@ -994,15 +1010,6 @@ export function Timeline() {
<div <div
key={track.id} key={track.id}
className="h-[60px] flex items-center px-3 border-b border-muted/30 bg-background group" className="h-[60px] flex items-center px-3 border-b border-muted/30 bg-background group"
onContextMenu={(e) => {
e.preventDefault();
setContextMenu({
type: "track",
trackId: track.id,
x: e.clientX,
y: e.clientY,
});
}}
> >
<div className="flex items-center flex-1 min-w-0"> <div className="flex items-center flex-1 min-w-0">
<div <div
@ -1055,30 +1062,21 @@ export function Timeline() {
) : ( ) : (
<> <>
{tracks.map((track, index) => ( {tracks.map((track, index) => (
<ContextMenu key={track.id}>
<ContextMenuTrigger asChild>
<div <div
key={track.id}
className="absolute left-0 right-0 border-b border-muted/30" className="absolute left-0 right-0 border-b border-muted/30"
style={{ style={{
top: `${index * 60}px`, top: `${index * 60}px`,
height: "60px", height: "60px",
}} }}
// Show context menu on right click
onContextMenu={(e) => {
e.preventDefault();
setContextMenu({
type: "track",
trackId: track.id,
x: e.clientX,
y: e.clientY,
});
}}
onClick={(e) => { onClick={(e) => {
// If clicking empty area (not on a clip), deselect all clips // If clicking empty area (not on a clip), deselect all clips
if ( if (
!(e.target as HTMLElement).closest(".timeline-clip") !(e.target as HTMLElement).closest(
".timeline-clip"
)
) { ) {
const { clearSelectedClips } =
useTimelineStore.getState();
clearSelectedClips(); clearSelectedClips();
} }
}} }}
@ -1086,10 +1084,40 @@ export function Timeline() {
<TimelineTrackContent <TimelineTrackContent
track={track} track={track}
zoomLevel={zoomLevel} zoomLevel={zoomLevel}
setContextMenu={setContextMenu}
contextMenu={contextMenu}
/> />
</div> </div>
</ContextMenuTrigger>
<ContextMenuContent>
<ContextMenuItem
onClick={() => {
toggleTrackMute(track.id);
}}
>
{track.muted ? (
<>
<Volume2 className="h-4 w-4 mr-2" />
Unmute Track
</>
) : (
<>
<VolumeX className="h-4 w-4 mr-2" />
Mute Track
</>
)}
</ContextMenuItem>
<ContextMenuSeparator />
<ContextMenuItem
onClick={() => {
removeTrack(track.id);
toast.success("Track deleted");
}}
className="text-destructive focus:text-destructive"
>
<Trash2 className="h-4 w-4 mr-2" />
Delete Track
</ContextMenuItem>
</ContextMenuContent>
</ContextMenu>
))} ))}
{/* Playhead for tracks area (scrubbable) */} {/* Playhead for tracks area (scrubbable) */}
@ -1119,155 +1147,6 @@ export function Timeline() {
</div> </div>
</div> </div>
</div> </div>
{/* Clean Unified Context Menu */}
{contextMenu && (
<div
className="fixed z-50 min-w-[160px] bg-popover border border-border rounded-md shadow-md py-1 text-sm"
style={{ left: contextMenu.x, top: contextMenu.y }}
onContextMenu={(e) => e.preventDefault()}
>
{contextMenu.type === "track" ? (
// Track context menu
<>
<button
className="flex items-center w-full px-3 py-2 hover:bg-accent hover:text-accent-foreground transition-colors text-left"
onClick={() => {
const track = tracks.find(
(t) => t.id === contextMenu.trackId
);
if (track) toggleTrackMute(track.id);
setContextMenu(null);
}}
>
{contextMenu.trackId ? (
<div className="flex items-center w-full px-3 py-2 hover:bg-accent hover:text-accent-foreground transition-colors text-left">
<Volume2 className="h-4 w-4 mr-2" />
Unmute Track
</div>
) : (
<div className="flex items-center w-full px-3 py-2 hover:bg-accent hover:text-accent-foreground transition-colors text-left">
<VolumeX className="h-4 w-4 mr-2" />
Mute Track
</div>
)}
</button>
<div className="h-px bg-border mx-1 my-1" />
<button
className="flex items-center w-full px-3 py-2 text-destructive hover:bg-destructive/10 transition-colors text-left"
onClick={() => {
removeTrack(contextMenu.trackId);
setContextMenu(null);
toast.success("Track deleted");
}}
>
<Trash2 className="h-4 w-4 mr-2" />
Delete Track
</button>
</>
) : (
// Clip context menu
<>
<button
className="flex items-center w-full px-3 py-2 hover:bg-accent hover:text-accent-foreground transition-colors text-left"
onClick={() => {
if (contextMenu.clipId) {
const track = tracks.find(
(t) => t.id === contextMenu.trackId
);
const clip = track?.clips.find(
(c) => c.id === contextMenu.clipId
);
if (clip && track) {
const splitTime = currentTime;
const effectiveStart = clip.startTime;
const effectiveEnd =
clip.startTime +
(clip.duration - clip.trimStart - clip.trimEnd);
if (
splitTime > effectiveStart &&
splitTime < effectiveEnd
) {
updateClipTrim(
track.id,
clip.id,
clip.trimStart,
clip.trimEnd + (effectiveEnd - splitTime)
);
useTimelineStore.getState().addClipToTrack(track.id, {
mediaId: clip.mediaId,
name: clip.name + " (split)",
duration: clip.duration,
startTime: splitTime,
trimStart:
clip.trimStart + (splitTime - effectiveStart),
trimEnd: clip.trimEnd,
});
toast.success("Clip split successfully");
} else {
toast.error("Playhead must be within clip to split");
}
}
}
setContextMenu(null);
}}
>
<Scissors className="h-4 w-4 mr-2" />
Split at Playhead
</button>
<button
className="flex items-center w-full px-3 py-2 hover:bg-accent hover:text-accent-foreground transition-colors text-left"
onClick={() => {
if (contextMenu.clipId) {
const track = tracks.find(
(t) => t.id === contextMenu.trackId
);
const clip = track?.clips.find(
(c) => c.id === contextMenu.clipId
);
if (clip && track) {
useTimelineStore.getState().addClipToTrack(track.id, {
mediaId: clip.mediaId,
name: clip.name + " (copy)",
duration: clip.duration,
startTime:
clip.startTime +
(clip.duration - clip.trimStart - clip.trimEnd) +
0.1,
trimStart: clip.trimStart,
trimEnd: clip.trimEnd,
});
toast.success("Clip duplicated");
}
}
setContextMenu(null);
}}
>
<Copy className="h-4 w-4 mr-2" />
Duplicate Clip
</button>
<div className="h-px bg-border mx-1 my-1" />
<button
className="flex items-center w-full px-3 py-2 text-destructive hover:bg-destructive/10 transition-colors text-left"
onClick={() => {
if (contextMenu.clipId) {
removeClipFromTrack(
contextMenu.trackId,
contextMenu.clipId
);
toast.success("Clip deleted");
}
setContextMenu(null);
}}
>
<Trash2 className="h-4 w-4 mr-2" />
Delete Clip
</button>
</>
)}
</div>
)}
</div> </div>
); );
} }
@ -1275,13 +1154,9 @@ export function Timeline() {
function TimelineTrackContent({ function TimelineTrackContent({
track, track,
zoomLevel, zoomLevel,
setContextMenu,
contextMenu,
}: { }: {
track: TimelineTrack; track: TimelineTrack;
zoomLevel: number; zoomLevel: number;
setContextMenu: (menu: ContextMenuState | null) => void;
contextMenu: ContextMenuState | null;
}) { }) {
const { mediaItems } = useMediaStore(); const { mediaItems } = useMediaStore();
const { const {
@ -1430,12 +1305,6 @@ function TimelineTrackContent({
} }
} }
// Close context menu if it's open
if (contextMenu) {
setContextMenu(null);
return; // Don't handle selection when closing context menu
}
// Skip selection logic for multi-selection (handled in mousedown) // Skip selection logic for multi-selection (handled in mousedown)
if (e.metaKey || e.ctrlKey || e.shiftKey) { if (e.metaKey || e.ctrlKey || e.shiftKey) {
return; return;
@ -1455,18 +1324,6 @@ function TimelineTrackContent({
} }
}; };
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,
});
};
const handleTrackDragOver = (e: React.DragEvent) => { const handleTrackDragOver = (e: React.DragEvent) => {
e.preventDefault(); e.preventDefault();
@ -1801,18 +1658,6 @@ function TimelineTrackContent({
return ( return (
<div <div
className="w-full h-full hover:bg-muted/20" className="w-full h-full hover:bg-muted/20"
onContextMenu={(e) => {
e.preventDefault();
// Only show track menu if we didn't click on a clip
if (!(e.target as HTMLElement).closest(".timeline-clip")) {
setContextMenu({
type: "track",
trackId: track.id,
x: e.clientX,
y: e.clientY,
});
}
}}
onClick={(e) => { onClick={(e) => {
// If clicking empty area (not on a clip), deselect all clips // If clicking empty area (not on a clip), deselect all clips
if (!(e.target as HTMLElement).closest(".timeline-clip")) { if (!(e.target as HTMLElement).closest(".timeline-clip")) {
@ -1852,17 +1697,91 @@ function TimelineTrackContent({
(c) => c.trackId === track.id && c.clipId === clip.id (c) => c.trackId === track.id && c.clipId === clip.id
); );
const handleClipSplit = () => {
const { currentTime } = usePlaybackStore();
const { updateClipTrim, addClipToTrack } = useTimelineStore();
const splitTime = currentTime;
const effectiveStart = clip.startTime;
const effectiveEnd =
clip.startTime +
(clip.duration - clip.trimStart - clip.trimEnd);
if (splitTime > effectiveStart && splitTime < effectiveEnd) {
updateClipTrim(
track.id,
clip.id,
clip.trimStart,
clip.trimEnd + (effectiveEnd - splitTime)
);
addClipToTrack(track.id, {
mediaId: clip.mediaId,
name: clip.name + " (split)",
duration: clip.duration,
startTime: splitTime,
trimStart: clip.trimStart + (splitTime - effectiveStart),
trimEnd: clip.trimEnd,
});
toast.success("Clip split successfully");
} else {
toast.error("Playhead must be within clip to split");
}
};
const handleClipDuplicate = () => {
const { addClipToTrack } = useTimelineStore.getState();
addClipToTrack(track.id, {
mediaId: clip.mediaId,
name: clip.name + " (copy)",
duration: clip.duration,
startTime:
clip.startTime +
(clip.duration - clip.trimStart - clip.trimEnd) +
0.1,
trimStart: clip.trimStart,
trimEnd: clip.trimEnd,
});
toast.success("Clip duplicated");
};
const handleClipDelete = () => {
const { removeClipFromTrack } = useTimelineStore.getState();
removeClipFromTrack(track.id, clip.id);
toast.success("Clip deleted");
};
return ( return (
<ContextMenu key={clip.id}>
<ContextMenuTrigger asChild>
<div>
<TimelineClip <TimelineClip
key={clip.id}
clip={clip} clip={clip}
track={track} track={track}
zoomLevel={zoomLevel} zoomLevel={zoomLevel}
isSelected={isSelected} isSelected={isSelected}
onContextMenu={handleClipContextMenu}
onClipMouseDown={handleClipMouseDown} onClipMouseDown={handleClipMouseDown}
onClipClick={handleClipClick} onClipClick={handleClipClick}
/> />
</div>
</ContextMenuTrigger>
<ContextMenuContent>
<ContextMenuItem onClick={handleClipSplit}>
<Scissors className="h-4 w-4 mr-2" />
Split at Playhead
</ContextMenuItem>
<ContextMenuItem onClick={handleClipDuplicate}>
<Copy className="h-4 w-4 mr-2" />
Duplicate Clip
</ContextMenuItem>
<ContextMenuSeparator />
<ContextMenuItem
onClick={handleClipDelete}
className="text-destructive focus:text-destructive"
>
<Trash2 className="h-4 w-4 mr-2" />
Delete Clip
</ContextMenuItem>
</ContextMenuContent>
</ContextMenu>
); );
})} })}
</> </>

View File

@ -7,7 +7,6 @@ export interface TimelineClipProps {
track: TimelineTrack; track: TimelineTrack;
zoomLevel: number; zoomLevel: number;
isSelected: boolean; isSelected: boolean;
onContextMenu: (e: React.MouseEvent, clipId: string) => void;
onClipMouseDown: (e: React.MouseEvent, clip: TimelineClip) => void; onClipMouseDown: (e: React.MouseEvent, clip: TimelineClip) => void;
onClipClick: (e: React.MouseEvent, clip: TimelineClip) => void; onClipClick: (e: React.MouseEvent, clip: TimelineClip) => void;
} }
@ -19,11 +18,3 @@ export interface ResizeState {
initialTrimStart: number; initialTrimStart: number;
initialTrimEnd: number; initialTrimEnd: number;
} }
export interface ContextMenuState {
type: "track" | "clip";
trackId: string;
clipId?: string;
x: number;
y: number;
}