'use client'; import { useMemo } from 'react'; import { useSearchParams } from 'next/navigation'; import { msg } from '@lingui/macro'; import { useLingui } from '@lingui/react'; import { DateTime } from 'luxon'; import type { DateTimeFormatOptions } from 'luxon'; import { UAParser } from 'ua-parser-js'; import { useUpdateSearchParams } from '@documenso/lib/client-only/hooks/use-update-search-params'; import { ZUrlSearchParamsSchema } from '@documenso/lib/types/search-params'; import { formatDocumentAuditLogAction } from '@documenso/lib/utils/document-audit-logs'; import { trpc } from '@documenso/trpc/react'; import type { DataTableColumnDef } from '@documenso/ui/primitives/data-table'; import { DataTable } from '@documenso/ui/primitives/data-table'; import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination'; import { Skeleton } from '@documenso/ui/primitives/skeleton'; import { TableCell } from '@documenso/ui/primitives/table'; export type DocumentLogsDataTableProps = { documentId: number; }; const dateFormat: DateTimeFormatOptions = { ...DateTime.DATETIME_SHORT, hourCycle: 'h12', }; export const DocumentLogsDataTable = ({ documentId }: DocumentLogsDataTableProps) => { const { _, i18n } = useLingui(); const searchParams = useSearchParams(); const updateSearchParams = useUpdateSearchParams(); const parsedSearchParams = ZUrlSearchParamsSchema.parse(Object.fromEntries(searchParams ?? [])); const { data, isLoading, isLoadingError } = trpc.document.findDocumentAuditLogs.useQuery( { documentId, page: parsedSearchParams.page, perPage: parsedSearchParams.perPage, }, { placeholderData: (previousData) => previousData, }, ); const onPaginationChange = (page: number, perPage: number) => { updateSearchParams({ page, perPage, }); }; const results = data ?? { data: [], perPage: 10, currentPage: 1, totalPages: 1, }; const columns = useMemo(() => { const parser = new UAParser(); return [ { header: _(msg`Time`), accessorKey: 'createdAt', cell: ({ row }) => i18n.date(row.original.createdAt, dateFormat), }, { header: _(msg`User`), accessorKey: 'name', cell: ({ row }) => row.original.name || row.original.email ? (
{row.original.name}
)} {row.original.email && ({row.original.email}
)}N/A
), }, { header: _(msg`Action`), accessorKey: 'type', cell: ({ row }) => {formatDocumentAuditLogAction(_, row.original).description}, }, { header: 'IP Address', accessorKey: 'ipAddress', }, { header: 'Browser', cell: ({ row }) => { if (!row.original.userAgent) { return 'N/A'; } parser.setUA(row.original.userAgent); const result = parser.getResult(); return result.browser.name ?? 'N/A'; }, }, ] satisfies DataTableColumnDef<(typeof results)['data'][number]>[]; }, []); return (