videofolxtv/client/src/components/video-edit-modal.tsx
sebastjanartic cce4d2456f Add ability to capture custom thumbnails from video frames
Adds a thumbnail generator component allowing users to capture frames from a video or upload their own custom images to serve as thumbnails.

Replit-Commit-Author: Agent
Replit-Commit-Session-Id: 11420304-80a9-4ef2-adff-cbdaa418ffa8
Replit-Commit-Checkpoint-Type: full_checkpoint
Replit-Commit-Screenshot-Url: https://storage.googleapis.com/screenshot-production-us-central1/8cc42625-c1f5-4e43-99bd-77f2c4dedee2/11420304-80a9-4ef2-adff-cbdaa418ffa8/yWSdtYr
2025-08-07 08:43:30 +00:00

284 lines
10 KiB
TypeScript

import { useState } from "react";
import { X, Upload, Save, Camera } from "lucide-react";
import { type Video, type UpdateVideo, updateVideoSchema } from "@shared/schema";
import { Button } from "@/components/ui/button";
import { Input } from "@/components/ui/input";
import { Textarea } from "@/components/ui/textarea";
import { Label } from "@/components/ui/label";
import { apiRequest, queryClient } from "@/lib/queryClient";
import { useToast } from "@/hooks/use-toast";
import { useMutation } from "@tanstack/react-query";
import ThumbnailGenerator from "./thumbnail-generator";
interface VideoEditModalProps {
video: Video;
isOpen: boolean;
onClose: () => void;
}
export default function VideoEditModal({ video, isOpen, onClose }: VideoEditModalProps) {
const [title, setTitle] = useState(video.title);
const [description, setDescription] = useState(video.description || "");
const [category, setCategory] = useState(video.category || "");
const [tags, setTags] = useState((video.tags || []).join(", "));
const [isPublic, setIsPublic] = useState(video.isPublic);
const [customThumbnail, setCustomThumbnail] = useState<File | null>(null);
const [thumbnailPreview, setThumbnailPreview] = useState<string | null>(video.customThumbnailUrl);
const [showThumbnailGenerator, setShowThumbnailGenerator] = useState(false);
const { toast } = useToast();
const updateVideoMutation = useMutation({
mutationFn: async (updates: UpdateVideo) => {
return apiRequest(`/api/videos/${video.id}`, {
method: "PATCH",
body: updates
});
},
onSuccess: () => {
toast({ title: "Video uspešno posodobljen!" });
queryClient.invalidateQueries({ queryKey: ["/api/videos"] });
onClose();
},
onError: () => {
toast({
title: "Napaka pri posodabljanju",
description: "Video ni bil posodobljen",
variant: "destructive"
});
}
});
const handleThumbnailChange = (e: React.ChangeEvent<HTMLInputElement>) => {
const file = e.target.files?.[0];
if (file) {
setCustomThumbnail(file);
const reader = new FileReader();
reader.onload = () => setThumbnailPreview(reader.result as string);
reader.readAsDataURL(file);
}
};
const handleThumbnailFromVideo = (thumbnailDataUrl: string) => {
setThumbnailPreview(thumbnailDataUrl);
setCustomThumbnail(null); // Clear file input since we're using generated thumbnail
setShowThumbnailGenerator(false);
toast({ title: "Thumbnail ustvarjen iz videoposnetka!" });
};
const handleSave = () => {
try {
const tagsArray = tags.split(",").map(tag => tag.trim()).filter(tag => tag.length > 0);
const updates: UpdateVideo = {
title: title.trim(),
description: description.trim(),
category: category.trim(),
tags: tagsArray,
isPublic
};
// If there's a custom thumbnail, we would need to upload it first
// For now, we'll just save the other metadata
if (customThumbnail) {
// TODO: Implement thumbnail upload to Bunny.net or storage service
toast({
title: "Obvestilo",
description: "Nalaganje slik bo dodano v prihodnji različici"
});
}
updateVideoMutation.mutate(updates);
} catch (error) {
toast({
title: "Napaka",
description: "Preverite vnesene podatke",
variant: "destructive"
});
}
};
if (!isOpen) return null;
return (
<div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4">
<div className="bg-white dark:bg-gray-900 rounded-lg max-w-2xl w-full max-h-[90vh] overflow-y-auto">
<div className="p-6">
{/* Header */}
<div className="flex items-center justify-between mb-6">
<h2 className="text-2xl font-bold text-gray-900 dark:text-white">
Uredi video
</h2>
<Button
onClick={onClose}
variant="ghost"
size="sm"
className="text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-200"
data-testid="button-close-edit"
>
<X className="w-5 h-5" />
</Button>
</div>
{/* Form */}
<div className="space-y-4">
{/* Title */}
<div>
<Label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
Naslov
</Label>
<Input
value={title}
onChange={(e) => setTitle(e.target.value)}
placeholder="Vnesite naslov videoposnetka"
className="w-full"
data-testid="input-video-title"
/>
</div>
{/* Description */}
<div>
<Label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
Opis
</Label>
<Textarea
value={description}
onChange={(e) => setDescription(e.target.value)}
placeholder="Opišite vsebino videoposnetka"
rows={4}
className="w-full"
data-testid="input-video-description"
/>
</div>
{/* Category */}
<div>
<Label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
Kategorija
</Label>
<Input
value={category}
onChange={(e) => setCategory(e.target.value)}
placeholder="npr. Izobraževanje, Zabava, Tehnologija"
className="w-full"
data-testid="input-video-category"
/>
</div>
{/* Tags */}
<div>
<Label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
Oznake (ločene z vejico)
</Label>
<Input
value={tags}
onChange={(e) => setTags(e.target.value)}
placeholder="npr. tutorial, spletno programiranje, react"
className="w-full"
data-testid="input-video-tags"
/>
</div>
{/* Thumbnail Upload */}
<div>
<Label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
Slika predogleda
</Label>
<div className="space-y-3">
{thumbnailPreview && (
<div className="w-48 h-27 rounded overflow-hidden border">
<img
src={thumbnailPreview}
alt="Predogled thumbnail"
className="w-full h-full object-cover"
/>
</div>
)}
<div className="flex gap-2">
<Button
type="button"
onClick={() => setShowThumbnailGenerator(true)}
variant="outline"
className="flex-1"
data-testid="button-generate-from-video"
>
<Camera className="w-4 h-4 mr-2" />
Generiraj iz videoposnetka
</Button>
<Label className="cursor-pointer flex-1">
<input
type="file"
accept="image/*"
onChange={handleThumbnailChange}
className="hidden"
data-testid="input-thumbnail-upload"
/>
<div className="flex items-center justify-center gap-2 px-4 py-2 bg-gray-100 dark:bg-gray-700 rounded-lg hover:bg-gray-200 dark:hover:bg-gray-600 transition-colors border text-sm font-medium h-10">
<Upload className="w-4 h-4" />
<span>Naloži sliko</span>
</div>
</Label>
</div>
</div>
</div>
{/* Privacy */}
<div className="flex items-center gap-3">
<input
type="checkbox"
id="public-video"
checked={isPublic}
onChange={(e) => setIsPublic(e.target.checked)}
className="w-4 h-4 text-blue-600 bg-gray-100 border-gray-300 rounded focus:ring-blue-500"
data-testid="checkbox-public-video"
/>
<Label htmlFor="public-video" className="text-sm font-medium text-gray-700 dark:text-gray-300">
Javno dostopen video
</Label>
</div>
</div>
{/* Footer */}
<div className="flex justify-end gap-3 mt-8 pt-6 border-t border-gray-200 dark:border-gray-700">
<Button
onClick={onClose}
variant="outline"
data-testid="button-cancel-edit"
>
Prekliči
</Button>
<Button
onClick={handleSave}
disabled={updateVideoMutation.isPending || !title.trim()}
className="bg-blue-600 hover:bg-blue-700"
data-testid="button-save-video"
>
{updateVideoMutation.isPending ? (
<div className="flex items-center gap-2">
<div className="w-4 h-4 border-2 border-white border-t-transparent rounded-full animate-spin" />
Shranjujem...
</div>
) : (
<div className="flex items-center gap-2">
<Save className="w-4 h-4" />
Shrani
</div>
)}
</Button>
</div>
</div>
</div>
{/* Thumbnail Generator Modal */}
<ThumbnailGenerator
videoUrl={video.videoUrl}
videoTitle={video.title}
isOpen={showThumbnailGenerator}
onClose={() => setShowThumbnailGenerator(false)}
onThumbnailSelect={handleThumbnailFromVideo}
/>
</div>
);
}