Files
pn-new-crm/src/components/files/file-upload-zone.tsx
Matt 1bdc856589 feat(documents-hub): NewDocumentMenu dropdown + FolderDropZone drag-drop
Replaces the bare "+ New document" Button on the documents hub with a
NewDocumentMenu dropdown so reps explicitly pick between:
- "Upload file" → opens a Dialog with FileUploadZone scoped to the
  current folder + entity context. No signing flow attached.
- "Generate document for signing" → navigates to /documents/new wizard.

Avoids the prior ambiguity where reps clicked "+ New document" intending
to attach a file and were dropped into the Documenso signer wizard.

Also adds FolderDropZone wrapping FlatFolderListing and EntityFolderView.
Dragging files from the OS over the current folder shows a drop overlay;
drop fires N parallel uploads carrying the folder + entity context.
Mirrors the per-entity Files tab UX but works in-place on the hub.

Both surfaces hit /api/v1/files/upload with folderId + entityType/Id +
the legacy clientId/companyId/yachtId FKs so files land on the right
entity AND inside the correct folder.

Also includes the in-flight prettier reformat from lint-staged on a
few previously-touched files (create-document-wizard, file-upload-zone,
admin/documenso/page) and adds the standalone prod-readiness audit
report to docs/superpowers/audits/ for permanent reference.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-11 17:59:34 +02:00

207 lines
6.7 KiB
TypeScript

'use client';
import { useCallback, useRef, useState } from 'react';
import { Upload, X } from 'lucide-react';
import { cn } from '@/lib/utils';
interface UploadingFile {
id: string;
name: string;
progress: number;
error?: string;
}
interface FileUploadZoneProps {
entityType?: string;
entityId?: string;
clientId?: string;
yachtId?: string;
companyId?: string;
/**
* Optional folder to deposit the file into. Hub uploads pass the
* currently-selected folderId so files land where the user expects.
*/
folderId?: string | null;
/**
* Fires per successful upload with the file metadata. The wizard /
* inline-upload flows use the returned id to wire follow-up actions
* (e.g. set as the source PDF for a Documenso signing flow).
*/
onUploadComplete?: (file?: { id: string; filename?: string }) => void;
}
export function FileUploadZone({
entityType,
entityId,
clientId,
yachtId,
companyId,
folderId,
onUploadComplete,
}: FileUploadZoneProps) {
const [isDragOver, setIsDragOver] = useState(false);
const [uploading, setUploading] = useState<UploadingFile[]>([]);
const inputRef = useRef<HTMLInputElement>(null);
const uploadFiles = useCallback(
async (fileList: FileList) => {
const newUploads: UploadingFile[] = Array.from(fileList).map((f) => ({
id: crypto.randomUUID(),
name: f.name,
progress: 0,
}));
setUploading((prev) => [...prev, ...newUploads]);
await Promise.all(
Array.from(fileList).map(async (file, i) => {
const uploadId = newUploads[i]!.id;
try {
const formData = new FormData();
formData.append('file', file);
formData.append('filename', file.name);
if (clientId) formData.append('clientId', clientId);
if (yachtId) formData.append('yachtId', yachtId);
if (companyId) formData.append('companyId', companyId);
if (entityType) formData.append('entityType', entityType);
if (entityId) formData.append('entityId', entityId);
if (folderId) formData.append('folderId', folderId);
setUploading((prev) =>
prev.map((u) => (u.id === uploadId ? { ...u, progress: 50 } : u)),
);
// Use fetch directly for FormData (apiFetch JSON-encodes body)
const portId = (await import('@/stores/ui-store')).useUIStore.getState().currentPortId;
const headers = new Headers();
if (portId) headers.set('X-Port-Id', portId);
const uploadRes = await fetch('/api/v1/files/upload', {
method: 'POST',
headers,
credentials: 'include',
body: formData,
});
if (!uploadRes.ok) {
throw new Error('Upload failed');
}
const uploadJson = (await uploadRes.json().catch(() => null)) as {
data?: { id?: string; filename?: string };
} | null;
if (uploadJson?.data?.id) {
onUploadComplete?.({
id: uploadJson.data.id,
filename: uploadJson.data.filename,
});
}
setUploading((prev) =>
prev.map((u) => (u.id === uploadId ? { ...u, progress: 100 } : u)),
);
} catch {
setUploading((prev) =>
prev.map((u) => (u.id === uploadId ? { ...u, error: 'Upload failed' } : u)),
);
}
}),
);
// Clear completed uploads after a moment
setTimeout(() => {
setUploading((prev) => prev.filter((u) => u.error));
onUploadComplete?.();
}, 1500);
},
[clientId, yachtId, companyId, entityType, entityId, folderId, onUploadComplete],
);
const handleDrop = useCallback(
(e: React.DragEvent<HTMLDivElement>) => {
e.preventDefault();
setIsDragOver(false);
if (e.dataTransfer.files.length > 0) {
void uploadFiles(e.dataTransfer.files);
}
},
[uploadFiles],
);
const handleChange = useCallback(
(e: React.ChangeEvent<HTMLInputElement>) => {
if (e.target.files && e.target.files.length > 0) {
void uploadFiles(e.target.files);
e.target.value = '';
}
},
[uploadFiles],
);
return (
<div className="space-y-3">
<div
role="button"
tabIndex={0}
className={cn(
'flex flex-col items-center justify-center rounded-lg border-2 border-dashed p-8 transition-colors cursor-pointer',
isDragOver
? 'border-primary bg-primary/5'
: 'border-muted-foreground/25 hover:border-primary/50 hover:bg-muted/30',
)}
onDragOver={(e) => {
e.preventDefault();
setIsDragOver(true);
}}
onDragLeave={() => setIsDragOver(false)}
onDrop={handleDrop}
onClick={() => inputRef.current?.click()}
onKeyDown={(e) => {
if (e.key === 'Enter' || e.key === ' ') inputRef.current?.click();
}}
>
<Upload className="h-8 w-8 text-muted-foreground mb-2" />
<p className="text-sm font-medium">Drop files here or click to upload</p>
<p className="text-xs text-muted-foreground mt-1">PDF, Word, Excel, images up to 50MB</p>
<input
ref={inputRef}
type="file"
multiple
className="hidden"
onChange={handleChange}
accept=".jpg,.jpeg,.png,.gif,.webp,.pdf,.doc,.docx,.xls,.xlsx,.txt,.csv"
/>
</div>
{uploading.length > 0 && (
<div className="space-y-2">
{uploading.map((u) => (
<div key={u.id} className="flex items-center gap-3 text-sm">
<span className="flex-1 truncate">{u.name}</span>
{u.error ? (
<span className="text-destructive text-xs">{u.error}</span>
) : (
<div className="flex items-center gap-2">
<div className="h-1.5 w-24 rounded-full bg-muted overflow-hidden">
<div
className="h-full rounded-full bg-primary transition-all"
style={{ width: `${u.progress}%` }}
/>
</div>
<span className="text-xs text-muted-foreground">{u.progress}%</span>
</div>
)}
{u.error && (
<button
type="button"
onClick={() => setUploading((prev) => prev.filter((x) => x.id !== u.id))}
>
<X className="h-3.5 w-3.5 text-muted-foreground" />
</button>
)}
</div>
))}
</div>
)}
</div>
);
}