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>
207 lines
6.7 KiB
TypeScript
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>
|
|
);
|
|
}
|