feat: add time formatting utility and update editor and preview components to use it

This commit is contained in:
Maze Winther
2025-07-01 01:30:02 +02:00
parent 1a01871cfc
commit c37c64c1b9
3 changed files with 29 additions and 12 deletions

View File

@ -6,6 +6,7 @@ import { ChevronLeft, Download } from "lucide-react";
import { useTimelineStore } from "@/stores/timeline-store"; import { useTimelineStore } from "@/stores/timeline-store";
import { HeaderBase } from "./header-base"; import { HeaderBase } from "./header-base";
import { ProjectNameEditor } from "./editor/project-name-editor"; import { ProjectNameEditor } from "./editor/project-name-editor";
import { formatTimeCode } from "@/lib/time";
export function EditorHeader() { export function EditorHeader() {
const { getTotalDuration } = useTimelineStore(); const { getTotalDuration } = useTimelineStore();
@ -15,13 +16,6 @@ export function EditorHeader() {
console.log("Export project"); console.log("Export project");
}; };
// Format duration from seconds to MM:SS format
const formatDuration = (seconds: number): string => {
const minutes = Math.floor(seconds / 60);
const remainingSeconds = Math.floor(seconds % 60);
return `${minutes}:${remainingSeconds.toString().padStart(2, "0")}`;
};
const leftContent = ( const leftContent = (
<div className="flex items-center gap-2"> <div className="flex items-center gap-2">
<Link <Link
@ -36,7 +30,7 @@ export function EditorHeader() {
const centerContent = ( const centerContent = (
<div className="flex items-center gap-2 text-xs text-muted-foreground"> <div className="flex items-center gap-2 text-xs text-muted-foreground">
<span>{formatDuration(getTotalDuration())}</span> <span>{formatTimeCode(getTotalDuration(), "HH:MM:SS:CS")}</span>
</div> </div>
); );

View File

@ -24,6 +24,7 @@ import {
import { Play, Pause, Volume2, VolumeX, Plus, Square } from "lucide-react"; import { Play, Pause, Volume2, VolumeX, Plus, Square } from "lucide-react";
import { useState, useRef, useEffect } from "react"; import { useState, useRef, useEffect } from "react";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import { formatTimeCode } from "@/lib/time";
interface ActiveClip { interface ActiveClip {
clip: TimelineClip; clip: TimelineClip;
@ -236,7 +237,7 @@ export function PreviewPanel() {
} }
function PreviewToolbar({ hasAnyClips }: { hasAnyClips: boolean }) { function PreviewToolbar({ hasAnyClips }: { hasAnyClips: boolean }) {
const { isPlaying, toggle } = usePlaybackStore(); const { isPlaying, toggle, currentTime } = usePlaybackStore();
const { const {
canvasSize, canvasSize,
canvasPresets, canvasPresets,
@ -244,7 +245,7 @@ function PreviewToolbar({ hasAnyClips }: { hasAnyClips: boolean }) {
setCanvasSizeFromAspectRatio, setCanvasSizeFromAspectRatio,
} = useEditorStore(); } = useEditorStore();
const { mediaItems } = useMediaStore(); const { mediaItems } = useMediaStore();
const { tracks } = useTimelineStore(); const { tracks, getTotalDuration } = useTimelineStore();
// Find the current preset based on canvas size // Find the current preset based on canvas size
const currentPreset = canvasPresets.find( const currentPreset = canvasPresets.find(
@ -289,11 +290,12 @@ function PreviewToolbar({ hasAnyClips }: { hasAnyClips: boolean }) {
<div> <div>
<p <p
className={cn( className={cn(
"text-xs text-muted-foreground", "text-xs text-muted-foreground tabular-nums",
!hasAnyClips && "opacity-50" !hasAnyClips && "opacity-50"
)} )}
> >
00:00:00:00/00:00:00:00 {formatTimeCode(currentTime, "HH:MM:SS:CS")}/
{formatTimeCode(getTotalDuration(), "HH:MM:SS:CS")}
</p> </p>
</div> </div>
<Button <Button

21
apps/web/src/lib/time.ts Normal file
View File

@ -0,0 +1,21 @@
// Time-related utility functions
// Helper function to format time in various formats (MM:SS, HH:MM:SS, HH:MM:SS:CS)
export const formatTimeCode = (
timeInSeconds: number,
format: "MM:SS" | "HH:MM:SS" | "HH:MM:SS:CS" = "HH:MM:SS:CS"
): string => {
const hours = Math.floor(timeInSeconds / 3600);
const minutes = Math.floor((timeInSeconds % 3600) / 60);
const seconds = Math.floor(timeInSeconds % 60);
const centiseconds = Math.floor((timeInSeconds % 1) * 100);
switch (format) {
case "MM:SS":
return `${minutes.toString().padStart(2, "0")}:${seconds.toString().padStart(2, "0")}`;
case "HH:MM:SS":
return `${hours.toString().padStart(2, "0")}:${minutes.toString().padStart(2, "0")}:${seconds.toString().padStart(2, "0")}`;
case "HH:MM:SS:CS":
return `${hours.toString().padStart(2, "0")}:${minutes.toString().padStart(2, "0")}:${seconds.toString().padStart(2, "0")}:${centiseconds.toString().padStart(2, "0")}`;
}
};