Files
claudemesh/apps/web/src/modules/common/destructive-action-dialog.tsx
Alejandro Gutiérrez d3163a5bff feat(db): mesh data model — meshes, members, invites, audit log
- pgSchema "mesh" with 4 tables isolating the peer mesh domain
- Enums: visibility, transport, tier, role
- audit_log is metadata-only (E2E encryption enforced at broker/client)
- Cascade on mesh delete, soft-delete via archivedAt/revokedAt

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-04 21:19:32 +01:00

143 lines
4.3 KiB
TypeScript

/* eslint-disable i18next/no-literal-string */
"use client";
import {
Accordion,
AccordionContent,
AccordionItem,
AccordionTrigger,
} from "@turbostarter/ui-web/accordion";
import {
AlertDialog,
AlertDialogAction,
AlertDialogCancel,
AlertDialogContent,
AlertDialogDescription,
AlertDialogFooter,
AlertDialogHeader,
AlertDialogTitle,
} from "@turbostarter/ui-web/alert-dialog";
import { Icons } from "@turbostarter/ui-web/icons";
interface ImpactItem {
id: string;
displayName: string;
}
interface CascadeImpact {
label: string;
count: number;
items: ImpactItem[];
hasMore: boolean;
}
interface DestructiveActionDialogProps {
open: boolean;
onOpenChange: (open: boolean) => void;
title: string;
description: string;
impacts: CascadeImpact[];
isLoading?: boolean;
isExecuting?: boolean;
onConfirm: () => void;
onCancel: () => void;
confirmLabel?: string;
}
export function DestructiveActionDialog({
open,
onOpenChange,
title,
description,
impacts,
isLoading = false,
isExecuting = false,
onConfirm,
onCancel,
confirmLabel = "Delete",
}: DestructiveActionDialogProps) {
const hasImpacts = impacts.length > 0;
return (
<AlertDialog open={open} onOpenChange={onOpenChange}>
<AlertDialogContent>
<AlertDialogHeader>
<AlertDialogTitle>{title}</AlertDialogTitle>
<AlertDialogDescription>{description}</AlertDialogDescription>
</AlertDialogHeader>
{isLoading ? (
<div className="flex items-center gap-2 text-sm text-muted-foreground">
<Icons.Loader2 className="h-4 w-4 animate-spin" />
Checking impact...
</div>
) : hasImpacts ? (
<div className="rounded-md border border-destructive/20 bg-destructive/5 p-3">
<div className="flex items-start gap-2">
<Icons.AlertTriangle className="mt-0.5 h-4 w-4 shrink-0 text-destructive" />
<div className="flex-1 text-sm">
<p className="font-medium text-destructive">
This will also delete:
</p>
<Accordion type="multiple" className="mt-2">
{impacts.map((impact) => (
<AccordionItem
key={impact.label}
value={impact.label}
className="border-b-0"
>
<AccordionTrigger className="py-1.5 text-sm text-muted-foreground hover:no-underline">
<span>
{impact.count} {impact.label}
</span>
</AccordionTrigger>
<AccordionContent className="pb-2 pt-0">
<ul className="space-y-0.5 text-xs text-muted-foreground">
{impact.items.map((item) => (
<li
key={item.id}
className="truncate pl-2"
title={item.displayName}
>
{item.displayName}
</li>
))}
{impact.hasMore && (
<li className="pl-2 italic">
+{impact.count - impact.items.length} more...
</li>
)}
</ul>
</AccordionContent>
</AccordionItem>
))}
</Accordion>
</div>
</div>
</div>
) : null}
<AlertDialogFooter>
<AlertDialogCancel onClick={onCancel} disabled={isExecuting}>
Cancel
</AlertDialogCancel>
<AlertDialogAction
onClick={onConfirm}
disabled={isLoading || isExecuting}
className="bg-destructive text-destructive-foreground hover:bg-destructive/90"
>
{isExecuting ? (
<>
<Icons.Loader2 className="mr-2 h-4 w-4 animate-spin" />
Deleting...
</>
) : (
confirmLabel
)}
</AlertDialogAction>
</AlertDialogFooter>
</AlertDialogContent>
</AlertDialog>
);
}