'use client'; import { ChangeEventHandler, useCallback, useEffect, useMemo, useState, } from 'react'; import { usePathname } from 'next/navigation'; import { useUiStore } from '@/stores/ui/ui.store'; import useSWR from 'swr'; import useSWRInfinite from 'swr/infinite'; import { CellContext, ColumnDef, SortingState } from '@tanstack/react-table'; import toast from 'react-hot-toast'; import { Icon } from '@iconify/react'; import Table from '@/components/Table'; import DebouncedTextInput from '@/components/input/DebouncedTextInput'; import Button from '@/components/Button'; import { useModal } from '@/components/Modal'; import ConfirmationModal from '@/components/modal/ConfirmationModal'; import PopoverButton from '@/components/popover/PopoverButton'; import PopoverContent from '@/components/popover/PopoverContent'; import SelectInput, { OptionType } from '@/components/input/SelectInput'; import RequirePermission from '@/components/helper/RequirePermission'; import StatusBadge from '@/components/helper/StatusBadge'; import PurchaseTableSkeleton from '@/components/pages/purchase/skeleton/PurchaseTableSkeleton'; import { cn, formatDate } from '@/lib/helper'; import { isResponseSuccess } from '@/lib/api-helper'; import { BaseApiResponse } from '@/types/api/api-general'; import { useTableFilter } from '@/services/hooks/useTableFilter'; import { ROWS_OPTIONS } from '@/config/constant'; import { Purchase } from '@/types/api/purchase/purchase'; import { PurchaseApi } from '@/services/api/purchase'; import { ExpenseApi } from '@/services/api/expense'; import { Expense } from '@/types/api/expense'; import { Color } from '@/types/theme'; import Link from 'next/link'; // ===== STATUS BADGE UTILITIES ===== const statusTextMap: Record = { APPROVED: 'Disetujui', Disetujui: 'Disetujui', REJECTED: 'Ditolak', Ditolak: 'Ditolak', CREATED: 'Dibuat', UPDATED: 'Diperbarui', }; const getStatusText = (status: string): string => { return statusTextMap[status] || status; }; const statusBadgeColorMap: Record = { APPROVED: 'success', Disetujui: 'success', approved: 'success', disetujui: 'success', REJECTED: 'error', Ditolak: 'error', rejected: 'error', ditolak: 'error', CREATED: 'neutral', Dibuat: 'neutral', created: 'neutral', dibuat: 'neutral', UPDATED: 'warning', Diperbarui: 'warning', updated: 'warning', diperbarui: 'warning', }; const getStatusBadgeColor = (status: string): Color => { return statusBadgeColorMap[status] || 'neutral'; }; // ===== ROW OPTIONS MENU ===== const RowOptionsMenu = ({ popoverPosition = 'bottom', props, deleteClickHandler, }: { popoverPosition: 'bottom' | 'top'; props: CellContext; deleteClickHandler: () => void; }) => { const popoverId = `purchase#${props.row.original.id}`; const popoverAnchorName = `--anchor-purchase#${props.row.original.id}`; const closePopover = () => { document.getElementById(popoverId)?.hidePopover(); }; return (
); }; const PurchaseTable = () => { const { searchValue, setSearchValue, setTableState } = useUiStore(); const pathname = usePathname(); // ===== STATE MANAGEMENT ===== const [isDeleteLoading, setIsDeleteLoading] = useState(false); const [selectedPurchase, setSelectedPurchase] = useState( null ); const [sorting, setSorting] = useState([]); // ===== TABLE FILTER STATE ===== const { state: tableFilterState, updateFilter, setPage, setPageSize, toQueryString: getTableFilterQueryString, } = useTableFilter({ initial: { search: '', }, paramMap: { page: 'page', pageSize: 'limit', }, }); // ===== MODAL HOOKS ===== const deleteModal = useModal(); // ===== API DATA FETCHING ===== const { data: purchaseRequests, isLoading, mutate: refreshPurchaseRequests, } = useSWR( `${PurchaseApi.basePath}${getTableFilterQueryString()}`, PurchaseApi.getAllFetcher ); const getKey = ( pageIndex: number, previousPageData: BaseApiResponse[] | null ) => { if (pageIndex > 0 && !previousPageData) return null; return `${ExpenseApi.basePath}?page=${pageIndex + 1}&limit=100`; }; const { data: expensesPages } = useSWRInfinite( getKey, ExpenseApi.getAllFetcher ); const expenseMap = useMemo(() => { const map = new Map(); if (!expensesPages) return map; expensesPages.forEach((page) => { if (isResponseSuccess(page)) { page.data.forEach((expense: Expense) => { map.set(expense.reference_number, expense.id); }); } }); return map; }, [expensesPages]); // ===== TABLE COLUMNS DEFINITION ===== const purchaseColumns: ColumnDef[] = [ { header: 'No. PR/PO', cell: (props) => { const { pr_number, po_number } = props.row.original; return po_number ? po_number : pr_number; }, }, { accessorKey: 'po_expedition', header: 'Ekspedisi PO', cell: (props) => { const poExpedition = props.row.original.po_expedition; if (!poExpedition || poExpedition.length === 0) return '-'; return (
    {poExpedition.map((exp, index) => { const expenseId = expenseMap.get(exp.refrence); if (expenseId) { return (
  • {exp.refrence}
  • ); } return
  • {exp.refrence}
  • ; })}
); }, }, { accessorKey: 'supplier', header: 'Vendor', cell: (props) => props.row.original.supplier.name, }, { accessorKey: 'requester_name', header: 'Nama Pengaju', cell: (props) => props.row.original.requester_name || '-', }, { accessorKey: 'products.name', header: 'Produk', cell: (props) => { const products = props.row.original.products; if (!products || products.length === 0) return '-'; return (
    {products.map((product, index) => (
  • {product.name}
  • ))}
); }, }, { accessorKey: 'location.name', header: 'Lokasi', cell: (props) => props.row.original.location?.name || '-', }, { accessorKey: 'po_date', header: 'Tgl. PO', cell: (props) => props.row.original.po_date ? formatDate(props.row.original.po_date, 'DD MMM YYYY') : '-', }, { accessorKey: 'due_date', header: 'Jatuh Tempo', cell: (props) => props.row.original.due_date ? formatDate(props.row.original.due_date, 'DD MMM YYYY') : '-', }, { header: 'Aging', cell: (props) => { const purchase = props.row.original; if (!purchase.po_date) return '-'; const poDate = new Date(purchase.po_date); const today = new Date(); const diffTime = Math.abs(today.getTime() - poDate.getTime()); const diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24)); return `${diffDays} hari`; }, }, { header: 'Status Approval', cell: (props) => { const approval = props.row.original.latest_approval; if (!approval) return '-'; const status = approval.action; let statusColor: Color = 'neutral'; if (status === 'REJECTED') { statusColor = getStatusBadgeColor(status); } else { switch (approval.step_number) { case 1: statusColor = 'neutral'; break; case 2: statusColor = 'primary'; break; case 3: statusColor = 'info'; break; case 4: statusColor = 'warning'; break; case 5: statusColor = 'success'; break; } } const statusText = approval.step_name || getStatusText(status); return ( ); }, }, { header: 'Aksi', cell: (props) => { const currentPageSize = props.table.getPaginationRowModel().rows.length; const currentPageRows = props.table.getPaginationRowModel().flatRows; const currentRowRelativeIndex = currentPageRows.findIndex((r) => r.id === props.row.id) + 1; const isLast2Rows = currentRowRelativeIndex > currentPageSize - 2; const deleteClickHandler = () => { setSelectedPurchase(props.row.original); deleteModal.openModal(); }; return ( ); }, }, ]; // ===== EVENT HANDLERS ===== const confirmationModalDeleteClickHandler = useCallback(async () => { setIsDeleteLoading(true); try { await PurchaseApi.delete(selectedPurchase?.id as number); refreshPurchaseRequests(); deleteModal.closeModal(); toast.success('Berhasil menghapus data permintaan pembelian!'); } catch { toast.error('Gagal menghapus data permintaan pembelian!'); } setIsDeleteLoading(false); }, [selectedPurchase?.id, refreshPurchaseRequests, deleteModal]); useEffect(() => { updateFilter('search', searchValue); }, [searchValue, updateFilter]); useEffect(() => { setTableState('purchase-table', pathname); }, [pathname, setTableState]); const searchChangeHandler: ChangeEventHandler = useCallback( (e) => { setSearchValue(e.target.value); updateFilter('search', e.target.value); }, [updateFilter, setSearchValue] ); const pageSizeChangeHandler = useCallback( (val: OptionType | OptionType[] | null) => { const newVal = val as OptionType; setPageSize(newVal.value as number); }, [setPageSize] ); return ( <>
} className={{ wrapper: 'w-full min-w-24 max-w-3xs', inputWrapper: 'rounded-xl! shadow-button-soft', input: 'placeholder:font-semibold placeholder:text-base-content/50', }} />
{/* Table Section */}
{isLoading ? (
) : !isResponseSuccess(purchaseRequests) || purchaseRequests.data?.length === 0 ? (
} />
) : ( data={ isResponseSuccess(purchaseRequests) ? purchaseRequests?.data : [] } columns={purchaseColumns} pageSize={tableFilterState.pageSize} page={ isResponseSuccess(purchaseRequests) ? purchaseRequests?.meta?.page : 0 } totalItems={ isResponseSuccess(purchaseRequests) ? purchaseRequests?.meta?.total_results : 0 } onPageChange={setPage} onPageSizeChange={setPageSize} isLoading={isLoading} sorting={sorting} setSorting={setSorting} className={{ containerClassName: cn('p-3 mb-0'), headerColumnClassName: 'text-nowrap', }} /> )}
{/* ===== MODAL COMPONENTS ===== */} deleteModal.closeModal(), }} primaryButton={{ text: 'Ya', color: 'error', isLoading: isDeleteLoading, onClick: confirmationModalDeleteClickHandler, }} /> ); }; export default PurchaseTable;