diff options
Diffstat (limited to 'apps/web/components/dashboard/lists/PendingInvitationsCard.tsx')
| -rw-r--r-- | apps/web/components/dashboard/lists/PendingInvitationsCard.tsx | 158 |
1 files changed, 158 insertions, 0 deletions
diff --git a/apps/web/components/dashboard/lists/PendingInvitationsCard.tsx b/apps/web/components/dashboard/lists/PendingInvitationsCard.tsx new file mode 100644 index 00000000..c453a91f --- /dev/null +++ b/apps/web/components/dashboard/lists/PendingInvitationsCard.tsx @@ -0,0 +1,158 @@ +"use client"; + +import { Button } from "@/components/ui/button"; +import { + Card, + CardContent, + CardDescription, + CardHeader, + CardTitle, +} from "@/components/ui/card"; +import { toast } from "@/components/ui/use-toast"; +import { useTranslation } from "@/lib/i18n/client"; +import { api } from "@/lib/trpc"; +import { Check, Loader2, Mail, X } from "lucide-react"; + +interface Invitation { + id: string; + role: string; + list: { + name: string; + icon?: string; + description?: string | null; + owner?: { + name?: string; + } | null; + }; +} + +function InvitationRow({ invitation }: { invitation: Invitation }) { + const { t } = useTranslation(); + const utils = api.useUtils(); + + const acceptInvitation = api.lists.acceptInvitation.useMutation({ + onSuccess: async () => { + toast({ + description: t("lists.invitations.accepted"), + }); + await Promise.all([ + utils.lists.getPendingInvitations.invalidate(), + utils.lists.list.invalidate(), + ]); + }, + onError: (error) => { + toast({ + variant: "destructive", + description: error.message || t("lists.invitations.failed_to_accept"), + }); + }, + }); + + const declineInvitation = api.lists.declineInvitation.useMutation({ + onSuccess: async () => { + toast({ + description: t("lists.invitations.declined"), + }); + await utils.lists.getPendingInvitations.invalidate(); + }, + onError: (error) => { + toast({ + variant: "destructive", + description: error.message || t("lists.invitations.failed_to_decline"), + }); + }, + }); + + return ( + <div className="flex items-center justify-between rounded-lg border p-4"> + <div className="flex-1"> + <div className="flex items-center gap-2"> + <span className="font-medium">{invitation.list.name}</span> + <span className="text-xs text-muted-foreground"> + {invitation.list.icon} + </span> + </div> + {invitation.list.description && ( + <div className="mt-1 text-sm text-muted-foreground"> + {invitation.list.description} + </div> + )} + <div className="mt-2 text-sm text-muted-foreground"> + {t("lists.invitations.invited_by")}{" "} + <span className="font-medium"> + {invitation.list.owner?.name || "Unknown"} + </span> + {" • "} + <span className="capitalize">{invitation.role}</span> + </div> + </div> + <div className="flex items-center gap-2"> + <Button + size="sm" + variant="outline" + onClick={() => + declineInvitation.mutate({ invitationId: invitation.id }) + } + disabled={declineInvitation.isPending || acceptInvitation.isPending} + > + {declineInvitation.isPending ? ( + <Loader2 className="h-4 w-4 animate-spin" /> + ) : ( + <> + <X className="mr-1 h-4 w-4" /> + {t("lists.invitations.decline")} + </> + )} + </Button> + <Button + size="sm" + onClick={() => + acceptInvitation.mutate({ invitationId: invitation.id }) + } + disabled={acceptInvitation.isPending || declineInvitation.isPending} + > + {acceptInvitation.isPending ? ( + <Loader2 className="h-4 w-4 animate-spin" /> + ) : ( + <> + <Check className="mr-1 h-4 w-4" /> + {t("lists.invitations.accept")} + </> + )} + </Button> + </div> + </div> + ); +} + +export function PendingInvitationsCard() { + const { t } = useTranslation(); + + const { data: invitations, isLoading } = + api.lists.getPendingInvitations.useQuery(); + + if (isLoading) { + return null; + } + + if (!invitations || invitations.length === 0) { + return null; + } + + return ( + <Card> + <CardHeader> + <CardTitle className="flex items-center gap-2"> + <Mail className="h-5 w-5" /> + {t("lists.invitations.pending")} ({invitations.length}) + </CardTitle> + <CardDescription>{t("lists.invitations.description")}</CardDescription> + </CardHeader> + <CardContent className="space-y-3"> + {invitations.map((invitation) => ( + <InvitationRow key={invitation.id} invitation={invitation} /> + ))} + </CardContent> + </Card> + ); +} |
