'use client'; import axios from 'axios'; import { ChangeEventHandler, useCallback, useEffect, useMemo, useState, } from 'react'; import useSWR from 'swr'; import { CellContext, ColumnDef, Row, 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 DateInput from '@/components/input/DateInput'; import SelectInput, { OptionType } from '@/components/input/SelectInput'; import TextArea from '@/components/input/TextArea'; import Button from '@/components/Button'; import Modal, { useModal } from '@/components/Modal'; import ConfirmationModal from '@/components/modal/ConfirmationModal'; import PopoverButton from '@/components/popover/PopoverButton'; import PopoverContent from '@/components/popover/PopoverContent'; import RealizationStatusBadge from '@/components/pages/expense/RealizationStatusBadge'; import ExpenseStatusBadge from '@/components/pages/expense/ExpenseStatusBadge'; import CheckboxInput from '@/components/input/CheckboxInput'; import ConfirmationModalWithNotes from '@/components/modal/ConfirmationModalWithNotes'; import RequirePermission from '@/components/helper/RequirePermission'; import ButtonFilter from '@/components/helper/ButtonFilter'; import ExpensesFilterModal from '@/components/pages/expense/filter/ExpensesFilterModal'; import ExpenseTableSkeleton from '@/components/pages/expense/skeleton/ExpenseTableSkeleton'; import Dropdown from '@/components/dropdown/Dropdown'; import { Expense } from '@/types/api/expense'; import { ExpenseApi } from '@/services/api/expense'; import { cn, formatCurrency, formatDate } from '@/lib/helper'; import { isResponseSuccess } from '@/lib/api-helper'; import { useTableFilter } from '@/services/hooks/useTableFilter'; import { BaseApiResponse } from '@/types/api/api-general'; type ExpenseTableFilters = { search: string; nameSort: string; transactionDate: string; realizationDate: string; locationId: string; vendorId: string; userId: string; }; const approvalStatusOptions = [ { value: 'HEAD_AREA', label: 'Approval Head Area' }, { value: 'UNIT_VICE_PRESIDENT', label: 'Approval Unit Vice President' }, { value: 'FINANCE', label: 'Approval Finance' }, { value: 'REALISASI', label: 'Realisasi' }, { value: 'SELESAI', label: 'Selesai' }, ] as const satisfies OptionType< 'HEAD_AREA' | 'UNIT_VICE_PRESIDENT' | 'FINANCE' | 'REALISASI' | 'SELESAI' >[]; type ApprovalStatusValue = | 'HEAD_AREA' | 'UNIT_VICE_PRESIDENT' | 'FINANCE' | 'REALISASI' | 'SELESAI'; const isApprovalDateRequired = (status?: ApprovalStatusValue) => status === 'REALISASI' || status === 'SELESAI'; const getExportErrorMessage = async ( error: unknown, fallbackMessage: string ) => { if (axios.isAxiosError(error)) { const responseData = error.response?.data; if (responseData instanceof Blob) { try { const parsed = JSON.parse(await responseData.text()) as { message?: string; }; return parsed.message || fallbackMessage; } catch { return fallbackMessage; } } if ( responseData && typeof responseData === 'object' && 'message' in responseData && typeof responseData.message === 'string' ) { return responseData.message; } return error.message || fallbackMessage; } if (error instanceof Error) { return error.message; } return fallbackMessage; }; const RowOptionsMenu = ({ popoverPosition = 'bottom', props, deleteClickHandler, }: { popoverPosition: 'bottom' | 'top'; props: CellContext; deleteClickHandler: () => void; }) => { const popoverId = `expense#${props.row.original.id}`; const popoverAnchorName = `--anchor-expense#${props.row.original.id}`; const closePopover = () => { document.getElementById(popoverId)?.hidePopover(); }; const showEditButton = props.row.original.latest_approval ? props.row.original.latest_approval.step_number !== 6 && (props.row.original.latest_approval.step_number === 1 || props.row.original.latest_approval.step_number === 2 || props.row.original.latest_approval.step_number === 3 || props.row.original.latest_approval.step_number === 4) : false; const showRealizationButton = props.row.original.latest_approval ? props.row.original.latest_approval.action !== 'REJECTED' && props.row.original.latest_approval.step_number === 4 : false; return (
{showEditButton && ( )} {showRealizationButton && ( )}
); }; const ExpensesTable = () => { const { state: tableFilterState, updateFilter, setPage, setPageSize, toQueryString: getTableFilterQueryString, } = useTableFilter({ initial: { page: 1, pageSize: 10, search: '', nameSort: '', transactionDate: '', realizationDate: '', locationId: '', vendorId: '', userId: '', }, paramMap: { page: 'page', pageSize: 'limit', nameSort: 'sort_name', transactionDate: 'transaction_date', realizationDate: 'realization_date', locationId: 'location_id', vendorId: 'vendor_id', userId: 'user_id', }, persist: true, storeName: 'expense-table', }); const { data: expenses, isLoading, mutate: refreshExpenses, } = useSWR( `${ExpenseApi.basePath}${getTableFilterQueryString()}`, ExpenseApi.getAllFetcher ); const deleteModal = useModal(); const approveModal = useModal(); const rejectModal = useModal(); const bulkApproveFormModal = useModal(); const exportProgressInputModal = useModal(); // ===== FILTER MODAL STATE ===== const filterModal = useModal(); const [selectedExpense, setSelectedExpense] = useState( undefined ); const [isDeleteLoading, setIsDeleteLoading] = useState(false); const [isApproveLoading, setIsApproveLoading] = useState(false); const [isRejectLoading, setIsRejectLoading] = useState(false); const [isExportProgressLoading, setIsExportProgressLoading] = useState(false); const [, setApprovalNotes] = useState(''); const [bulkApprovalStatus, setBulkApprovalStatus] = useState | null>(null); const [bulkApprovalDate, setBulkApprovalDate] = useState(''); const [bulkApprovalNotes, setBulkApprovalNotes] = useState(''); const [exportProgressStartDate, setExportProgressStartDate] = useState(''); const [exportProgressEndDate, setExportProgressEndDate] = useState(''); const [sorting, setSorting] = useState([]); const [rowSelection, setRowSelection] = useState>({}); const selectedRowIds = Object.keys(rowSelection).map((item) => parseInt(item) ); const isAllSelectedRowLatestApprovalOnHeadArea = useMemo(() => { return selectedRowIds.every((rowId) => { if (!isResponseSuccess(expenses)) return false; const expenseItem = expenses.data.find((item) => item.id === rowId); const isLatestApprovalRejected = expenseItem?.latest_approval.action === 'REJECTED'; const isCurrentApprovalOnHeadArea = !isLatestApprovalRejected && expenseItem?.latest_approval.step_number === 1; return isCurrentApprovalOnHeadArea; }); }, [expenses, selectedRowIds]); const isAllSelectedRowLatestApprovalOnUnitVicePresident = useMemo(() => { return selectedRowIds.every((rowId) => { if (!isResponseSuccess(expenses)) return false; const expenseItem = expenses.data.find((item) => item.id === rowId); const isLatestApprovalRejected = expenseItem?.latest_approval.action === 'REJECTED'; const isCurrentApprovalOnUnitVicePresident = !isLatestApprovalRejected && expenseItem?.latest_approval.step_number === 2; return isCurrentApprovalOnUnitVicePresident; }); }, [expenses, selectedRowIds]); const isAllSelectedRowLatestApprovalOnFinance = useMemo(() => { return selectedRowIds.every((rowId) => { if (!isResponseSuccess(expenses)) return false; const expenseItem = expenses.data.find((item) => item.id === rowId); const isLatestApprovalRejected = expenseItem?.latest_approval.action === 'REJECTED'; const isCurrentApprovalOnFinance = !isLatestApprovalRejected && expenseItem?.latest_approval.step_number === 3; return isCurrentApprovalOnFinance; }); }, [expenses, selectedRowIds]); const expensesColumns: ColumnDef[] = [ { id: 'select', header: ({ table }) => (
), cell: ({ row }) => { const isCheckboxDisabled = !row.getCanSelect() || !row.original.latest_approval || row.original.latest_approval.action === 'REJECTED'; return (
); }, }, { accessorKey: 'reference_number', header: 'Nomor Referensi', cell: (props) => { return props.row.original.reference_number ?? '-'; }, }, { accessorKey: 'transaction_date', header: 'Tanggal Pengajuan', cell: (props) => props.row.original.transaction_date ? formatDate(props.row.original.transaction_date, 'DD MMM YYYY') : '-', }, { accessorKey: 'realization_date', header: 'Tanggal Realisasi', cell: (props) => props.row.original.realization_date ? formatDate(props.row.original.realization_date, 'DD MMM YYYY') : '-', }, { accessorKey: 'location', header: 'Lokasi', cell: (props) => props.row.original.location?.name ?? '-', }, { accessorFn: (row) => row.created_user.name ?? '-', header: 'Nama Pengaju', }, { accessorFn: (row) => row.supplier.name ?? '-', header: 'Vendor', }, { accessorKey: 'grand_total', header: 'Nominal', cell: (props) => props.row.original.grand_total ? formatCurrency(props.row.original.grand_total) : '-', }, { header: 'Status Pencairan', cell: (props) => ( ), }, { header: 'Status BOP', cell: (props) => ( ), }, { 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 = () => { setSelectedExpense(props.row.original); deleteModal.openModal(); }; return ( ); }, }, ]; const tableEnableRowSelectionHandler: (row: Row) => boolean = ( row ) => { if (!row.original.latest_approval) return false; return ( row.original.latest_approval.action !== 'REJECTED' && row.original.latest_approval.step_number !== 6 ); }; const resetBulkApproveForm = useCallback(() => { setBulkApprovalStatus(null); setBulkApprovalDate(''); setBulkApprovalNotes(''); }, []); const openBulkApproveForm = useCallback( (presetStatus?: ApprovalStatusValue) => { resetBulkApproveForm(); if (presetStatus) { const selectedStatus = approvalStatusOptions.find( (option) => option.value === presetStatus ); if (selectedStatus) { setBulkApprovalStatus(selectedStatus); } } bulkApproveFormModal.openModal(); }, [bulkApproveFormModal, resetBulkApproveForm] ); const bulkApproveClickHandler = () => { openBulkApproveForm(); }; const bulkApproveHeadAreaClickHandler = () => { openBulkApproveForm('HEAD_AREA'); }; const bulkApproveUnitVicePresidentClickHandler = () => { openBulkApproveForm('UNIT_VICE_PRESIDENT'); }; const bulkApproveFinanceClickHandler = () => { openBulkApproveForm('FINANCE'); }; const bulkRejectClickHandler = () => { setApprovalNotes(''); rejectModal.openModal(); }; const bulkApprovalDateChangeHandler: ChangeEventHandler = ( e ) => { setBulkApprovalDate(e.target.value); }; const bulkApprovalNotesChangeHandler: ChangeEventHandler< HTMLTextAreaElement > = (e) => { setBulkApprovalNotes(e.target.value); }; const resetExportProgressForm = useCallback(() => { setExportProgressStartDate(''); setExportProgressEndDate(''); }, []); const exportProgressStartDateChangeHandler: ChangeEventHandler< HTMLInputElement > = (e) => { setExportProgressStartDate(e.target.value); }; const exportProgressEndDateChangeHandler: ChangeEventHandler< HTMLInputElement > = (e) => { setExportProgressEndDate(e.target.value); }; const exportProgressInputToExcelClickHandler = () => { resetExportProgressForm(); exportProgressInputModal.openModal(); }; const submitExportProgressInputHandler = async () => { if (!exportProgressStartDate || !exportProgressEndDate) { return; } setIsExportProgressLoading(true); try { await ExpenseApi.exportInputProgressToExcel( exportProgressStartDate, exportProgressEndDate ); exportProgressInputModal.closeModal(); resetExportProgressForm(); toast.success('Ekspor berhasil'); } catch (error) { toast.error( await getExportErrorMessage(error, 'Gagal mengekspor input progress') ); } finally { setIsExportProgressLoading(false); } }; const confirmationModalDeleteClickHandler = async () => { setIsDeleteLoading(true); const deleteResponse = await ExpenseApi.delete( selectedExpense?.id as number ); if (isResponseSuccess(deleteResponse)) { refreshExpenses(); deleteModal.closeModal(); toast.success('Berhasil menghapus biaya operasional!'); } else { deleteModal.closeModal(); toast.error('Gagal menghapus biaya operasional!'); } setIsDeleteLoading(false); }; const confirmationModalApproveClickHandler = async (notes: string) => { setIsApproveLoading(true); let bulkApproveResponse: BaseApiResponse | undefined = undefined; if (isAllSelectedRowLatestApprovalOnHeadArea) { bulkApproveResponse = await ExpenseApi.bulkApproveHeadArea( selectedRowIds, notes ); } else if (isAllSelectedRowLatestApprovalOnUnitVicePresident) { bulkApproveResponse = await ExpenseApi.bulkApproveUnitVicePresident( selectedRowIds, notes ); } else if (isAllSelectedRowLatestApprovalOnFinance) { bulkApproveResponse = await ExpenseApi.bulkApproveFinance( selectedRowIds, notes ); } if (isResponseSuccess(bulkApproveResponse)) { refreshExpenses(); approveModal.closeModal(); toast.success( `Berhasil approve ${selectedRowIds.length} data biaya operasional!` ); setApprovalNotes(''); setRowSelection({}); } else { approveModal.closeModal(); toast.error( `Gagal approve ${selectedRowIds.length} data biaya operasional!` ); } setIsApproveLoading(false); }; const bulkApproveSubmitHandler = async () => { if (!bulkApprovalStatus) { return; } if (isApprovalDateRequired(bulkApprovalStatus.value) && !bulkApprovalDate) { toast.error('Tanggal realisasi wajib diisi.'); return; } if (!bulkApprovalNotes.trim()) { toast.error('Catatan wajib diisi.'); return; } setIsApproveLoading(true); const bulkApproveResponse = await ExpenseApi.bulkApprovals( selectedRowIds, bulkApprovalStatus.value, isApprovalDateRequired(bulkApprovalStatus.value) ? bulkApprovalDate : '', bulkApprovalNotes ); if (isResponseSuccess(bulkApproveResponse)) { refreshExpenses(); bulkApproveFormModal.closeModal(); toast.success( `Berhasil approve ${selectedRowIds.length} data biaya operasional!` ); resetBulkApproveForm(); setRowSelection({}); } else { toast.error( bulkApproveResponse?.message ?? `Gagal approve ${selectedRowIds.length} data biaya operasional!` ); } setIsApproveLoading(false); }; const confirmationModalRejectClickHandler = async (notes: string) => { setIsRejectLoading(true); let bulkRejectResponse: BaseApiResponse | undefined = undefined; if (isAllSelectedRowLatestApprovalOnHeadArea) { bulkRejectResponse = await ExpenseApi.bulkRejectHeadArea( selectedRowIds, notes ); } else if (isAllSelectedRowLatestApprovalOnUnitVicePresident) { bulkRejectResponse = await ExpenseApi.bulkRejectUnitVicePresident( selectedRowIds, notes ); } else if (isAllSelectedRowLatestApprovalOnFinance) { bulkRejectResponse = await ExpenseApi.bulkRejectFinance( selectedRowIds, notes ); } if (isResponseSuccess(bulkRejectResponse)) { refreshExpenses(); rejectModal.closeModal(); toast.success( `Berhasil reject ${selectedRowIds.length} data biaya operasional!` ); setApprovalNotes(''); setRowSelection({}); } else { rejectModal.closeModal(); toast.error( `Gagal reject ${selectedRowIds.length} data biaya operasional!` ); } setIsRejectLoading(false); }; const searchChangeHandler: ChangeEventHandler = (e) => { updateFilter('search', e.target.value); }; // ===== FILTER MODAL HANDLERS ===== const handleFilterModalOpen = () => { filterModal.openModal(); }; const handleFilterSubmit = (values: { transaction_date?: string | null; realization_date?: string | null; location_id?: string | null; vendor_id?: string | null; }) => { updateFilter('transactionDate', values.transaction_date || ''); updateFilter('realizationDate', values.realization_date || ''); updateFilter('locationId', values.location_id || ''); updateFilter('vendorId', values.vendor_id || ''); }; const handleFilterReset = () => { updateFilter('transactionDate', ''); updateFilter('realizationDate', ''); updateFilter('locationId', ''); updateFilter('vendorId', ''); }; // track sorting useEffect(() => { const isNameSorted = sorting.find((sortItem) => sortItem.id === 'name'); if (!isNameSorted) { updateFilter('nameSort', '', false); } else { updateFilter('nameSort', isNameSorted.desc ? 'desc' : 'asc'); } }, [sorting, updateFilter]); return ( <>
{/* Action Buttons */}
{selectedRowIds.length > 0 && (
)}
{/* Search and Filter */}
} 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', }} />
Export
} >
{/* Table Section */}
{isLoading ? (
) : !isResponseSuccess(expenses) || expenses.data?.length === 0 ? (
} />
) : ( data={isResponseSuccess(expenses) ? expenses?.data : []} columns={expensesColumns} pageSize={tableFilterState.pageSize} page={isResponseSuccess(expenses) ? expenses?.meta?.page : 0} totalItems={ isResponseSuccess(expenses) ? expenses?.meta?.total_results : 0 } onPageChange={setPage} onPageSizeChange={setPageSize} isLoading={isLoading} sorting={sorting} setSorting={setSorting} rowSelection={rowSelection} setRowSelection={setRowSelection} enableRowSelection={tableEnableRowSelectionHandler} className={{ containerClassName: cn('p-3 mb-0'), headerColumnClassName: 'text-nowrap', }} /> )}
{ setApprovalNotes(''); approveModal.closeModal(); }, }} primaryButton={{ text: 'Ya', color: 'success', isLoading: isApproveLoading, onClick: confirmationModalApproveClickHandler, }} /> { setApprovalNotes(''); rejectModal.closeModal(); }, }} primaryButton={{ text: 'Ya', color: 'error', isLoading: isRejectLoading, onClick: confirmationModalRejectClickHandler, }} />

Bulk Approve Expense

{ const nextValue = val as OptionType | null; setBulkApprovalStatus(nextValue); if (!isApprovalDateRequired(nextValue?.value)) { setBulkApprovalDate(''); } }} placeholder='Pilih status approval' isClearable /> {isApprovalDateRequired(bulkApprovalStatus?.value) && ( )}