feat: add per-patient Local folder in Documents page backed by Cloud Storage

- Documents page shows a "Local Folder" card for each selected patient
  with an "Open in Cloud Storage" button that deep-links to their folder
- Cloud Storage page reads ?folderId URL param on mount and auto-opens
  the folder panel for seamless navigation from Documents
- Backend: GET /api/cloud-storage/patient-folder/:patientId endpoint
  that idempotently gets or creates a top-level CloudFolder per patient
- CloudFolder schema gains optional patientId field linked to Patient
- Disk directories for cloud storage folders now use the folder's name
  (e.g. "Xiaohui Wang/") instead of the opaque "folder-{id}/" path

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
ff
2026-06-05 23:01:56 -04:00
parent 2457e12b5c
commit d5bc96ff39
158 changed files with 1539 additions and 130 deletions

View File

@@ -1,4 +1,5 @@
import { useState } from "react";
import { useEffect, useState } from "react";
import { useSearch } from "wouter";
import { useQueryClient } from "@tanstack/react-query";
import { Button } from "@/components/ui/button";
import { Folder as FolderIcon, Search as SearchIcon } from "lucide-react";
@@ -20,6 +21,7 @@ export default function CloudStoragePage() {
const { toast } = useToast();
const { user } = useAuth();
const qc = useQueryClient();
const search = useSearch();
// panel open + initial folder id to show when opening
const [panelOpen, setPanelOpen] = useState(false);
@@ -27,6 +29,16 @@ export default function CloudStoragePage() {
number | null
>(null);
// Deep-link: if navigated here with ?folderId=XXX, open that folder automatically
useEffect(() => {
const params = new URLSearchParams(search);
const id = Number(params.get("folderId"));
if (id > 0) {
setPanelInitialFolderId(id);
setPanelOpen(true);
}
}, [search]);
// key to remount recent card to clear its internal selection when needed
const [recentKey, setRecentKey] = useState(0);

View File

@@ -1,5 +1,6 @@
import { useEffect, useMemo, useState } from "react";
import { useQuery, useMutation } from "@tanstack/react-query";
import { useLocation } from "wouter";
import {
Card,
CardHeader,
@@ -10,7 +11,8 @@ import {
import { Button } from "@/components/ui/button";
import { toast } from "@/hooks/use-toast";
import { apiRequest, queryClient } from "@/lib/queryClient";
import { Eye, Trash, Download, FolderOpen, FileText } from "lucide-react";
import { Eye, Trash, Download, FolderOpen, FileText, HardDrive } from "lucide-react";
import { useAuth } from "@/hooks/use-auth";
import { DeleteConfirmationDialog } from "@/components/ui/deleteDialog";
import { PatientTable } from "@/components/patients/patient-table";
import { Patient, PdfFile } from "@repo/db/types";
@@ -32,6 +34,8 @@ import {
} from "@/lib/api/documents";
export default function DocumentsPage() {
const { user } = useAuth();
const [, setLocation] = useLocation();
const [selectedPatient, setSelectedPatient] = useState<Patient | null>(null);
const [expandedGroupId, setExpandedGroupId] = useState<number | null>(null);
@@ -200,6 +204,22 @@ export default function DocumentsPage() {
return { map, orderedKeys };
}, [groups]);
// FETCH patient's dedicated local (cloud storage) folder
const { data: patientCloudFolder, isLoading: isLoadingCloudFolder } = useQuery({
queryKey: ["patientCloudFolder", selectedPatient?.id, user?.id],
enabled: !!selectedPatient?.id && !!user?.id,
staleTime: 5 * 60 * 1000,
queryFn: async () => {
const res = await apiRequest(
"GET",
`/api/cloud-storage/patient-folder/${selectedPatient!.id}?userId=${user!.id}`
);
const json = await res.json();
if (json.error) throw new Error(json.message);
return json.data as { id: number; name: string };
},
});
// FETCH PDFs for selected group with pagination (limit & offset)
const {
data: groupPdfsResponse,
@@ -324,6 +344,32 @@ export default function DocumentsPage() {
<CardContent className="space-y-4">
{/* Local Folder (Cloud Storage) */}
<div className="border rounded p-3">
<div className="flex items-center justify-between">
<div className="flex items-center gap-3">
<HardDrive className="w-5 h-5 text-blue-500" />
<div>
<div className="font-semibold">
{isLoadingCloudFolder
? "Loading…"
: patientCloudFolder?.name ?? "Local Folder"}
</div>
<div className="text-xs text-muted-foreground">Local disk storage</div>
</div>
</div>
<Button
size="sm"
variant="outline"
disabled={!patientCloudFolder}
onClick={() =>
setLocation(`/cloud-storage?folderId=${patientCloudFolder!.id}`)
}
>
Open in Cloud Storage
</Button>
</div>
</div>
{/* Existing Groups Section */}
<div>