import React, { useState, useEffect } from "react"; import { useTranslation } from "../../hooks/useTranslation"; import { Modal } from "../Modal"; import { updateProject, toggleArchiveProject } from "../../api/projects"; import { getClients } from "../../api/clients"; import { useWorkspace } from "../../context/WorkspaceContext"; import { Archive, RefreshCcw } from "lucide-react"; import { Select } from "../ui/Select"; import { Input } from "../ui/input"; import { TextAreaInput } from "../ui/TextAreaInput"; import { toast } from "sonner"; import { PROJECTS_ARCHIVE, PROJECTS_EDIT, canWorkspace } from "../../lib/permissions"; interface ProjectEditModalProps { isOpen: boolean; onClose: () => void; project: any; } export const ProjectEditModal: React.FC = ({ isOpen, onClose, project }) => { const { t } = useTranslation(); const { activeWorkspace } = useWorkspace(); const canEditProject = canWorkspace(activeWorkspace?.my_role, PROJECTS_EDIT); const canArchiveProject = canWorkspace(activeWorkspace?.my_role, PROJECTS_ARCHIVE); const [loading, setLoading] = useState(false); const [clients, setClients] = useState([]); const [formData, setFormData] = useState({ name: "", description: "", color: "#3B82F6", client: "", }); const [thumbnailFile, setThumbnailFile] = useState(null); const [thumbnailUrl, setThumbnailUrl] = useState(null); const [thumbnailPreview, setThumbnailPreview] = useState(null); const [clearThumbnail, setClearThumbnail] = useState(false); const [loadingClients, setLoadingClients] = useState(false); useEffect(() => { if (isOpen && activeWorkspace) { setLoadingClients(true); getClients(activeWorkspace.id) .then((res: any) => setClients(res.results || res)) .catch((err) => toast.error(t.projects?.clientFetchError || err.message || "Failed to load clients")) .finally(() => setLoadingClients(false)); } }, [isOpen, activeWorkspace]); useEffect(() => { if (project) { setFormData({ name: project.name || "", description: project.description || "", color: project.color || "#3B82F6", client: project.client ? project.client.id : "", }); setThumbnailUrl(project.thumbnail || null); setThumbnailFile(null); setClearThumbnail(false); } }, [project]); useEffect(() => { if (!thumbnailFile) { setThumbnailPreview(null); return; } const objectUrl = URL.createObjectURL(thumbnailFile); setThumbnailPreview(objectUrl); return () => URL.revokeObjectURL(objectUrl); }, [thumbnailFile]); const handleThumbnailChange = (file: File | null) => { if (!file) return; if (!["image/jpeg", "image/png", "image/webp"].includes(file.type)) { toast.error(t.workspace?.thumbnailInvalidType || "Unsupported image type. Use JPG, PNG, or WebP."); return; } if (file.size > 2 * 1024 * 1024) { toast.error(t.workspace?.thumbnailMaxSizeError || "Image size must be 2MB or less."); return; } setThumbnailFile(file); setClearThumbnail(false); }; const handleSubmit = async (e?: React.FormEvent) => { e?.preventDefault(); if (!project || !formData.name) return; setLoading(true); try { const updated = await updateProject(project.id, { name: formData.name, description: formData.description, color: formData.color, client: formData.client || null, thumbnail: thumbnailFile, clear_thumbnail: clearThumbnail, }); toast.success(t.projects?.updateSuccess || "Project updated successfully."); window.dispatchEvent(new CustomEvent("project_updated", { detail: updated })); onClose(); } catch (error) { console.error(error); toast.error(t.projects?.updateError || "Failed to update project."); } finally { setLoading(false); } }; const handleArchiveToggle = async () => { if (!project) return; setLoading(true); try { const updated = await toggleArchiveProject(project.id); toast.success( project?.is_archived ? t.projects?.restoreSuccess || t.projects?.updateSuccess || "Project updated successfully." : t.projects?.archiveSuccess || t.projects?.updateSuccess || "Project updated successfully.", ); window.dispatchEvent(new CustomEvent("project_updated", { detail: updated })); onClose(); } catch (error) { console.error(error); toast.error(t.projects?.updateError || "Failed to update project."); } finally { setLoading(false); } }; const footer = (
{canArchiveProject ? ( ) : (
)}
); if (!canEditProject) return null; return (
setFormData({ ...formData, name: e.target.value })} className="w-full px-3 py-2 border rounded-lg dark:bg-slate-800 dark:border-slate-700 outline-none focus:ring-2 focus:ring-blue-500" />
C
setFormData({ ...formData, color: e.target.value })} className="absolute -top-2 -left-2 w-16 h-16 cursor-pointer" />
{thumbnailPreview ? ( ) : !clearThumbnail && thumbnailUrl ? ( ) : ( formData.name.trim().charAt(0).toUpperCase() || "P" )}
handleThumbnailChange(event.target.files?.[0] || null)} /> {(thumbnailFile || (!clearThumbnail && thumbnailUrl)) ? ( ) : null}
setFormData({ ...formData, description: e.target.value })} className="w-full px-3 py-2 border rounded-lg dark:text-slate-100 dark:bg-slate-800 dark:border-slate-700 outline-none focus:ring-2 focus:ring-blue-500 min-h-[80px] resize-y" />