'use client'; import { ChangeEventHandler, useCallback, useMemo, useState } from 'react'; import useSWR from 'swr'; import { CellContext, ColumnDef, SortingState } from '@tanstack/react-table'; import toast from 'react-hot-toast'; import { useFormik } from 'formik'; import { Icon } from '@iconify/react'; import Table from '@/components/Table'; import DebouncedTextInput from '@/components/input/DebouncedTextInput'; import Button from '@/components/Button'; import Modal, { useModal } from '@/components/Modal'; import ConfirmationModal from '@/components/modal/ConfirmationModal'; import RequirePermission from '@/components/helper/RequirePermission'; import PopoverButton from '@/components/popover/PopoverButton'; import PopoverContent from '@/components/popover/PopoverContent'; import WarehouseTableSkeleton from '@/components/pages/master-data/warehouse/skeleton/WarehouseTableSkeleton'; import { OptionType } from '@/components/input/SelectInput'; import ButtonFilter from '@/components/helper/ButtonFilter'; import { Warehouse } from '@/types/api/master-data/warehouse'; import { WarehouseApi, AreaApi } from '@/services/api/master-data'; import { isResponseError, isResponseSuccess } from '@/lib/api-helper'; import { useTableFilter } from '@/services/hooks/useTableFilter'; import { WarehouseFilterSchema, WarehouseFilterType, } from '@/components/pages/master-data/warehouse/filter/WarehouseFilter'; import { Area } from '@/types/api/master-data/area'; import SelectInput, { useSelect } from '@/components/input/SelectInput'; import SelectInputRadio from '@/components/input/SelectInputRadio'; const RowOptionsMenu = ({ popoverPosition = 'bottom', props, deleteClickHandler, }: { popoverPosition: 'bottom' | 'top'; props: CellContext; deleteClickHandler: () => void; }) => { const popoverId = `warehouse#${props.row.original.id}`; const popoverAnchorName = `--anchor-warehouse#${props.row.original.id}`; const closePopover = () => { document.getElementById(popoverId)?.hidePopover(); }; return (
); }; const WarehousesTable = () => { const { state: tableFilterState, updateFilter, setPage, setPageSize, toQueryString: getTableFilterQueryString, } = useTableFilter({ initial: { search: '', areaFilter: '', activeProjectFlockFilter: '', }, paramMap: { page: 'page', pageSize: 'limit', areaFilter: 'area_id', activeProjectFlockFilter: 'active_project_flock', }, persist: true, storeName: 'warehouses-table', }); // ===== FILTER MODAL STATE ===== const filterModal = useModal(); // ===== FORMIK SETUP ===== const formik = useFormik({ initialValues: { area_id: tableFilterState.areaFilter || null, active_project_flock: tableFilterState.activeProjectFlockFilter === 'true', }, validationSchema: WarehouseFilterSchema, onSubmit: (values, { setSubmitting }) => { updateFilter('areaFilter', values.area_id || '', true); updateFilter( 'activeProjectFlockFilter', values.active_project_flock === true ? 'true' : '', true ); filterModal.closeModal(); setSubmitting(false); }, }); const formikResetHandler = () => { updateFilter('areaFilter', '', true); updateFilter('activeProjectFlockFilter', '', true); formik.resetForm({ values: { area_id: null, active_project_flock: false, }, }); filterModal.closeModal(); }; // ===== AREA OPTIONS ===== const { setInputValue: setAreaInputValue, options: areaOptions, isLoadingOptions: isLoadingAreaOptions, loadMore: loadMoreAreas, } = useSelect( filterModal.open ? AreaApi.basePath : null, 'id', 'name', 'search' ); // ===== ACTIVE PROJECT FLOCK OPTIONS ===== const activeProjectFlockOptions = useMemo( () => [ { value: 'true', label: 'Kandang Aktif' }, { value: 'false', label: 'Semua Kandang' }, ], [] ); // ===== FILTER HANDLERS ===== const handleFilterAreaChange = useCallback( (val: OptionType | OptionType[] | null) => { const area = val as OptionType | null; const areaId = area?.value ? String(area.value) : null; formik.setFieldValue('area_id', areaId); }, [formik] ); const handleFilterActiveProjectFlockChange = useCallback( (val: OptionType | OptionType[] | null) => { const option = val as OptionType | null; const boolValue = option?.value === 'true' ? true : option?.value === 'false' ? false : null; formik.setFieldValue('active_project_flock', boolValue); }, [formik] ); // ===== FILTER HELPERS ===== const areaIdValue = useMemo(() => { if (!formik.values.area_id) return null; return ( areaOptions.find((opt) => String(opt.value) === formik.values.area_id) || null ); }, [formik.values.area_id, areaOptions]); const activeProjectFlockValue = useMemo(() => { if (formik.values.active_project_flock === null) return null; return ( activeProjectFlockOptions.find( (opt) => opt.value === String(formik.values.active_project_flock) ) || activeProjectFlockOptions[1] ); }, [formik.values.active_project_flock, activeProjectFlockOptions]); // ===== HANDLE FILTER MODAL OPEN ===== const handleFilterModalOpen = () => { filterModal.openModal(); formik.validateForm(); }; const [sorting, setSorting] = useState([]); const { data: warehouses, isLoading, mutate: refreshWarehouses, } = useSWR( `${WarehouseApi.basePath}${getTableFilterQueryString()}`, WarehouseApi.getAllFetcher ); const deleteModal = useModal(); const [selectedWarehouse, setSelectedWarehouse] = useState< Warehouse | undefined >(undefined); const [isDeleteLoading, setIsDeleteLoading] = useState(false); const searchChangeHandler: ChangeEventHandler = (e) => { updateFilter('search', e.target.value, true); }; const confirmationModalDeleteClickHandler = async () => { setIsDeleteLoading(true); const deleteResponse = await WarehouseApi.delete( selectedWarehouse?.id as number ); if (isResponseError(deleteResponse)) { toast.error(deleteResponse.message); setIsDeleteLoading(false); return; } refreshWarehouses(); deleteModal.closeModal(); toast.success('Successfully delete Warehouse!'); setIsDeleteLoading(false); }; const warehousesColumns: ColumnDef[] = useMemo( () => [ { header: 'No', cell: (props) => tableFilterState.pageSize * (tableFilterState.page - 1) + props.row.index + 1, }, { accessorKey: 'name', header: 'Nama', }, { accessorKey: 'type', header: 'Tipe', }, { accessorFn: (row) => row.area?.name ?? '-', header: 'Area', }, { accessorKey: 'location', header: 'Lokasi', cell: (props) => { if ( props.row.original.type === 'LOKASI' || props.row.original.type === 'KANDANG' ) { return props.row.original.location?.name ?? '-'; } return '-'; }, }, { accessorKey: 'kandang', header: 'Kandang', cell: (props) => { if (props.row.original.type === 'KANDANG') { return props.row.original.kandang?.name ?? '-'; } return '-'; }, }, { 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 = () => { setSelectedWarehouse(props.row.original); deleteModal.openModal(); }; return ( ); }, }, ], [tableFilterState.pageSize, tableFilterState.page, deleteModal] ); 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(warehouses) || warehouses.data?.length === 0 ? (
} />
) : ( data={warehouses?.data} columns={warehousesColumns} pageSize={tableFilterState.pageSize} page={warehouses?.meta?.page ?? 0} totalItems={warehouses?.meta?.total_results ?? 0} onPageChange={setPage} onPageSizeChange={setPageSize} isLoading={false} sorting={sorting} setSorting={setSorting} className={{ containerClassName: 'p-3 mb-0', headerColumnClassName: 'text-nowrap', }} /> )}
{/* Filter Modal */} {/* Modal Header */}

Filter Data

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