'use client'; import { ChangeEventHandler, useCallback, useEffect, useMemo, useState, } from 'react'; import { usePathname } from 'next/navigation'; import useSWR, { mutate } from 'swr'; import { SortingState, CellContext, ColumnDef } from '@tanstack/react-table'; import { useFormik } from 'formik'; import Table from '@/components/Table'; import { Icon } from '@iconify/react'; import { Movement } from '@/types/api/inventory/movement'; import { MovementApi } from '@/services/api/inventory'; import { WarehouseApi, ProductApi } from '@/services/api/master-data'; import { cn } from '@/lib/helper'; import { isResponseSuccess } from '@/lib/api-helper'; import { useTableFilter } from '@/services/hooks/useTableFilter'; import { useUiStore } from '@/stores/ui/ui.store'; import ConfirmationModal from '@/components/modal/ConfirmationModal'; import toast from 'react-hot-toast'; import Button from '@/components/Button'; import DebouncedTextInput from '@/components/input/DebouncedTextInput'; import SelectInput, { useSelect } from '@/components/input/SelectInput'; import { OptionType } from '@/components/input/SelectInput'; import ButtonFilter from '@/components/helper/ButtonFilter'; import Modal, { useModal } from '@/components/Modal'; import RequirePermission from '@/components/helper/RequirePermission'; import PopoverButton from '@/components/popover/PopoverButton'; import PopoverContent from '@/components/popover/PopoverContent'; import MovementTableSkeleton from '@/components/pages/inventory/movement/skeleton/MovementTableSkeleton'; import { Warehouse } from '@/types/api/master-data/warehouse'; import { Product } from '@/types/api/master-data/product'; import { MovementFilterSchema, MovementFilterType, } from '@/components/pages/inventory/movement/filter/MovementFilter'; const RowOptionsMenu = ({ popoverPosition = 'bottom', props, deleteClickHandler, }: { popoverPosition: 'bottom' | 'top'; props: CellContext; deleteClickHandler: () => void; }) => { const popoverId = `movement#${props.row.original.id}`; const popoverAnchorName = `--anchor-movement#${props.row.original.id}`; const closePopover = () => { document.getElementById(popoverId)?.hidePopover(); }; return (
); }; const MovementTable = () => { const { searchValue, setSearchValue, setTableState } = useUiStore(); const pathname = usePathname(); const { state: tableFilterState, updateFilter, setPage, setPageSize, toQueryString: getTableFilterQueryString, } = useTableFilter({ initial: { search: '', productFilter: '', warehouseFilter: '', }, paramMap: { page: 'page', pageSize: 'limit', productFilter: 'product_id', warehouseFilter: 'warehouse_id', }, }); // ===== FILTER MODAL STATE ===== const filterModal = useModal(); // ===== FORMIK SETUP ===== const formik = useFormik({ initialValues: { product_id: null, warehouse_id: null, }, validationSchema: MovementFilterSchema, onSubmit: (values, { setSubmitting }) => { updateFilter('productFilter', values.product_id || ''); updateFilter('warehouseFilter', values.warehouse_id || ''); filterModal.closeModal(); setSubmitting(false); }, onReset: () => { updateFilter('productFilter', ''); updateFilter('warehouseFilter', ''); }, }); // ===== PRODUCT OPTIONS ===== const { setInputValue: setProductInputValue, options: productOptions, isLoadingOptions: isLoadingProductOptions, loadMore: loadMoreProducts, } = useSelect( filterModal.open ? ProductApi.basePath : null, 'id', 'name', 'search' ); // ===== WAREHOUSE OPTIONS ===== const { setInputValue: setWarehouseInputValue, options: warehouseOptions, isLoadingOptions: isLoadingWarehouseOptions, loadMore: loadMoreWarehouses, } = useSelect( filterModal.open ? WarehouseApi.basePath : null, 'id', 'name', 'search' ); // ===== FILTER HANDLERS ===== const handleFilterProductChange = useCallback( (val: OptionType | OptionType[] | null) => { const product = val as OptionType | null; const productId = product?.value ? String(product.value) : null; formik.setFieldValue('product_id', productId); }, [formik] ); const handleFilterWarehouseChange = useCallback( (val: OptionType | OptionType[] | null) => { const warehouse = val as OptionType | null; const warehouseId = warehouse?.value ? String(warehouse.value) : null; formik.setFieldValue('warehouse_id', warehouseId); }, [formik] ); // ===== FILTER HELPERS ===== const productIdValue = useMemo(() => { if (!formik.values.product_id) return null; return ( productOptions.find( (opt) => String(opt.value) === formik.values.product_id ) || null ); }, [formik.values.product_id, productOptions]); const warehouseIdValue = useMemo(() => { if (!formik.values.warehouse_id) return null; return ( warehouseOptions.find( (opt) => String(opt.value) === formik.values.warehouse_id ) || null ); }, [formik.values.warehouse_id, warehouseOptions]); // ===== HANDLE FILTER MODAL OPEN ===== const handleFilterModalOpen = () => { filterModal.openModal(); formik.validateForm(); }; const [sorting, setSorting] = useState([]); const [selectedMovement, setSelectedMovement] = useState< Movement | undefined >(undefined); const [isDeleteLoading, setIsDeleteLoading] = useState(false); const singleDeleteModal = useModal(); const { data: movements, isLoading, mutate: refreshMovements, } = useSWR( `${MovementApi.basePath}${getTableFilterQueryString()}`, MovementApi.getAllFetcher ); const singleDeleteHandler = async () => { setIsDeleteLoading(true); const response = await MovementApi.delete(selectedMovement?.id as number); singleDeleteModal.closeModal(); setIsDeleteLoading(false); if (isResponseSuccess(response)) { toast.success(response?.message || 'Successfully delete Movement!'); refreshMovements(); } else { toast.error(response?.message || 'Failed to delete Movement'); } }; useEffect(() => { updateFilter('search', searchValue); }, [searchValue, updateFilter]); useEffect(() => { setTableState('movement-table', pathname); }, [pathname, setTableState]); const searchChangeHandler: ChangeEventHandler = (e) => { setSearchValue(e.target.value); updateFilter('search', e.target.value); }; const movementColumns: ColumnDef[] = useMemo( () => [ { header: 'No', cell: (props) => tableFilterState.pageSize * (tableFilterState.page - 1) + props.row.index + 1, }, { accessorFn: (row) => row.source_warehouse?.name, header: 'Gudang Asal', }, { accessorFn: (row) => row.destination_warehouse?.name, header: 'Gudang Tujuan', }, { accessorKey: 'transfer_reason', header: 'Catatan', }, { accessorKey: 'transfer_date', header: 'Tanggal', cell: (props) => new Date(props.row.original.transfer_date).toLocaleDateString( 'id-ID' ), }, { accessorFn: (row) => { const totalCost = row.deliveries?.reduce( (sum, d) => sum + (d.shipping_cost_total || 0), 0 ); return totalCost?.toLocaleString('id-ID'); }, header: 'Biaya Pengiriman', }, { header: 'Aksi', cell: (props: CellContext) => { 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 = () => { setSelectedMovement(props.row.original); singleDeleteModal.openModal(); }; return ( ); }, }, ], [ tableFilterState.pageSize, tableFilterState.page, singleDeleteModal, setSelectedMovement, ] ); return ( <>
{/* Header Section */}
{/* Action Buttons */}
{/* 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', }} />
{/* Table Section */}
{isLoading ? (
) : !isResponseSuccess(movements) || movements.data?.length === 0 ? (
} />
) : ( data={isResponseSuccess(movements) ? movements?.data : []} columns={movementColumns} pageSize={tableFilterState.pageSize} page={isResponseSuccess(movements) ? movements?.meta?.page : 0} totalItems={ isResponseSuccess(movements) ? movements?.meta?.total_results : 0 } onPageChange={setPage} onPageSizeChange={setPageSize} isLoading={isLoading} sorting={sorting} setSorting={setSorting} className={{ containerClassName: cn('p-3 mb-0'), headerColumnClassName: 'text-nowrap', }} /> )}
{/* Filter Modal */} {/* Modal Header */}

Filter Data

{/* Modal Footer */}
); }; export default MovementTable;