'use client'; import { ChangeEventHandler, useMemo, 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 RequirePermission from '@/components/helper/RequirePermission'; import PopoverButton from '@/components/popover/PopoverButton'; import PopoverContent from '@/components/popover/PopoverContent'; import ProductTableSkeleton from '@/components/pages/master-data/product/skeleton/ProductTableSkeleton'; import { Product } from '@/types/api/master-data/product'; import { ProductApi } from '@/services/api/master-data'; import { formatCurrency } from '@/lib/helper'; import { isResponseError, isResponseSuccess } from '@/lib/api-helper'; import { useTableFilter } from '@/services/hooks/useTableFilter'; const RowOptionsMenu = ({ popoverPosition = 'bottom', props, deleteClickHandler, }: { popoverPosition: 'bottom' | 'top'; props: CellContext; deleteClickHandler: () => void; }) => { const popoverId = `product#${props.row.original.id}`; const popoverAnchorName = `--anchor-product#${props.row.original.id}`; const closePopover = () => { document.getElementById(popoverId)?.hidePopover(); }; return (
); }; const ProductsTable = () => { const { state: tableFilterState, updateFilter, setPage, setPageSize, toQueryString: getTableFilterQueryString, } = useTableFilter({ initial: { search: '', }, paramMap: { page: 'page', pageSize: 'limit', }, }); const [sorting, setSorting] = useState([]); const { data: products, isLoading, mutate: refreshProducts, } = useSWR( `${ProductApi.basePath}${getTableFilterQueryString()}`, ProductApi.getAllFetcher ); const deleteModal = useModal(); const [selectedProduct, setSelectedProduct] = useState( undefined ); const [isDeleteLoading, setIsDeleteLoading] = useState(false); const searchChangeHandler: ChangeEventHandler = (e) => { updateFilter('search', e.target.value); }; const confirmationModalDeleteClickHandler = async () => { setIsDeleteLoading(true); const deleteResponse = await ProductApi.delete( selectedProduct?.id as number ); if (isResponseError(deleteResponse)) { toast.error(deleteResponse.message); setIsDeleteLoading(false); return; } refreshProducts(); deleteModal.closeModal(); toast.success('Successfully delete Product!'); setIsDeleteLoading(false); }; const productsColumns: ColumnDef[] = useMemo( () => [ { header: 'No', cell: (props) => tableFilterState.pageSize * (tableFilterState.page - 1) + props.row.index + 1, }, { accessorKey: 'name', header: 'Nama', }, { accessorKey: 'sku', header: 'SKU', }, { accessorKey: 'brand', header: 'Merek', }, { accessorFn: (row) => row.product_category?.name ?? '-', header: 'Kategori', }, { accessorFn: (row) => row.uom?.name ?? '-', header: 'Satuan', }, { accessorKey: 'product_price', header: 'Harga Produk', cell: (props) => props.row.original.product_price ? formatCurrency(props.row.original.product_price) : '-', }, { accessorKey: 'selling_price', header: 'Harga Jual', cell: (props) => props.row.original.selling_price ? formatCurrency(props.row.original.selling_price) : '-', }, { accessorKey: 'tax', header: 'Pajak (%)', cell: (props) => props.row.original.tax ?? '-', }, { accessorKey: 'expiry_period', header: 'Kadaluarsa (hari)', cell: (props) => props.row.original.expiry_period ?? '-', }, { accessorFn: (row) => row.suppliers?.map((s) => s.name).join(', ') || '-', header: 'Supplier', }, { accessorKey: 'flag', header: 'Flag', cell: (props) => props.row.original.flag ? props.row.original.flag : '-', }, { accessorFn: (row) => row.sub_flags?.length ? row.sub_flags.join(', ') : '-', header: 'Kategori Flags', }, { 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 = () => { setSelectedProduct(props.row.original); deleteModal.openModal(); }; return ( ); }, }, ], [tableFilterState.pageSize, tableFilterState.page, deleteModal] ); return ( <>
{/* Header Section */}
{/* Action Buttons */}
{/* Search */}
} 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(products) || products.data?.length === 0 ? (
} />
) : ( data={products?.data} columns={productsColumns} pageSize={tableFilterState.pageSize} page={products?.meta?.page ?? 0} totalItems={products?.meta?.total_results ?? 0} onPageChange={setPage} onPageSizeChange={setPageSize} isLoading={false} sorting={sorting} setSorting={setSorting} className={{ containerClassName: 'p-3 mb-0', headerColumnClassName: 'text-nowrap', }} /> )}
); }; export default ProductsTable;