import Styles from "@/constants/Styles"; import { apiApproveRejectProjectTask, apiDeleteProjectTask, apiGetProjectOne, apiGetProjectTaskApprovals, apiSubmitProjectTask } from "@/lib/api"; import { setUpdateProject } from "@/lib/projectUpdate"; import { useAuthSession } from "@/providers/AuthProvider"; import { useTheme } from "@/providers/ThemeProvider"; import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons"; import { router, useLocalSearchParams } from "expo-router"; import { useEffect, useState } from "react"; import { View } from "react-native"; import Toast from "react-native-toast-message"; import { useDispatch, useSelector } from "react-redux"; import DrawerBottom from "../drawerBottom"; import ItemSectionTanggalTugas from "../itemSectionTanggalTugas"; import MenuItemRow from "../menuItemRow"; import ModalConfirmation from "../ModalConfirmation"; import ModalRiwayatApproval from "../ModalRiwayatApproval"; import ModalTolakApproval from "../ModalTolakApproval"; import SkeletonTask from "../skeletonTask"; import Text from "../Text"; import ModalListDetailTugasProject from "./modalListDetailTugasProject"; type Props = { id: string; title: string; desc: string; status: number; dateStart: string; dateEnd: string; createdAt: string; files?: { name: string; extension: string }[]; }; type ApprovalRecord = { id: string status: number note?: string submitter: { name: string } approver?: { name: string } createdAt: string } export default function SectionTanggalTugasProject({ status, member, refreshing, idGroup }: { status: number | undefined, member: boolean, refreshing?: boolean, idGroup: string }) { const { colors } = useTheme(); const entityUser = useSelector((state: any) => state.user) const dispatch = useDispatch() const update = useSelector((state: any) => state.projectUpdate) const [isModal, setModal] = useState(false); const { token, decryptToken } = useAuthSession(); const [modalDetail, setModalDetail] = useState(false) const [modalRiwayat, setModalRiwayat] = useState(false) const [modalTolak, setModalTolak] = useState(false) const [modalKonfirmasiSetujui, setModalKonfirmasiSetujui] = useState(false) const [modalPersetujuan, setModalPersetujuan] = useState(false) const { id } = useLocalSearchParams<{ id: string }>(); const [data, setData] = useState([]); const [loading, setLoading] = useState(true) const [loadingAction, setLoadingAction] = useState(false) const [loadingRiwayat, setLoadingRiwayat] = useState(false) const [riwayatData, setRiwayatData] = useState([]) const arrSkeleton = Array.from({ length: 5 }); const [tugas, setTugas] = useState({ id: '', status: 0 }) const [showDeleteModal, setShowDeleteModal] = useState(false) const isApprover = (entityUser.isApprover && entityUser.idGroup === idGroup) || ['supadmin', 'developer'].includes(entityUser.role) const isAdmin = entityUser.role !== 'user' && entityUser.role !== 'coadmin' async function handleLoad(loading: boolean) { try { setLoading(loading) const hasil = await decryptToken(String(token?.current)); const response = await apiGetProjectOne({ user: hasil, cat: "task", id: id }); setData(response.data); } catch (error) { console.error(error); } finally { setLoading(false) } } useEffect(() => { handleLoad(false) }, [update.task]); useEffect(() => { if (refreshing) handleLoad(false) }, [refreshing]); useEffect(() => { handleLoad(true) }, []); async function handleLoadRiwayat() { try { setLoadingRiwayat(true) const hasil = await decryptToken(String(token?.current)) const response = await apiGetProjectTaskApprovals({ user: hasil, id: tugas.id }) setRiwayatData(response.data ?? []) } catch (error) { console.error(error) } finally { setLoadingRiwayat(false) } } async function handleSubmitAjukan() { try { setLoadingAction(true) const hasil = await decryptToken(String(token?.current)) const response = await apiSubmitProjectTask({ user: hasil, id: tugas.id }) if (response.success) { dispatch(setUpdateProject({ ...update, progress: !update.progress, task: !update.task })) Toast.show({ type: 'small', text1: 'Berhasil mengajukan task untuk persetujuan' }) } else { Toast.show({ type: 'small', text1: response.message }) } } catch (error: any) { const message = error?.response?.data?.message || "Gagal mengajukan persetujuan" Toast.show({ type: 'small', text1: message }) } finally { setLoadingAction(false) setModal(false) } } async function handleSetujui() { try { setLoadingAction(true) const hasil = await decryptToken(String(token?.current)) const response = await apiApproveRejectProjectTask({ user: hasil, id: tugas.id, action: 'approve' }) if (response.success) { dispatch(setUpdateProject({ ...update, progress: !update.progress, task: !update.task })) Toast.show({ type: 'small', text1: 'Tugas berhasil disetujui' }) } else { Toast.show({ type: 'small', text1: response.message }) } } catch (error: any) { const message = error?.response?.data?.message || "Gagal menyetujui tugas" Toast.show({ type: 'small', text1: message }) } finally { setLoadingAction(false) setModalKonfirmasiSetujui(false) } } async function handleTolak(note: string) { try { setLoadingAction(true) const hasil = await decryptToken(String(token?.current)) const response = await apiApproveRejectProjectTask({ user: hasil, id: tugas.id, action: 'reject', note }) if (response.success) { dispatch(setUpdateProject({ ...update, progress: !update.progress, task: !update.task })) Toast.show({ type: 'small', text1: 'Tugas berhasil ditolak' }) } else { Toast.show({ type: 'small', text1: response.message }) } } catch (error: any) { const message = error?.response?.data?.message || "Gagal menolak tugas" Toast.show({ type: 'small', text1: message }) } finally { setLoadingAction(false) setModalTolak(false) } } async function handleDelete() { try { const hasil = await decryptToken(String(token?.current)); const response = await apiDeleteProjectTask({ user: hasil, idProject: id }, tugas.id); if (response.success) { dispatch(setUpdateProject({ ...update, progress: !update.progress, task: !update.task })) Toast.show({ type: 'small', text1: 'Berhasil menghapus data' }) } } catch (error: any) { const message = error?.response?.data?.message || "Gagal menghapus data" Toast.show({ type: 'small', text1: message }) } } const canTakeAction = member || isAdmin const showAjukan = (member || isApprover) && tugas.status === 0 && status !== 3 const showApproverActions = isApprover && tugas.status === 2 return ( <> Tanggal & Tugas {loading ? arrSkeleton.map((_, index) => ) : data.length > 0 ? data.map((item, index) => ( { setTugas({ id: item.id, status: item.status }) setModal(true) }} /> )) : Tidak ada tugas } {/* Drawer menu */} {/* Baris 1 — selalu tampil */} } title="File Tugas" onPress={() => { setModal(false) router.push(`/project/${id}/tugas-file/${tugas.id}?member=${member}`) }} /> } title="Detail Waktu" onPress={() => { setModal(false) setTimeout(() => setModalDetail(true), 600) }} /> } title="Riwayat" onPress={() => { setModal(false) handleLoadRiwayat() setTimeout(() => setModalRiwayat(true), 600) }} /> {/* Separator antar baris */} {(showAjukan || showApproverActions || (canTakeAction && isAdmin && status !== 3)) && ( )} {/* Baris 2 — semua aksi kondisional dalam satu baris */} {showAjukan && ( } title="Ajukan Selesai" disabled={loadingAction} onPress={() => { setModal(false) setTimeout(() => handleSubmitAjukan(), 600) }} /> )} {showApproverActions && ( } title="Persetujuan" disabled={loadingAction} onPress={() => { setModal(false) setTimeout(() => setModalPersetujuan(true), 600) }} /> )} {canTakeAction && isAdmin && status !== 3 && ( <> } title="Edit Tugas" onPress={() => { setModal(false) router.push(`/project/update/${tugas.id}`) }} /> } title="Hapus Tugas" onPress={() => { setModal(false) setTimeout(() => setShowDeleteModal(true), 600) }} /> )} {/* Drawer persetujuan */} } title="Setujui" color={colors.success} disabled={loadingAction} onPress={() => { setModalPersetujuan(false) setTimeout(() => setModalKonfirmasiSetujui(true), 600) }} /> } title="Tolak" color={colors.error} disabled={loadingAction} onPress={() => { setModalPersetujuan(false) setTimeout(() => setModalTolak(true), 600) }} /> { setShowDeleteModal(false); handleDelete() }} onCancel={() => setShowDeleteModal(false)} confirmText="Hapus" cancelText="Batal" /> setModalKonfirmasiSetujui(false)} confirmText="Setujui" cancelText="Batal" /> ); }