refactor(FE-87-106): refactor api integration untuk project flock dan project flock kandang

This commit is contained in:
randy-ar
2025-11-10 04:08:08 +07:00
parent fcc2fced06
commit e0c347c3d5
19 changed files with 961 additions and 506 deletions
@@ -170,8 +170,8 @@ const DetailChickin = () => {
<div className='font-semibold text-sm'>Flock</div> <div className='font-semibold text-sm'>Flock</div>
<div className='text-sm'> <div className='text-sm'>
{ {
chickin.data.project_flock_kandang?.project_flock.flock chickin?.data?.project_flock_kandang?.project_flock?.flock
.name ?.name
} }
</div> </div>
</div> </div>
@@ -225,8 +225,8 @@ const DetailChickin = () => {
<div className='font-semibold text-sm'>Flock Kandang</div> <div className='font-semibold text-sm'>Flock Kandang</div>
<div className='text-sm'> <div className='text-sm'>
{ {
chickin.data.project_flock_kandang?.project_flock.flock chickin?.data?.project_flock_kandang?.project_flock?.flock
.name ?.name
}{' '} }{' '}
- {chickin.data.project_flock_kandang?.kandang.name} - {chickin.data.project_flock_kandang?.kandang.name}
</div> </div>
@@ -280,7 +280,7 @@ const DetailChickin = () => {
<ConfirmationModal <ConfirmationModal
ref={deleteModal.ref} ref={deleteModal.ref}
type='error' type='error'
text={`Apakah anda yakin ingin menghapus data Project Flock ini (${chickin?.data.project_flock_kandang?.project_flock.flock.name} - ${chickin?.data.project_flock_kandang?.kandang.name})?`} text={`Apakah anda yakin ingin menghapus data Project Flock ini (${chickin?.data?.project_flock_kandang?.project_flock.flock?.name} - ${chickin?.data.project_flock_kandang?.kandang.name})?`}
secondaryButton={{ secondaryButton={{
text: 'Tidak', text: 'Tidak',
}} }}
@@ -320,7 +320,7 @@ const DetailChickin = () => {
text={`Apakah anda yakin ingin ${ text={`Apakah anda yakin ingin ${
approvalAction == 'APPROVED' ? 'approve' : 'reject' approvalAction == 'APPROVED' ? 'approve' : 'reject'
} chickin berikut? (${ } chickin berikut? (${
chickin?.data.project_flock_kandang?.project_flock.flock.name chickin?.data?.project_flock_kandang?.project_flock?.flock?.name
} - ${chickin?.data.project_flock_kandang?.kandang.name})?`} } - ${chickin?.data.project_flock_kandang?.kandang.name})?`}
secondaryButton={{ secondaryButton={{
text: 'Tidak', text: 'Tidak',
@@ -2,8 +2,11 @@
import ProjectFlockForm from '@/components/pages/production/project-flock/form/ProjectFlockForm'; import ProjectFlockForm from '@/components/pages/production/project-flock/form/ProjectFlockForm';
import { isResponseError, isResponseSuccess } from '@/lib/api-helper'; import { isResponseError, isResponseSuccess } from '@/lib/api-helper';
import { FlockApi } from '@/services/api/master-data';
import { ProjectFlockApi } from '@/services/api/production'; import { ProjectFlockApi } from '@/services/api/production';
import { ProjectFlock } from '@/types/api/production/project-flock';
import { useRouter, useSearchParams } from 'next/navigation'; import { useRouter, useSearchParams } from 'next/navigation';
import { useState } from 'react';
import useSWR from 'swr'; import useSWR from 'swr';
const ProjectFlockDetail = () => { const ProjectFlockDetail = () => {
@@ -12,12 +15,21 @@ const ProjectFlockDetail = () => {
const projectFlockId = searchParams.get('projectFlockId'); const projectFlockId = searchParams.get('projectFlockId');
const [projectName, setProjectName] = useState();
const { const {
data: projectFlock, data: projectFlock,
isLoading: isLoadingProjectFlock, isLoading: isLoadingProjectFlock,
mutate: refreshProjectFlock, mutate: refreshProjectFlock,
} = useSWR(projectFlockId, (id: number) => ProjectFlockApi.getSingle(id)); } = useSWR(projectFlockId, (id: number) => ProjectFlockApi.getSingle(id));
const flockUrl = `${FlockApi.basePath}`;
const {
data: flock,
isLoading: isLoadingFlock,
mutate: refreshFlock,
} = useSWR(flockUrl, FlockApi.getAllFetcher);
if (!projectFlockId) { if (!projectFlockId) {
router.back(); router.back();
@@ -36,15 +48,37 @@ const ProjectFlockDetail = () => {
return; return;
} }
// Attach flock id to project flock
let projectFlockAttached: ProjectFlock | undefined;
if (isResponseSuccess(projectFlock) && isResponseSuccess(flock)) {
projectFlockAttached = {
...projectFlock.data,
flock: flock.data.find(
(flock) =>
flock.name ==
projectFlock.data.flock_name
.trim()
.split(/\s+/)
.slice(0, -1)
.join(' ')
),
};
console.log('projectFlockAttached');
console.log(projectFlockAttached);
console.log('flocks');
console.log(flock.data);
}
return ( return (
<div className='w-full p-4 flex flex-row justify-center'> <div className='w-full p-4 flex flex-row justify-center'>
{isLoadingProjectFlock && ( {isLoadingProjectFlock && (
<span className='loading loading-spinner loading-xl' /> <span className='loading loading-spinner loading-xl' />
)} )}
{!isLoadingProjectFlock && isResponseSuccess(projectFlock) && ( {projectFlockAttached && (
<ProjectFlockForm <ProjectFlockForm
formType='detail' formType='detail'
initialValues={projectFlock.data} initialValues={projectFlockAttached}
refreshProjectFlocks={refreshProjectFlock} refreshProjectFlocks={refreshProjectFlock}
/> />
)} )}
@@ -3,7 +3,7 @@
import Button from '@/components/Button'; import Button from '@/components/Button';
import CheckboxInput from '@/components/input/CheckboxInput'; import CheckboxInput from '@/components/input/CheckboxInput';
import { OptionType } from '@/components/input/SelectInput'; import { OptionType } from '@/components/input/SelectInput';
import { useModal } from '@/components/Modal'; import Modal, { useModal } from '@/components/Modal';
import ConfirmationModal from '@/components/modal/ConfirmationModal'; import ConfirmationModal from '@/components/modal/ConfirmationModal';
import Table from '@/components/Table'; import Table from '@/components/Table';
import RowCollapseOptions from '@/components/table/RowCollapseOptions'; import RowCollapseOptions from '@/components/table/RowCollapseOptions';
@@ -12,7 +12,7 @@ import { TableRowSizeSelector } from '@/components/table/TableRowSizeSelector';
import { TableToolbar } from '@/components/table/TableToolbar'; import { TableToolbar } from '@/components/table/TableToolbar';
import { ROWS_OPTIONS } from '@/config/constant'; import { ROWS_OPTIONS } from '@/config/constant';
import { isResponseSuccess } from '@/lib/api-helper'; import { isResponseSuccess } from '@/lib/api-helper';
import { cn } from '@/lib/helper'; import { cn, formatCurrency, formatVechicleNumber } from '@/lib/helper';
import { MarketingApi } from '@/services/api/marketing/marketing'; import { MarketingApi } from '@/services/api/marketing/marketing';
import { useTableFilter } from '@/services/hooks/useTableFilter'; import { useTableFilter } from '@/services/hooks/useTableFilter';
import { Marketing, MarketingProduct } from '@/types/api/marketing/marketing'; import { Marketing, MarketingProduct } from '@/types/api/marketing/marketing';
@@ -83,6 +83,7 @@ const SalesOrderTable = () => {
const [approveAction, setApproveAction] = useState< const [approveAction, setApproveAction] = useState<
'approve' | 'reject' | null 'approve' | 'reject' | null
>(null); >(null);
const [selectedItem, setSelectedItem] = useState<Marketing | null>(null);
const [rowSelection, setRowSelection] = useState<Record<string, boolean>>({}); const [rowSelection, setRowSelection] = useState<Record<string, boolean>>({});
const selectedRowIds = Object.keys(rowSelection).filter( const selectedRowIds = Object.keys(rowSelection).filter(
(id) => rowSelection[id] (id) => rowSelection[id]
@@ -96,6 +97,7 @@ const SalesOrderTable = () => {
const deleteModal = useModal(); const deleteModal = useModal();
const confirmationModal = useModal(); const confirmationModal = useModal();
const productsModal = useModal();
const searchChangeHandler = useCallback( const searchChangeHandler = useCallback(
(e: React.ChangeEvent<HTMLInputElement>) => { (e: React.ChangeEvent<HTMLInputElement>) => {
@@ -123,6 +125,11 @@ const SalesOrderTable = () => {
confirmationModal.openModal(); confirmationModal.openModal();
}; };
const productsClickHandler = (item: Marketing) => {
setSelectedItem(item);
productsModal.openModal();
};
const { const {
state: tableFilterState, state: tableFilterState,
updateFilter, updateFilter,
@@ -162,6 +169,7 @@ const SalesOrderTable = () => {
color='success' color='success'
onClick={approveClickHandler} onClick={approveClickHandler}
className='justify-start text-sm' className='justify-start text-sm'
disabled={!selectedRowIds.length}
> >
<Icon icon='material-symbols:check' width={24} height={24} /> <Icon icon='material-symbols:check' width={24} height={24} />
Approve Approve
@@ -171,6 +179,7 @@ const SalesOrderTable = () => {
color='error' color='error'
onClick={rejectClickHandler} onClick={rejectClickHandler}
className='justify-start text-sm' className='justify-start text-sm'
disabled={!selectedRowIds.length}
> >
<Icon icon='material-symbols:close' width={24} height={24} /> <Icon icon='material-symbols:close' width={24} height={24} />
Reject Reject
@@ -229,16 +238,28 @@ const SalesOrderTable = () => {
{ {
accessorKey: 'marketing_products.length', accessorKey: 'marketing_products.length',
header: 'Product Details', header: 'Product Details',
cell: (props) => ( cell: (props) => {
<ul className='list-disc list-inside'> if (props?.row?.original?.marketing_products?.length) {
{props.row.original.marketing_products?.map((product) => ( if (props?.row?.original?.marketing_products?.length > 1) {
<li key={product.id}> return (
{product.product_warehouse.product.name} - Qty:{' '} <Button
{product.qty} variant='link'
</li> color='success'
))} className='p-0 text-none'
</ul> onClick={() => {
), productsClickHandler(props?.row?.original);
}}
>
Lihat {props?.row?.original?.marketing_products?.length}{' '}
Produk
</Button>
);
} else {
const product = props?.row?.original?.marketing_products[0];
return <>{product?.product_warehouse?.product?.name}</>;
}
}
},
}, },
{ {
header: 'Aksi', header: 'Aksi',
@@ -321,6 +342,64 @@ const SalesOrderTable = () => {
color: approveAction === 'approve' ? 'success' : 'error', color: approveAction === 'approve' ? 'success' : 'error',
}} }}
/> />
<Modal
ref={productsModal.ref}
className={{
modalBox: 'max-w-2/5 z-100',
}}
closeOnBackdrop
>
<div className='flex flex-row justify-between items-center mb-3'>
<h4 className='text-xl font-semibold'>Daftar Produk</h4>
<Button
variant='ghost'
color='error'
onClick={productsModal.closeModal}
className='justify-start text-sm rounded-full'
>
<Icon icon='mdi:close' width={16} height={16} />
</Button>
</div>
<Table<MarketingProduct>
data={
isResponseSuccess(marketing) && selectedItem
? (selectedItem?.marketing_products ?? [])
: []
}
columns={[
{
header: 'Kandang',
accessorFn(row) {
return row.product_warehouse.warehouse.name;
},
},
{
header: 'Produk',
accessorFn(row) {
return row.product_warehouse.product.name;
},
},
{
header: 'Harga Satuan (Rp)',
accessorFn(row) {
return formatCurrency(row.unit_price);
},
},
]}
className={{
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',
paginationClassName: 'hidden',
}}
/>
</Modal>
</> </>
); );
}; };
@@ -6,7 +6,6 @@ import { FormHeader } from '@/components/helper/form/FormHeader';
import { useModal } from '@/components/Modal'; import { useModal } from '@/components/Modal';
import ConfirmationModal from '@/components/modal/ConfirmationModal'; import ConfirmationModal from '@/components/modal/ConfirmationModal';
import Table from '@/components/Table'; import Table from '@/components/Table';
import { isResponseSuccess } from '@/lib/api-helper';
import { import {
cn, cn,
formatCurrency, formatCurrency,
@@ -33,6 +32,7 @@ const SalesOrderDetail = ({
const deleteModal = useModal(); const deleteModal = useModal();
const confirmationModal = useModal(); const confirmationModal = useModal();
const deliveryModal = useModal();
const approveClickHandler = () => { const approveClickHandler = () => {
setApprovalAction('approve'); setApprovalAction('approve');
@@ -44,6 +44,10 @@ const SalesOrderDetail = ({
confirmationModal.openModal(); confirmationModal.openModal();
}; };
const deliveryClickHandler = () => {
deliveryModal.openModal();
};
const deleteClickHandler = () => { const deleteClickHandler = () => {
deleteModal.openModal(); deleteModal.openModal();
}; };
@@ -59,16 +63,25 @@ const SalesOrderDetail = ({
const confirmationModalApproveClickHandler = async () => { const confirmationModalApproveClickHandler = async () => {
setIsLoading(true); setIsLoading(true);
await MarketingApi.singleApproval( // await MarketingApi.singleApproval(
initialValues?.id as number, // initialValues?.id as number,
approvalAction // approvalAction
); // );
setIsLoading(false); setIsLoading(false);
confirmationModal.closeModal(); confirmationModal.closeModal();
toast.success('Successfully approved Sales Order!'); toast.success('Successfully approved Sales Order!');
refreshValues?.(); refreshValues?.();
}; };
const confirmationModalDeliveryClickHandler = async () => {
setIsLoading(true);
// await MarketingApi.delivery(initialValues?.id as number);
setIsLoading(false);
deliveryModal.closeModal();
toast.success('Successfully delivered Sales Order!');
refreshValues?.();
};
return ( return (
<> <>
<div className='flex flex-col w-full gap-4'> <div className='flex flex-col w-full gap-4'>
@@ -77,14 +90,32 @@ const SalesOrderDetail = ({
backUrl='/marketing/sales-orders' backUrl='/marketing/sales-orders'
/> />
<div className='flex-row flex gap-3'> <div className='flex-row flex gap-3'>
<Button color='success' onClick={approveClickHandler}> {initialValues?.approval?.step_number != 3 && (
<>
<Button
color='success'
onClick={approveClickHandler}
disabled={initialValues?.approval?.step_number != 1}
>
<Icon icon='mdi:check' width={24} height={24} /> <Icon icon='mdi:check' width={24} height={24} />
Approve Approve
</Button> </Button>
<Button color='error' onClick={rejectClickHandler}> <Button
color='error'
onClick={rejectClickHandler}
disabled={initialValues?.approval?.step_number != 2}
>
<Icon icon='mdi:close' width={24} height={24} /> <Icon icon='mdi:close' width={24} height={24} />
Reject Reject
</Button> </Button>
</>
)}
{initialValues?.approval?.step_number == 2 && (
<Button color='success' onClick={deliveryClickHandler}>
<Icon icon='mdi:check' width={24} height={24} />
Delivery Order
</Button>
)}
</div> </div>
<Card <Card
title='Informasi Sales Order' title='Informasi Sales Order'
@@ -110,7 +141,7 @@ const SalesOrderDetail = ({
<tr> <tr>
<td className='font-semibold'>Status</td> <td className='font-semibold'>Status</td>
<td>:</td> <td>:</td>
<td>{initialValues?.status}</td> <td>{initialValues?.approval?.step_name}</td>
</tr> </tr>
<tr> <tr>
<td className='font-semibold'>Tanggal Penjualan</td> <td className='font-semibold'>Tanggal Penjualan</td>
@@ -214,7 +245,11 @@ const SalesOrderDetail = ({
</Card> </Card>
)} )}
<div className='flex flex-row gap-3'> <div className='flex flex-row gap-3'>
<Button color='warning'> <Button
color='warning'
type='button'
href={`/marketing/sales-orders/detail/edit?salesOrderId=${initialValues?.id}`}
>
<Icon icon='mdi:pencil' width={24} height={24} /> <Icon icon='mdi:pencil' width={24} height={24} />
Edit Edit
</Button> </Button>
@@ -235,12 +270,13 @@ const SalesOrderDetail = ({
text: 'Ya', text: 'Ya',
color: 'error', color: 'error',
isLoading: isLoading, isLoading: isLoading,
onClick: confirmationModalDeleteClickHandler,
}} }}
/> />
<ConfirmationModal <ConfirmationModal
ref={confirmationModal.ref} ref={confirmationModal.ref}
type={approvalAction === 'approve' ? 'success' : 'error'} type={approvalAction === 'approve' ? 'success' : 'error'}
text={`Apakah anda yakin ingin ${approvalAction} data penjualan (${initialValues?.id})?`} text={`Apakah anda yakin ingin ${approvalAction} data penjualan ini?`}
secondaryButton={{ secondaryButton={{
text: 'Tidak', text: 'Tidak',
}} }}
@@ -251,6 +287,20 @@ const SalesOrderDetail = ({
onClick: confirmationModalApproveClickHandler, onClick: confirmationModalApproveClickHandler,
}} }}
/> />
<ConfirmationModal
ref={deliveryModal.ref}
type={'success'}
text={`Apakah anda yakin ingin deliver penjualan ini?`}
secondaryButton={{
text: 'Tidak',
}}
primaryButton={{
text: 'Ya',
color: 'success',
isLoading: isLoading,
onClick: confirmationModalDeliveryClickHandler,
}}
/>
</> </>
); );
}; };
@@ -10,7 +10,8 @@ import SelectInput, {
} from '@/components/input/SelectInput'; } from '@/components/input/SelectInput';
import TextArea from '@/components/input/TextArea'; import TextArea from '@/components/input/TextArea';
import Modal, { useModal } from '@/components/Modal'; import Modal, { useModal } from '@/components/Modal';
import Table from '@/components/Table'; import * as TanStack from '@tanstack/react-table';
import Table from '@/components/Table'; // Keep this import
import { cn, formatCurrency, formatNumber } from '@/lib/helper'; import { cn, formatCurrency, formatNumber } from '@/lib/helper';
import { import {
CreateMarketingPayload, CreateMarketingPayload,
@@ -19,7 +20,7 @@ import {
MarketingProduct, MarketingProduct,
} from '@/types/api/marketing/marketing'; } from '@/types/api/marketing/marketing';
import { Icon } from '@iconify/react'; import { Icon } from '@iconify/react';
import { useEffect, useState } from 'react'; import { useCallback, useEffect, useMemo, useState } from 'react';
import MarketingProductForm from './repeater/MarketingProductForm'; import MarketingProductForm from './repeater/MarketingProductForm';
import CheckboxInput from '@/components/input/CheckboxInput'; import CheckboxInput from '@/components/input/CheckboxInput';
import { Customer } from '@/types/api/master-data/customer'; import { Customer } from '@/types/api/master-data/customer';
@@ -29,6 +30,9 @@ import { MarketingFormValues, MarketingSchema } from './SalesForm.schema';
import { isResponseError, isResponseSuccess } from '@/lib/api-helper'; import { isResponseError, isResponseSuccess } from '@/lib/api-helper';
import { MarketingApi } from '@/services/api/marketing/marketing'; import { MarketingApi } from '@/services/api/marketing/marketing';
import { MarketingProductFormValues } from './repeater/MarketingProduct.schema'; import { MarketingProductFormValues } from './repeater/MarketingProduct.schema';
import ConfirmationModal from '@/components/modal/ConfirmationModal';
import toast from 'react-hot-toast';
import { useRouter } from 'next/navigation';
const SalesForm = ({ const SalesForm = ({
formType = 'add', formType = 'add',
@@ -37,11 +41,14 @@ const SalesForm = ({
formType?: 'add' | 'edit'; formType?: 'add' | 'edit';
initialValues?: Marketing; initialValues?: Marketing;
}) => { }) => {
const router = useRouter();
const addProductModal = useModal(); const addProductModal = useModal();
const deleteModal = useModal();
const [isLoading, setIsLoading] = useState(false);
const [selectedMarketingProduct, setSelectedMarketingProduct] = const [selectedMarketingProduct, setSelectedMarketingProduct] =
useState<MarketingProduct | null>(null); useState<MarketingProduct | null>(null);
const [marketingProducts, setMarketingProducts] = useState< const [rawMarketingProducts, setRawMarketingProducts] = useState<
MarketingProduct[] MarketingProduct[]
>(initialValues?.marketing_products || []); >(initialValues?.marketing_products || []);
const [selectedCustomer, setSelectedCustomer] = useState<OptionType | null>( const [selectedCustomer, setSelectedCustomer] = useState<OptionType | null>(
@@ -53,7 +60,13 @@ const SalesForm = ({
const selectedRowIds = Object.keys(rowSelection).map((item) => const selectedRowIds = Object.keys(rowSelection).map((item) =>
parseInt(item) parseInt(item)
); );
const [grandTotal, setGrandTotal] = useState<number>(0); const [grandTotal, setGrandTotal] = useState<number>(
initialValues?.grand_total ?? 0
);
const marketingProducts = useMemo(
() => rawMarketingProducts,
[rawMarketingProducts]
);
const { const {
options: customerOptions, options: customerOptions,
@@ -61,23 +74,28 @@ const SalesForm = ({
isLoadingOptions: isLoadingCustomerOptions, isLoadingOptions: isLoadingCustomerOptions,
} = useSelect<Customer>(CustomerApi.basePath, 'id', 'name'); } = useSelect<Customer>(CustomerApi.basePath, 'id', 'name');
const handleAddProduct = () => { const handleAddProduct = useCallback(() => {
addProductModal.openModal(); addProductModal.openModal();
}; }, [addProductModal]);
const handleDeleteProduct = (id: number) => { const handleDeleteProduct = useCallback((id: number) => {
setMarketingProducts((prev) => prev.filter((product) => product.id !== id)); setRawMarketingProducts((prev) => prev.filter((p) => p.id !== id));
}; }, []);
const handleBulkDeleteProduct = () => { const handleBulkDeleteProduct = () => {
setMarketingProducts((prev) => setRawMarketingProducts((prev) =>
prev.filter((product) => !selectedRowIds.includes(product.id)) prev.filter((product) => !selectedRowIds.includes(product.id))
); );
}; };
const handleAddSubmitProduct = async ( const handleDelete = () => {
deleteModal.openModal();
};
const handleAddSubmitProduct = useCallback(
async (
tableValue: CreateMarketingProductPayload, tableValue: CreateMarketingProductPayload,
fieldValues: MarketingProductFormValues fieldValues: MarketingProductFormValues
) => { ) => {
const newMarketingProduct: MarketingProduct = { const newMarketingProduct: MarketingProduct = {
id: marketingProducts.length + 1, id: rawMarketingProducts.length + 1,
product_warehouse: tableValue.product_warehouse!, product_warehouse: tableValue.product_warehouse!,
unit_price: tableValue.unit_price as number, unit_price: tableValue.unit_price as number,
total_weight: tableValue.total_weight as number, total_weight: tableValue.total_weight as number,
@@ -85,7 +103,7 @@ const SalesForm = ({
avg_weight: tableValue.avg_weight as number, avg_weight: tableValue.avg_weight as number,
total_price: tableValue.total_price as number, total_price: tableValue.total_price as number,
marketing_delivery_products: { marketing_delivery_products: {
id: marketingProducts.length + 1, id: rawMarketingProducts.length + 1,
vehicle_number: tableValue.vehicle_number as string, vehicle_number: tableValue.vehicle_number as string,
delivery_date: tableValue.delivery_date as string, delivery_date: tableValue.delivery_date as string,
unit_price: tableValue.unit_price as number, unit_price: tableValue.unit_price as number,
@@ -96,20 +114,26 @@ const SalesForm = ({
}, },
}; };
setMarketingProducts((prev) => [...prev, newMarketingProduct]); setRawMarketingProducts((prev) => [...prev, newMarketingProduct]);
formik.setValues({ formik.setValues({
...formik.values, ...formik.values,
marketing_products: [...formik.values.marketing_products, fieldValues], marketing_products: [...formik.values.marketing_products, fieldValues],
}); });
setGrandTotal((prev) => prev + (tableValue.total_price as number)); setGrandTotal((prev) => prev + (tableValue.total_price as number));
addProductModal.closeModal(); addProductModal.closeModal();
}; },
const handleChangeCustomer = (val: OptionType | OptionType[] | null) => { [rawMarketingProducts.length, addProductModal]
);
const handleChangeCustomer = useCallback(
(val: OptionType | OptionType[] | null) => {
setSelectedCustomer(val as OptionType); setSelectedCustomer(val as OptionType);
formik.setFieldValue('customer_id', (val as OptionType)?.value); formik.setFieldValue('customer_id', (val as OptionType)?.value);
}; formik.setFieldValue('customer', val as OptionType);
},
[selectedCustomer, setSelectedCustomer]
);
const createProjectFlockHandler = async (values: CreateMarketingPayload) => { const createMarketingHandler = async (values: CreateMarketingPayload) => {
console.log(values); console.log(values);
const createMarketingRes = await MarketingApi.create(values); const createMarketingRes = await MarketingApi.create(values);
if (isResponseSuccess(createMarketingRes)) { if (isResponseSuccess(createMarketingRes)) {
@@ -118,8 +142,10 @@ const SalesForm = ({
if (isResponseError(createMarketingRes)) { if (isResponseError(createMarketingRes)) {
console.log(createMarketingRes); console.log(createMarketingRes);
} }
toast.success('Successfully created Sales Order!');
router.push('/marketing/sales-orders');
}; };
const updateProjectFlockHandler = async (values: CreateMarketingPayload) => { const updateMarketingHandler = async (values: CreateMarketingPayload) => {
console.log(values); console.log(values);
const createMarketingRes = await MarketingApi.update( const createMarketingRes = await MarketingApi.update(
initialValues?.id as number, initialValues?.id as number,
@@ -131,6 +157,50 @@ const SalesForm = ({
if (isResponseError(createMarketingRes)) { if (isResponseError(createMarketingRes)) {
console.log(createMarketingRes); console.log(createMarketingRes);
} }
toast.success('Successfully updated Sales Order!');
router.push('/marketing/sales-orders');
};
const deleteMarketingHandler = async () => {
setIsLoading(true);
console.log(initialValues?.id);
const deleteMarketingRes = await MarketingApi.delete(
initialValues?.id as number
);
if (isResponseSuccess(deleteMarketingRes)) {
console.log(deleteMarketingRes);
}
if (isResponseError(deleteMarketingRes)) {
console.log(deleteMarketingRes);
}
toast.success('Successfully deleted Sales Order!');
setIsLoading(false);
deleteModal.closeModal();
router.push('/marketing/sales-orders');
};
const MarketingProductToFieldValues = (
product: MarketingProduct
): MarketingProductFormValues => {
return {
vehicle_number: product.marketing_delivery_products?.vehicle_number,
kandang_id: product.product_warehouse.warehouse.id,
kandang: {
value: product.product_warehouse.warehouse.id,
label: product.product_warehouse.warehouse.name,
},
product_warehouse: {
value: product.product_warehouse.product.id,
label: product.product_warehouse.product.name,
},
product_warehouse_id: product.product_warehouse.product.id,
unit_price: product.unit_price,
total_weight: product.total_weight,
qty: product.qty,
uom: product.product_warehouse?.product?.uom?.name,
avg_weight: product.avg_weight,
total_price: product.total_price,
delivery_date: product.marketing_delivery_products?.delivery_date,
};
}; };
const formik = useFormik<MarketingFormValues>({ const formik = useFormik<MarketingFormValues>({
@@ -143,7 +213,10 @@ const SalesForm = ({
value: initialValues?.customer?.id as number, value: initialValues?.customer?.id as number,
label: initialValues?.customer?.name as string, label: initialValues?.customer?.name as string,
}, },
marketing_products: [], marketing_products:
initialValues?.marketing_products?.map((product) =>
MarketingProductToFieldValues(product)
) ?? [],
}, },
validationSchema: MarketingSchema, validationSchema: MarketingSchema,
onSubmit: async (values) => { onSubmit: async (values) => {
@@ -155,10 +228,10 @@ const SalesForm = ({
} as CreateMarketingPayload; } as CreateMarketingPayload;
switch (formType) { switch (formType) {
case 'add': case 'add':
createProjectFlockHandler(payload); createMarketingHandler(payload);
break; break;
case 'edit': case 'edit':
updateProjectFlockHandler(payload); updateMarketingHandler(payload);
break; break;
default: default:
break; break;
@@ -172,6 +245,85 @@ const SalesForm = ({
formikSetValues(formik.initialValues); formikSetValues(formik.initialValues);
}, [formikSetValues, formik.initialValues]); }, [formikSetValues, formik.initialValues]);
const columns = useMemo(
() => [
{
id: 'select',
header: ({ table }: { table: TanStack.Table<MarketingProduct> }) => (
<div className='w-full flex flex-row justify-center'>
<CheckboxInput
name='allRow'
checked={table.getIsAllRowsSelected()}
indeterminate={table.getIsSomeRowsSelected()}
onChange={table.getToggleAllRowsSelectedHandler()}
/>
</div>
),
cell: ({ row }: { row: TanStack.Row<MarketingProduct> }) => (
<div>
<CheckboxInput
name='row'
checked={row.getIsSelected()}
disabled={!row.getCanSelect()}
indeterminate={row.getIsSomeSelected()}
onChange={row.getToggleSelectedHandler()}
/>
</div>
),
},
{
accessorFn: (row: MarketingProduct) =>
row.marketing_delivery_products?.vehicle_number,
header: 'No. Polisi',
},
{
accessorFn: (row: MarketingProduct) =>
row.product_warehouse.warehouse.name,
header: 'Kandang',
},
{
accessorFn: (row: MarketingProduct) =>
row.product_warehouse.product.name,
header: 'Produk',
},
{
accessorFn: (row: MarketingProduct) => formatCurrency(row.unit_price),
header: 'Harga Satuan (Rp)',
},
{
accessorFn: (row: MarketingProduct) => formatNumber(row.total_weight),
header: 'Total Bobot (Kg)',
},
{
accessorFn: (row: MarketingProduct) => formatNumber(row.qty),
header: 'Kuantitas',
},
{
accessorFn: (row: MarketingProduct) => formatNumber(row.avg_weight),
header: 'Avg. Bobot (Kg)',
},
{
accessorFn: (row: MarketingProduct) => formatCurrency(row.total_price),
header: 'Total Penjualan (Rp)',
},
{
header: 'Aksi',
cell: (props: TanStack.CellContext<MarketingProduct, unknown>) => (
<div className='flex flex-row gap-1 items-center justify-end h-full mt-2'>
<Button
color='error'
className='p-1'
onClick={() => handleDeleteProduct(props.row.original.id)}
>
<Icon icon='mdi:trash' width={16} height={16} />
</Button>
</div>
),
},
],
[handleDeleteProduct] // dependensi tunggal
);
return ( return (
<> <>
<form <form
@@ -224,98 +376,7 @@ const SalesForm = ({
rowSelection={rowSelection} rowSelection={rowSelection}
setRowSelection={setRowSelection} setRowSelection={setRowSelection}
data={marketingProducts} data={marketingProducts}
columns={[ columns={columns}
{
id: 'select',
header: ({ table }) => (
<div className='w-full flex flex-row justify-center'>
<CheckboxInput
name='allRow'
checked={table.getIsAllRowsSelected()}
indeterminate={table.getIsSomeRowsSelected()}
onChange={table.getToggleAllRowsSelectedHandler()}
/>
</div>
),
cell: ({ row }) => (
<div>
<CheckboxInput
name='row'
checked={row.getIsSelected()}
disabled={!row.getCanSelect()}
indeterminate={row.getIsSomeSelected()}
onChange={row.getToggleSelectedHandler()}
/>
</div>
),
},
{
accessorFn(row) {
return row.marketing_delivery_products?.vehicle_number;
},
header: 'No. Polisi',
},
{
accessorFn(row) {
return row.product_warehouse.warehouse.name;
},
header: 'Kandang',
},
{
accessorFn(row) {
return row.product_warehouse.product.name;
},
header: 'Produk',
},
{
accessorFn(row) {
return formatCurrency(row.unit_price);
},
header: 'Harga Satuan (Rp)',
},
{
accessorFn(row) {
return formatNumber(row.total_weight);
},
header: 'Total Bobot (Kg)',
},
{
accessorFn(row) {
return formatNumber(row.qty);
},
header: 'Kuantitas',
},
{
accessorFn(row) {
return formatNumber(row.avg_weight);
},
header: 'Avg. Bobot (Kg)',
},
{
accessorFn(row) {
return formatCurrency(row.total_price);
},
header: 'Total Penjualan (Rp)',
},
{
header: 'Aksi',
cell: (props) => {
return (
<div className='flex flex-row gap-1 items-center justify-end h-full mt-2'>
<Button
color='error'
className='p-1'
onClick={() => {
handleDeleteProduct(props.row.original.id);
}}
>
<Icon icon='mdi:trash' width={16} height={16} />
</Button>
</div>
);
},
},
]}
className={{ className={{
tableWrapperClassName: 'overflow-x-auto min-h-full!', tableWrapperClassName: 'overflow-x-auto min-h-full!',
tableClassName: 'font-inter w-full table-auto min-h-full!', tableClassName: 'font-inter w-full table-auto min-h-full!',
@@ -395,7 +456,16 @@ const SalesForm = ({
Submit Submit
</Button> </Button>
</div> </div>
{JSON.stringify(formik.errors)}
</form> </form>
{formType == 'edit' && (
<div className='flex flex-row justify-start'>
<Button type='button' color='error' onClick={handleDelete}>
<Icon icon='mdi:trash' width={24} height={24} />
Hapus
</Button>
</div>
)}
<Modal <Modal
ref={addProductModal.ref} ref={addProductModal.ref}
closeOnBackdrop closeOnBackdrop
@@ -420,12 +490,25 @@ const SalesForm = ({
<MarketingProductForm <MarketingProductForm
onSubmitForm={handleAddSubmitProduct} onSubmitForm={handleAddSubmitProduct}
modalRef={addProductModal.ref} modalRef={addProductModal.ref}
data={marketingProducts} data={rawMarketingProducts}
initialValues={selectedMarketingProduct ?? undefined} initialValues={selectedMarketingProduct ?? undefined}
/> />
</div> </div>
</div> </div>
</Modal> </Modal>
<ConfirmationModal
ref={deleteModal.ref}
type='error'
text={`Apakah anda yakin ingin menghapus data penjualan ini?`}
secondaryButton={{
text: 'Tidak',
}}
primaryButton={{
text: 'Ya',
color: 'error',
onClick: deleteMarketingHandler,
}}
/>
</> </>
); );
}; };
@@ -1,3 +1,37 @@
import * as Yup from 'yup'; import * as Yup from 'yup';
import { Product } from '@/types/api/master-data/product';
import { Supplier } from '@/types/api/master-data/supplier'; type ChickinRequestSchemaType = {
chick_in_date: string;
note?: string | undefined | null;
product_warehouse_id: number;
};
type ChickinSchemaType = {
project_flock_kandang_id: number;
chickin_requests: ChickinRequestSchemaType[];
};
export const ChickinRequestSchema: Yup.ObjectSchema<ChickinRequestSchemaType> =
Yup.object({
chick_in_date: Yup.string().nullable().required('Tanggal wajib diisi!'),
note: Yup.string().nullable(),
product_warehouse_id: Yup.number()
.min(1, 'Produk wajib diisi!')
.required('Produk wajib diisi!'),
});
export const ChickinSchema: Yup.ObjectSchema<ChickinSchemaType> = Yup.object({
project_flock_kandang_id: Yup.number()
.min(1, 'Project Flock Kandang wajib diisi!')
.required('Project Flock Kandang wajib diisi!'),
chickin_requests: Yup.array()
.of(ChickinRequestSchema)
.min(1, 'Minimal harus ada 1 produk!')
.required('Produk wajib diisi!'),
});
export type ChickinRequestFormValues = Yup.InferType<
typeof ChickinRequestSchema
>;
export type ChickinFormValues = Yup.InferType<typeof ChickinSchema>;
@@ -6,7 +6,7 @@ import { FormHeader } from '@/components/helper/form/FormHeader';
import DateInput from '@/components/input/DateInput'; import DateInput from '@/components/input/DateInput';
import FileInput from '@/components/input/FileInput'; import FileInput from '@/components/input/FileInput';
import NumberInput from '@/components/input/NumberInput'; import NumberInput from '@/components/input/NumberInput';
import SelectInput from '@/components/input/SelectInput'; import SelectInput, { OptionType } from '@/components/input/SelectInput';
import TextInput from '@/components/input/TextInput'; import TextInput from '@/components/input/TextInput';
import Table from '@/components/Table'; import Table from '@/components/Table';
import { formatNumber } from '@/lib/helper'; import { formatNumber } from '@/lib/helper';
@@ -15,8 +15,19 @@ import {
AvailableQty, AvailableQty,
ProjectFlockKandang, ProjectFlockKandang,
} from '@/types/api/production/project-flock-kandang'; } from '@/types/api/production/project-flock-kandang';
import { useFormik } from 'formik';
import {
ChickinFormValues,
ChickinRequestFormValues,
ChickinSchema,
} from './ChickinForm.schema';
import { useCallback, useEffect, useState } from 'react';
import { useRouter } from 'next/navigation'; import { useRouter } from 'next/navigation';
import { CreateChickinPayload } from '@/types/api/production/chickin';
import { ChickinApi } from '@/services/api/production';
import { isResponseError } from '@/lib/api-helper';
import toast from 'react-hot-toast';
import { flushSync } from 'react-dom';
const ChickinFormKandang = ({ const ChickinFormKandang = ({
formType = 'add', formType = 'add',
initialValues, initialValues,
@@ -27,13 +38,104 @@ const ChickinFormKandang = ({
afterSubmit?: () => void; afterSubmit?: () => void;
}) => { }) => {
const router = useRouter(); const router = useRouter();
const [chickinErrorMessage, setChickinErrorMessage] = useState('');
const createChickin = useCallback(
async (payload: CreateChickinPayload) => {
const createChickinRes = await ChickinApi.create(payload);
if (isResponseError(createChickinRes)) {
setChickinErrorMessage(createChickinRes.message);
return;
}
toast.success(createChickinRes?.message as string);
router.push(
`/production/project-flock/chickin/add?projectFlockId=${initialValues?.project_flock?.id}`
);
},
[router]
);
const handleReset = async () => {
flushSync(() => {
formik.resetForm({
values: {
project_flock_kandang_id: initialValues?.id,
chickin_requests: initialValues?.available_qtys
? initialValues.available_qtys.map((availableQty) => ({
chick_in_date: '',
product_warehouse_id: availableQty.product_warehouse.id,
available_qty: availableQty.available_qty,
note: `Chickin project-flock-kandang-${initialValues?.id} product-warehouse-${availableQty.product_warehouse.id}`,
}))
: [],
},
});
});
formik.setTouched({
chickin_requests: initialValues?.available_qtys?.map(() => ({
chick_in_date: true,
})),
});
const errors = await formik.validateForm();
formik.setErrors(errors);
};
const formik = useFormik<ChickinFormValues>({
enableReinitialize: true,
validationSchema: ChickinSchema,
initialValues: {
project_flock_kandang_id: initialValues?.id,
chickin_requests: initialValues?.available_qtys
? initialValues.available_qtys.map((availableQty) => ({
chick_in_date: '',
product_warehouse_id: availableQty.product_warehouse.id,
available_qty: availableQty.available_qty,
note: `Chickin project-flock-kandang-${initialValues?.id} product-warehouse-${availableQty.product_warehouse.id}`,
}))
: [],
},
onSubmit: (values) => {
setChickinErrorMessage('');
createChickin(values as CreateChickinPayload);
if (afterSubmit) {
afterSubmit();
}
},
});
const { setValues: formikSetValues } = formik;
useEffect(() => {
formikSetValues({
project_flock_kandang_id: initialValues?.id,
chickin_requests: initialValues?.available_qtys
? initialValues.available_qtys.map((availableQty) => ({
chick_in_date: '',
product_warehouse_id: availableQty.product_warehouse.id,
available_qty: availableQty.available_qty,
note: `Chickin project-flock-kandang-${initialValues?.id} product-warehouse-${availableQty.product_warehouse.id}`,
}))
: [],
});
}, [formikSetValues, initialValues]);
return ( return (
<div className='flex flex-col gap-4'> <div className='flex flex-col gap-4'>
<FormHeader <FormHeader
type='add' type='add'
title='Chick In DOC' title='Chick In DOC'
backUrl={`/production/project-flock/chickin/add?projectFlockId=${initialValues.project_flock.id}`} backUrl={`/production/project-flock/chickin/add?projectFlockId=${initialValues?.project_flock?.id}`}
/> />
<form
className='flex flex-col gap-4'
onReset={(e) => {
handleReset();
}}
onSubmit={formik.handleSubmit}
>
<Card <Card
title='Informasi Kandang' title='Informasi Kandang'
className={{ className={{
@@ -48,20 +150,22 @@ const ChickinFormKandang = ({
</span> </span>
</div> </div>
} }
data={[initialValues.kandang]} data={[initialValues?.kandang]}
columns={[ columns={[
{ {
header: 'Area', header: 'Area',
accessorFn: () => initialValues.project_flock?.area.name || '-', accessorFn: () =>
initialValues?.project_flock?.area.name || '-',
}, },
{ {
header: 'Lokasi', header: 'Lokasi',
accessorFn: () => accessorFn: () =>
initialValues.project_flock?.location.name || '-', initialValues?.project_flock?.location.name || '-',
}, },
{ {
header: 'Flock', header: 'Flock',
accessorFn: () => initialValues.project_flock?.flock.name || '-', accessorFn: () =>
initialValues?.project_flock?.flock_name || '-',
}, },
{ {
header: 'Kandang', header: 'Kandang',
@@ -96,8 +200,8 @@ const ChickinFormKandang = ({
wrapper: 'w-full bg-white', wrapper: 'w-full bg-white',
}} }}
> >
<Table<AvailableQty> <Table<ChickinRequestFormValues>
data={initialValues.available_qtys || []} data={formik.values.chickin_requests || []}
columns={[ columns={[
{ {
accessorFn: (row) => row.chick_in_date, accessorFn: (row) => row.chick_in_date,
@@ -105,57 +209,32 @@ const ChickinFormKandang = ({
cell(props) { cell(props) {
return ( return (
<DateInput <DateInput
name='chick_in_date[]' name={`chickin_requests[${props.row.index}].chick_in_date`}
value={props.row.original.chick_in_date} value={
onChange={(e) => { formik.values.chickin_requests[props.row.index]
props.row.original.chick_in_date = e.target.value; ?.chick_in_date as string
}} }
onChange={formik.handleChange}
/> />
); );
}, },
}, },
{ {
accessorFn: (row) => row.po_number, accessorFn: (row) => row.product_warehouse_id,
header: 'No. Surat Jalan', header: 'Produk',
cell(props) { cell(props) {
return ( const availableQty = initialValues?.available_qtys?.find(
<TextInput (availableQty) =>
name='po_number[]' availableQty.product_warehouse.id ===
value={props.row.original.po_number} props.row.original.product_warehouse_id
onChange={(e) => {
props.row.original.po_number = e.target.value;
}}
/>
); );
},
},
{
header: 'Dokumen Surat Jalan',
cell(props) {
return (
<FileInput
name='document_path[]'
onChange={(e) => {
props.row.original.document_path = e.target.value;
}}
/>
);
},
},
{
accessorFn: (row) => row.supplier?.name,
header: 'Supplier',
cell(props) {
return ( return (
<SelectInput <SelectInput
value={ value={
props.row.original.supplier?.name && {
props.row.original.supplier?.id label: availableQty?.product_warehouse?.product?.name,
? { value: availableQty?.product_warehouse?.product?.id,
label: props.row.original.supplier.name, } as OptionType
value: props.row.original.supplier.id,
}
: undefined
} }
options={[]} options={[]}
isDisabled isDisabled
@@ -164,29 +243,18 @@ const ChickinFormKandang = ({
}, },
}, },
{ {
accessorFn: (row) => row.product_warehouse.product.name, accessorFn: (row) => row.product_warehouse_id,
header: 'Produk',
cell(props) {
return (
<SelectInput
value={{
label: props.row.original.product_warehouse.product.name,
value: props.row.original.product_warehouse.product.id,
}}
options={[]}
isDisabled
/>
);
},
},
{
accessorFn: (row) => row.product_warehouse.quantity,
header: 'Jumlah (ekor)', header: 'Jumlah (ekor)',
cell(props) { cell(props) {
const availableQty = initialValues?.available_qtys?.find(
(availableQty) =>
availableQty.product_warehouse.id ===
props.row.original.product_warehouse_id
);
return ( return (
<NumberInput <NumberInput
name='qty[]' name='qty[]'
value={props.row.original.product_warehouse.quantity} value={availableQty?.available_qty}
/> />
); );
}, },
@@ -212,14 +280,21 @@ const ChickinFormKandang = ({
} }
/> />
</Card> </Card>
{JSON.stringify(formik.values)}
<div className='flex flex-row justify-center gap-3'> <div className='flex flex-row justify-center gap-3'>
<Button type='reset' color='warning'> <Button type='reset' color='warning' disabled={formik.isSubmitting}>
Reset Reset
</Button> </Button>
<Button type='submit' color='primary'> <Button
type='submit'
color='primary'
disabled={!formik.isValid || formik.isSubmitting}
>
Submit Submit
</Button> </Button>
</div> </div>
{JSON.stringify(formik.errors)}
</form>
</div> </div>
); );
}; };
@@ -426,7 +426,7 @@ const ProjectFlockTable = () => {
}, },
{ {
accessorKey: 'flock.name', accessorKey: 'flock_name',
header: 'Flock', header: 'Flock',
}, },
{ {
@@ -3,15 +3,22 @@
import Badge from '@/components/Badge'; import Badge from '@/components/Badge';
import Button from '@/components/Button'; import Button from '@/components/Button';
import Card from '@/components/Card'; import Card from '@/components/Card';
import SelectInput, { OptionType } from '@/components/input/SelectInput'; import SelectInput, {
OptionType,
useSelect,
} from '@/components/input/SelectInput';
import PillBadge from '@/components/PillBadge'; import PillBadge from '@/components/PillBadge';
import Table from '@/components/Table'; import Table from '@/components/Table';
import { isResponseSuccess } from '@/lib/api-helper'; import { isResponseSuccess } from '@/lib/api-helper';
import { cn } from '@/lib/helper'; import { cn } from '@/lib/helper';
import { ProjectFlockApi } from '@/services/api/production'; import {
ProjectFlockApi,
ProjectFlockKandangApi,
} from '@/services/api/production';
import { useTableFilter } from '@/services/hooks/useTableFilter'; import { useTableFilter } from '@/services/hooks/useTableFilter';
import { Kandang } from '@/types/api/master-data/kandang'; import { Kandang } from '@/types/api/master-data/kandang';
import { ProjectFlock } from '@/types/api/production/project-flock'; import { ProjectFlock } from '@/types/api/production/project-flock';
import { ProjectFlockKandang } from '@/types/api/production/project-flock-kandang';
import { Icon } from '@iconify/react'; import { Icon } from '@iconify/react';
import { useRouter } from 'next/navigation'; import { useRouter } from 'next/navigation';
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
@@ -37,28 +44,30 @@ const ProjectFlockChickinDetail = ({
const [projectFlock, setProjectFlock] = useState<ProjectFlock>(); const [projectFlock, setProjectFlock] = useState<ProjectFlock>();
// Fetch Data // Fetch Data
const { data: listProjectFlock, isLoading: isLoadingListProjectFlock } = const {
useSWR( data: listProjectFlockKandang,
`${ProjectFlockApi.basePath}?${new URLSearchParams({ isLoading: isLoadingListProjectFlockKandang,
} = useSWR(
`${ProjectFlockKandangApi.basePath}?${new URLSearchParams({
search: searchProjectFlock, search: searchProjectFlock,
project_flock_id:
projectFlock?.id?.toString() ?? projectFlockId?.toString() ?? '',
}).toString()}`, }).toString()}`,
ProjectFlockApi.getAllFetcher ProjectFlockKandangApi.getAllFetcher
); );
// Mapping Options const {
const options = isResponseSuccess(listProjectFlock) options: options,
? listProjectFlock?.data.map((projectFlock) => { isLoadingOptions: isLoadingListProjectFlock,
return { rawData: listProjectFlock,
value: projectFlock.id, } = useSelect<ProjectFlock>(ProjectFlockApi.basePath, 'id', 'flock_name');
label: `${projectFlock?.flock?.name} - Periode ${projectFlock?.period}`,
};
})
: [];
// Handle Function // Handle Function
const handleChickinClick = async (kandang: Kandang) => { const handleChickinClick = async (
projectFlockKandang: ProjectFlockKandang
) => {
router.push( router.push(
`/production/project-flock/chickin/add/kandang?projectFlockKandangId=${kandang.project_flock_kandang_id}&projectFlockId=${projectFlockId ?? selectedProjectFlock?.value}` `/production/project-flock/chickin/add/kandang?projectFlockKandangId=${projectFlockKandang.id}&projectFlockId=${projectFlockId ?? selectedProjectFlock?.value}`
); );
}; };
@@ -146,6 +155,10 @@ const ProjectFlockChickinDetail = ({
} }
data={projectFlock ? [projectFlock] : []} data={projectFlock ? [projectFlock] : []}
columns={[ columns={[
{
header: 'ID',
accessorKey: 'id',
},
{ {
header: 'Area', header: 'Area',
accessorKey: 'area.name', accessorKey: 'area.name',
@@ -222,7 +235,7 @@ const ProjectFlockChickinDetail = ({
wrapper: 'w-full bg-white', wrapper: 'w-full bg-white',
}} }}
> >
<Table<Kandang> <Table<ProjectFlockKandang>
emptyContent={ emptyContent={
<div className='w-full p-5 text-center'> <div className='w-full p-5 text-center'>
<span className='text-lg opacity-50'> <span className='text-lg opacity-50'>
@@ -230,7 +243,11 @@ const ProjectFlockChickinDetail = ({
</span> </span>
</div> </div>
} }
data={projectFlock ? projectFlock.kandangs : []} data={
isResponseSuccess(listProjectFlockKandang)
? listProjectFlockKandang.data
: []
}
columns={[ columns={[
{ {
header: '#', header: '#',
@@ -240,25 +257,48 @@ const ProjectFlockChickinDetail = ({
1, 1,
}, },
{ {
accessorFn: () => (projectFlock ? projectFlock.area.name : ''), accessorFn: (row) => row?.project_flock?.area?.name,
header: 'Area', header: 'Area',
}, },
{ {
accessorFn: () => accessorFn: (row) => row?.project_flock?.location?.name,
projectFlock ? projectFlock.location.name : '',
header: 'Lokasi', header: 'Lokasi',
}, },
{ {
accessorKey: 'name', accessorKey: 'kandang.name',
header: 'Kandang', header: 'Kandang',
}, },
{ {
accessorKey: 'capacity', accessorKey: 'kandang.capacity',
header: 'Kapasitas', header: 'Kapasitas',
}, },
{ {
accessorKey: 'pic.name', accessorKey: 'approval.step_name',
header: 'Penanggung Jawab', header: 'Status',
cell: (props) => {
return props.row.original.approval?.step_name ? (
<PillBadge
color={(() => {
switch (
props.row.original.approval?.step_name.toUpperCase()
) {
case 'DISETUJUI':
return 'green';
case 'DITOLAK':
return 'red';
default:
return 'gray';
}
})()}
content={props.row.original.approval?.step_name
.toLowerCase()
.replace(/_/g, ' ')
.replace(/\b\w/g, (char) => char.toUpperCase())}
/>
) : (
<PillBadge color='gray' content={'Belum Chick In'} />
);
},
}, },
{ {
header: 'Aksi', header: 'Aksi',
@@ -271,11 +311,12 @@ const ProjectFlockChickinDetail = ({
onClick={() => { onClick={() => {
handleChickinClick(props.row.original); handleChickinClick(props.row.original);
}} }}
className='p-1'
> >
<Icon <Icon
icon='mdi:home-import-outline' icon='mdi:home-import-outline'
width={24} width={18}
height={24} height={18}
/> />
Chickin Chickin
</Button> </Button>
@@ -6,9 +6,7 @@ export const ProjectFlockFormSchema = Yup.object({
value: Yup.number().required('ID Flock wajib diisi!'), value: Yup.number().required('ID Flock wajib diisi!'),
label: Yup.string().required('Nama Flock wajib diisi!'), label: Yup.string().required('Nama Flock wajib diisi!'),
}).nullable(), }).nullable(),
flock_id: Yup.number() flock_name: Yup.string().required('Nama Flock wajib diisi!'),
.min(1, 'Flock wajib diisi!')
.required('Flock wajib diisi!'),
// Area // Area
area: Yup.object({ area: Yup.object({
@@ -1,7 +1,10 @@
'use client'; 'use client';
import Button from '@/components/Button'; import Button from '@/components/Button';
import SelectInput, { OptionType } from '@/components/input/SelectInput'; import SelectInput, {
OptionType,
useSelect,
} from '@/components/input/SelectInput';
import { isResponseError, isResponseSuccess } from '@/lib/api-helper'; import { isResponseError, isResponseSuccess } from '@/lib/api-helper';
import { import {
AreaApi, AreaApi,
@@ -13,7 +16,7 @@ import {
import { Icon } from '@iconify/react'; import { Icon } from '@iconify/react';
import { useFormik } from 'formik'; import { useFormik } from 'formik';
import { useRouter } from 'next/navigation'; import { useRouter } from 'next/navigation';
import { useEffect, useMemo, useState } from 'react'; import { use, useEffect, useMemo, useState } from 'react';
import useSWR, { KeyedMutator } from 'swr'; import useSWR, { KeyedMutator } from 'swr';
import { import {
ProjectFlockFormSchema, ProjectFlockFormSchema,
@@ -58,7 +61,6 @@ const ProjectFlockForm = ({
const [selectedLocation, setSelectedLocation] = useState(''); const [selectedLocation, setSelectedLocation] = useState('');
const [disabledLocation, setDisabledLocation] = useState(true); const [disabledLocation, setDisabledLocation] = useState(true);
const [optionsLocation, setOptionsLocation] = useState<OptionType[]>([]);
const [openSelectKandangs, setOpenSelectKandangs] = useState( const [openSelectKandangs, setOpenSelectKandangs] = useState(
initialValues?.kandangs && initialValues?.kandangs?.length > 0 initialValues?.kandangs && initialValues?.kandangs?.length > 0
@@ -105,38 +107,29 @@ const ProjectFlockForm = ({
}, [initialValues]); }, [initialValues]);
// Fetch Data // Fetch Data
const flockUrl = `${FlockApi.basePath}?${new URLSearchParams({ const {
search: '', rawData: flocks,
}).toString()}`; isLoadingOptions: isLoadingFlocks,
const { data: flocks, isLoading: isLoadingFlocks } = useSWR( options: optionsFlock,
flockUrl, } = useSelect(FlockApi.basePath, 'id', 'name');
FlockApi.getAllFetcher
);
const areaUrl = `${AreaApi.basePath}?${new URLSearchParams({ const {
search: '', options: optionsArea,
}).toString()}`; isLoadingOptions: isLoadingAreas,
const { data: areas, isLoading: isLoadingAreas } = useSWR( rawData: areas,
areaUrl, } = useSelect(AreaApi.basePath, 'id', 'name');
AreaApi.getAllFetcher
);
const locationUrl = `${LocationApi.basePath}?${new URLSearchParams({ const {
search: '', options: optionsLocation,
area_id: selectedArea, isLoadingOptions: isLoadingLocations,
}).toString()}`; rawData: locations,
const { data: locations, isLoading: isLoadingLocations } = useSWR( } = useSelect(LocationApi.basePath, 'id', 'name');
locationUrl,
LocationApi.getAllFetcher
);
const fcrUrl = `${FcrApi.basePath}?${new URLSearchParams({ const {
search: '', options: optionsFcr,
}).toString()}`; isLoadingOptions: isLoadingFcrs,
const { data: fcrs, isLoading: isLoadingFcrs } = useSWR( rawData: fcrs,
fcrUrl, } = useSelect(FcrApi.basePath, 'id', 'name');
FcrApi.getAllFetcher
);
const kandangUrl = `${KandangApi.basePath}?${new URLSearchParams({ const kandangUrl = `${KandangApi.basePath}?${new URLSearchParams({
search: '', search: '',
@@ -159,36 +152,6 @@ const ProjectFlockForm = ({
) )
); );
// Map Data to Options
const optionsArea = isResponseSuccess(areas)
? areas?.data.map((area) => ({
value: area.id,
label: area.name,
}))
: [];
const optionsFcr = isResponseSuccess(fcrs)
? fcrs?.data.map((fcr) => ({
value: fcr.id,
label: fcr.name,
}))
: [];
const optionsFlock = isResponseSuccess(flocks)
? flocks?.data.map((flock) => ({
value: flock.id,
label: flock.name,
}))
: [];
useEffect(() => {
if (isResponseSuccess(locations)) {
const options = locations.data.map((location) => ({
value: location.id,
label: location.name,
}));
setOptionsLocation(options);
}
}, [locations, setSelectedLocation]);
useEffect(() => { useEffect(() => {
if (isResponseSuccess(kandang)) { if (isResponseSuccess(kandang)) {
if (selectedLocation) { if (selectedLocation) {
@@ -246,8 +209,13 @@ const ProjectFlockForm = ({
`${inputName}_id`, `${inputName}_id`,
val ? (val as OptionType)?.value : 0 val ? (val as OptionType)?.value : 0
); );
formik.setFieldValue(
`${inputName}_name`,
val ? (val as OptionType)?.label : 0
);
formik.setFieldTouched(`${inputName}_id`, true); formik.setFieldTouched(`${inputName}_id`, true);
formik.setFieldTouched(`${inputName}_name`, true);
}; };
const categoryChangeHandler = (val: OptionType | OptionType[] | null) => { const categoryChangeHandler = (val: OptionType | OptionType[] | null) => {
@@ -293,15 +261,15 @@ const ProjectFlockForm = ({
const formikInitialValues = useMemo<ProjectFlockFormValues>(() => { const formikInitialValues = useMemo<ProjectFlockFormValues>(() => {
return { return {
name: initialValues?.name ?? '', name: initialValues?.name ?? '',
flock: initialValues?.flock flock: initialValues?.flock_name
? { ? {
value: initialValues.flock.id, value: initialValues?.flock?.id ?? 0,
label: initialValues.flock.name, label: initialValues?.flock_name,
} }
: null, : null,
area: initialValues?.area area: initialValues?.area
? { ? {
value: initialValues.area.id, value: initialValues.area?.id,
label: initialValues.area.name, label: initialValues.area.name,
} }
: null, : null,
@@ -313,30 +281,31 @@ const ProjectFlockForm = ({
: null, : null,
fcr: initialValues?.fcr fcr: initialValues?.fcr
? { ? {
value: initialValues.fcr.id, value: initialValues.fcr?.id,
label: initialValues.fcr.name, label: initialValues.fcr.name,
} }
: null, : null,
location: initialValues?.location location: initialValues?.location
? { ? {
value: initialValues.location.id, value: initialValues.location?.id,
label: initialValues.location.name, label: initialValues.location.name,
} }
: null, : null,
flock_id: initialValues?.flock?.id ?? 0, flock_id: initialValues?.flock?.id ?? 0,
flock_name: initialValues?.flock_name ?? '',
area_id: initialValues?.area?.id ?? 0, area_id: initialValues?.area?.id ?? 0,
category: initialValues?.category as NonNullable< category: initialValues?.category as NonNullable<
'GROWING' | 'LAYING' | undefined 'GROWING' | 'LAYING' | undefined
>, >,
fcr_id: initialValues?.fcr?.id ?? 0, fcr_id: initialValues?.fcr?.id ?? 0,
location_id: initialValues?.location?.id ?? 0, location_id: initialValues?.location?.id ?? 0,
period: initialValues?.period ?? 0, period: initialValues?.period ?? 1,
kandang_ids: initialValues?.kandangs?.map((k: Kandang) => k.id) as ( kandang_ids: initialValues?.kandangs?.map((k: Kandang) => k.id) as (
| number | number
| undefined | undefined
)[], )[],
}; };
}, [initialValues]); }, [initialValues, flocks]);
// Formik // Formik
const formik = useFormik<ProjectFlockFormValues>({ const formik = useFormik<ProjectFlockFormValues>({
@@ -350,7 +319,7 @@ const ProjectFlockForm = ({
onSubmit: async (values) => { onSubmit: async (values) => {
setProjectFlockFormErrorMessage(''); setProjectFlockFormErrorMessage('');
const payload: CreateProjectFlockPayload = { const payload: CreateProjectFlockPayload = {
flock_id: values.flock_id as number, flock_name: values.flock_name as string,
area_id: values.area_id as number, area_id: values.area_id as number,
category: values.category as string, category: values.category as string,
fcr_id: values.fcr_id as number, fcr_id: values.fcr_id as number,
@@ -377,8 +346,8 @@ const ProjectFlockForm = ({
useEffect(() => { useEffect(() => {
if (formType == 'detail') { if (formType == 'detail') {
formik.setFieldValue('area', { formik.setFieldValue('area', {
value: initialValues?.area.id, value: initialValues?.area?.id,
label: initialValues?.area.name, label: initialValues?.area?.name,
}); });
formik.setFieldValue('area_id', initialValues?.area_id); formik.setFieldValue('area_id', initialValues?.area_id);
if (initialValues?.area_id) { if (initialValues?.area_id) {
@@ -402,10 +371,12 @@ const ProjectFlockForm = ({
// Set lokasi otomatis berdasarkan initialValues saat formType = 'detail' // Set lokasi otomatis berdasarkan initialValues saat formType = 'detail'
useEffect(() => { useEffect(() => {
if (formType != 'add' && initialValues?.location?.id) { if (formType != 'add') {
if (initialValues?.location?.id) {
setSelectedLocation(initialValues.location?.id.toString()); setSelectedLocation(initialValues.location?.id.toString());
setDisabledLocation(false); // biar dropdown lokasi aktif juga setDisabledLocation(false); // biar dropdown lokasi aktif juga
} }
}
}, [formType, initialValues]); }, [formType, initialValues]);
useEffect(() => { useEffect(() => {
@@ -459,7 +430,7 @@ const ProjectFlockForm = ({
method: 'POST', method: 'POST',
payload: { payload: {
action: action, action: action,
approvable_ids: [initialValues.id], approvable_ids: [initialValues?.id],
}, },
}); });
@@ -467,14 +438,6 @@ const ProjectFlockForm = ({
if (refreshProjectFlocks) { if (refreshProjectFlocks) {
await refreshProjectFlocks(); await refreshProjectFlocks();
} }
// if (action == 'APPROVED') {
// setIsApprovedDisabled(true);
// setIsRejectedDisabled(false);
// }
// if (action == 'REJECTED') {
// setIsRejectedDisabled(true);
// setIsApprovedDisabled(false);
// }
toast.success(approveProjectFlockRes.message as string); toast.success(approveProjectFlockRes.message as string);
} }
if (isResponseError(approveProjectFlockRes)) { if (isResponseError(approveProjectFlockRes)) {
@@ -591,9 +554,10 @@ const ProjectFlockForm = ({
options={optionsFlock} options={optionsFlock}
isLoading={isLoadingFlocks} isLoading={isLoadingFlocks}
isError={ isError={
formik.touched.flock_id && Boolean(formik.errors.flock_id) formik.touched.flock_name &&
Boolean(formik.errors.flock_name)
} }
errorMessage={formik.errors.flock_id as string} errorMessage={formik.errors.flock_name as string}
isClearable isClearable
isDisabled={formType === 'detail'} isDisabled={formType === 'detail'}
/> />
@@ -724,6 +688,7 @@ const ProjectFlockForm = ({
</div> </div>
)} )}
</div> </div>
{JSON.stringify(formik.errors)}
</form> </form>
{formType != 'add' && ( {formType != 'add' && (
<div className='flex flex-row gap-2 mb-6'> <div className='flex flex-row gap-2 mb-6'>
@@ -215,7 +215,7 @@ const RecordingForm = ({ type = 'add', initialValues }: RecordingFormProps) => {
const flockOptions = isResponseSuccess(projectFlocks) const flockOptions = isResponseSuccess(projectFlocks)
? projectFlocks.data.map((flock) => ({ ? projectFlocks.data.map((flock) => ({
value: flock.id, value: flock.id,
label: flock.flock.name, label: flock.flock?.name || '',
})) }))
: []; : [];
+86 -7
View File
@@ -144,7 +144,7 @@ export const dummyProductWarehouses: ProductWarehouse[] = [
name: 'Pakan Ayam Premium', name: 'Pakan Ayam Premium',
sku: 'PAK-001', sku: 'PAK-001',
category: 'PAKAN', category: 'PAKAN',
} as unknown as Product, // bisa diganti sesuai tipe Product } as unknown as Product,
warehouse: dummyWarehouses[0], warehouse: dummyWarehouses[0],
created_user: createdUser, created_user: createdUser,
created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
@@ -160,7 +160,7 @@ export const dummyProductWarehouses: ProductWarehouse[] = [
name: 'Vitamin Ayam Super', name: 'Vitamin Ayam Super',
sku: 'VIT-002', sku: 'VIT-002',
category: 'VITAMIN', category: 'VITAMIN',
} as unknown as Product, // bisa diganti sesuai tipe Product } as unknown as Product,
warehouse: dummyWarehouses[1], warehouse: dummyWarehouses[1],
created_user: createdUser, created_user: createdUser,
created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
@@ -172,6 +172,7 @@ export const dummyProductWarehouses: ProductWarehouse[] = [
// 💼 Marketing Dummy // 💼 Marketing Dummy
// ====================== // ======================
export const dummyMarketings: Marketing[] = [ export const dummyMarketings: Marketing[] = [
// Step 1: Pengajuan Order
{ {
id: 1, id: 1,
status: 'APPROVED', status: 'APPROVED',
@@ -197,7 +198,7 @@ export const dummyMarketings: Marketing[] = [
grand_total: 7500000, grand_total: 7500000,
approval: { approval: {
step_number: 1, step_number: 1,
step_name: 'Manager Approval', step_name: 'Pengajuan Order',
action: 'APPROVED', action: 'APPROVED',
action_by: createdUser, action_by: createdUser,
action_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), action_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
@@ -227,9 +228,11 @@ export const dummyMarketings: Marketing[] = [
created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
updated_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), updated_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
}, },
// Step 2: Sales Order
{ {
id: 2, id: 2,
status: 'PENDING', status: 'APPROVED',
so_number: 'SO-002-2025', so_number: 'SO-002-2025',
so_docs: 'https://example.com/docs/so002.pdf', so_docs: 'https://example.com/docs/so002.pdf',
so_date: format(new Date(), 'yyyy-MM-dd'), so_date: format(new Date(), 'yyyy-MM-dd'),
@@ -251,9 +254,9 @@ export const dummyMarketings: Marketing[] = [
notes: 'Pesanan kedua untuk stok akhir tahun.', notes: 'Pesanan kedua untuk stok akhir tahun.',
grand_total: 3750000, grand_total: 3750000,
approval: { approval: {
step_number: 1, step_number: 2,
step_name: 'Manager Approval', step_name: 'Sales Order',
action: 'PENDING', action: 'APPROVED',
action_by: createdUser, action_by: createdUser,
action_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), action_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
}, },
@@ -282,4 +285,80 @@ export const dummyMarketings: Marketing[] = [
created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
updated_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'), updated_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
}, },
// Step 3: Delivery Order
{
id: 3,
status: 'APPROVED',
so_number: 'SO-003-2025',
so_docs: 'https://example.com/docs/so003.pdf',
so_date: format(new Date(), 'yyyy-MM-dd'),
customer: {
id: 3,
name: 'UD Ternak Sejahtera',
pic_id: 3,
pic: createdUser,
type: 'Reseller',
address: 'Jl. Pasteur No. 88',
phone: '083333333333',
email: 'halo@ternaksejahtera.com',
account_number: '1122334455',
created_user: createdUser,
created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
updated_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
},
sales_person: createdUser,
notes: 'Order untuk pengiriman ke luar kota.',
grand_total: 5600000,
approval: {
step_number: 3,
step_name: 'Delivery Order',
action: 'APPROVED',
action_by: createdUser,
action_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
},
marketing_products: [
{
id: 3,
qty: 80,
unit_price: 70000,
avg_weight: 2.4,
total_weight: 192,
total_price: 5600000,
product_warehouse: dummyProductWarehouses[0],
marketing_delivery_products: {
id: 3,
qty: 80,
unit_price: 70000,
avg_weight: 2.4,
total_weight: 192,
total_price: 5600000,
delivery_date: format(new Date(), 'yyyy-MM-dd'),
vehicle_number: 'D 9090 ZZ',
},
},
{
id: 4,
qty: 80,
unit_price: 70000,
avg_weight: 2.4,
total_weight: 192,
total_price: 5600000,
product_warehouse: dummyProductWarehouses[0],
marketing_delivery_products: {
id: 3,
qty: 80,
unit_price: 70000,
avg_weight: 2.4,
total_weight: 192,
total_price: 5600000,
delivery_date: format(new Date(), 'yyyy-MM-dd'),
vehicle_number: 'D 9090 ZZ',
},
},
],
created_user: createdUser,
created_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
updated_at: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
},
]; ];
+14 -2
View File
@@ -47,14 +47,25 @@ export class MarketingService extends BaseApiService<
// simulasi delay // simulasi delay
await new Promise((res) => setTimeout(res, 500)); await new Promise((res) => setTimeout(res, 500));
const marketing = dummyMarketings.find((marketing) => {
console.log('marketing', marketing);
console.log('id-m', marketing.id);
console.log('id-p', id);
console.log('id', marketing.id == id);
return marketing.id == id;
});
console.log('marketings', dummyMarketings);
console.log('marketing', marketing);
if (marketing) {
// misalnya fetch dari dummy // misalnya fetch dari dummy
return { return {
code: 200, code: 200,
status: 'success', status: 'success',
message: 'Data marketing berhasil diambil.', message: 'Data marketing berhasil diambil.',
data: dummyMarketings[0], data: marketing,
}; };
} else {
// jika tidak ditemukan // jika tidak ditemukan
throw { throw {
code: 404, code: 404,
@@ -62,6 +73,7 @@ export class MarketingService extends BaseApiService<
message: 'Data marketing tidak ditemukan.', message: 'Data marketing tidak ditemukan.',
}; };
} }
}
/** /**
* Approve single marketing data * Approve single marketing data
+1 -1
View File
@@ -20,7 +20,7 @@ export const ProjectFlockApi = new BaseApiService<
ProjectFlock, ProjectFlock,
CreateProjectFlockPayload, CreateProjectFlockPayload,
UpdateProjectFlockPayload UpdateProjectFlockPayload
>('/production/project_flocks'); >('/production/project-flocks');
export const ProjectFlockKandangApi = new BaseApiService< export const ProjectFlockKandangApi = new BaseApiService<
ProjectFlockKandang, ProjectFlockKandang,
unknown, unknown,
+1
View File
@@ -41,6 +41,7 @@ export type MarketingDeliveryProducts = {
total_price: number; total_price: number;
delivery_date: string; delivery_date: string;
vehicle_number: string; vehicle_number: string;
do_number?: string | undefined;
}; };
export type Marketing = BaseMetadata & BaseMarketing; export type Marketing = BaseMetadata & BaseMarketing;
+4 -2
View File
@@ -14,9 +14,11 @@ export type Chickin = BaseMetadata & BaseChickin;
export type CreateChickinPayload = { export type CreateChickinPayload = {
project_flock_kandang_id: number; project_flock_kandang_id: number;
chickin_requests: {
chick_in_date: string; chick_in_date: string;
note: string; note?: string;
quantity?: number; product_warehouse_id: number;
}[];
}; };
export type UpdateChickinPayload = CreateChickinPayload & { export type UpdateChickinPayload = CreateChickinPayload & {
+4 -3
View File
@@ -2,6 +2,7 @@ import { Kandang } from '@/type/master-data/kandang';
import { ProjectFlock } from '@/types/api/production/project-flock'; import { ProjectFlock } from '@/types/api/production/project-flock';
import { ProductWarehouse } from '@/types/api/inventory/product-warehouse'; import { ProductWarehouse } from '@/types/api/inventory/product-warehouse';
import { Supplier } from '../master-data/supplier'; import { Supplier } from '../master-data/supplier';
import { BaseApproval } from '../api-general';
export type BaseProjectFlockKandang = { export type BaseProjectFlockKandang = {
id: number; id: number;
@@ -10,14 +11,14 @@ export type BaseProjectFlockKandang = {
kandang: Kandang; kandang: Kandang;
project_flock: ProjectFlock; project_flock: ProjectFlock;
available_qtys?: AvailableQty[]; available_qtys?: AvailableQty[];
approval: BaseApproval;
}; };
export type AvailableQty = { export type AvailableQty = {
chick_in_date?: string; chick_in_date?: string;
po_number?: string; available_qty: number;
document_path?: string;
supplier?: Supplier;
product_warehouse: ProductWarehouse; product_warehouse: ProductWarehouse;
note?: string;
}; };
export type ProjectFlockKandang = BaseProjectFlockKandang; export type ProjectFlockKandang = BaseProjectFlockKandang;
+4 -3
View File
@@ -9,8 +9,9 @@ export type BaseProjectFlock = {
id: number; id: number;
name: string; name: string;
status: string; status: string;
flock: Flock; flock?: Flock;
flock_id: number; flock_i?: number;
flock_name: string;
area: Area; area: Area;
area_id: number; area_id: number;
category: string; category: string;
@@ -32,7 +33,7 @@ export type PeriodFlock = {
export type ProjectFlock = BaseMetadata & BaseProjectFlock; export type ProjectFlock = BaseMetadata & BaseProjectFlock;
export type CreateProjectFlockPayload = { export type CreateProjectFlockPayload = {
flock_id: number; flock_name: string;
area_id: number; area_id: number;
category: string; category: string;
fcr_id: number; fcr_id: number;