'use client'; import { ChangeEventHandler, useCallback, useState } from 'react'; import useSWR from 'swr'; 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 SelectInput, { OptionType } from '@/components/input/SelectInput'; import RowDropdownOptions from '@/components/table/RowDropdownOptions'; import RowCollapseOptions from '@/components/table/RowCollapseOptions'; import RowOptionsMenuWrapper from '@/components/table/RowOptionsMenuWrapper'; import { cn, formatDate } from '@/lib/helper'; import { isResponseSuccess } from '@/lib/api-helper'; 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'; // ===== INTERFACES ===== interface RowOptionsMenuProps { type: 'dropdown' | 'collapse'; props: CellContext; deleteClickHandler: () => void; } const RowOptionsMenu = ({ type = 'dropdown', props, deleteClickHandler, }: RowOptionsMenuProps) => { return ( {/**/} {/* */} {/* Edit*/} {/**/} ); }; const PurchaseTable = () => { // ===== 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 ); // ===== 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: 'supplier', header: 'Vendor', cell: (props) => props.row.original.supplier.name, }, { accessorKey: 'po_date', header: 'Tgl. PO', cell: (props) => props.row.original.po_date ? formatDate(props.row.original.po_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: '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 ( <> {currentPageSize > 2 && ( )} {currentPageSize <= 2 && ( )} ); }, }, ]; // ===== 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]); const searchChangeHandler: ChangeEventHandler = useCallback( (e) => { updateFilter('search', e.target.value); }, [updateFilter] ); const pageSizeChangeHandler = useCallback( (val: OptionType | OptionType[] | null) => { const newVal = val as OptionType; setPageSize(newVal.value as number); }, [setPageSize] ); return ( <>
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} isLoading={isLoading} sorting={sorting} setSorting={setSorting} className={{ containerClassName: cn({ 'mb-20': isResponseSuccess(purchaseRequests) && purchaseRequests?.data?.length === 0, }), tableWrapperClassName: 'overflow-x-auto min-h-full!', tableClassName: 'font-inter w-full table-auto min-h-full!', headerRowClassName: 'border-b border-b-gray-200', headerColumnClassName: 'px-6 py-3 text-xs font-semibold text-gray-500 last:flex last:flex-row last:justify-end', bodyRowClassName: 'border-b border-b-gray-200', bodyColumnClassName: 'px-6 py-3 last:flex last:flex-row last:justify-end', }} />
{/* ===== MODAL COMPONENTS ===== */} ); }; export default PurchaseTable;