[Valentino Heman Budiarto] 5e005e8524 29 Mei 2026
2026-05-29 18:55:54 +07:00

156 lines
6.7 KiB
TypeScript

"use client";
import { useState, useEffect } from "react";
import axios from "axios";
import { ShieldCheck, Check, X } from "lucide-react";
export default function ApprovalsPage() {
const [activeTab, setActiveTab] = useState("Pending");
const [bookings, setBookings] = useState<any[]>([]);
// 1. FUNGSI UNTUK MENGAMBIL DATA ASLI DARI DATABASE
const fetchBookings = async () => {
try {
const token = localStorage.getItem("token");
// Mengambil dari fungsi GetAllBookings di backend
const res = await axios.get("http://172.17.110.6:8080/api/bookings", {
headers: { Authorization: `Bearer ${token}` }
});
setBookings(res.data.data);
} catch (err: any) {
console.error("Gagal mengambil data peminjaman:", err);
}
};
useEffect(() => {
fetchBookings();
}, []);
// 2. FUNGSI UNTUK MENGUBAH STATUS (APPROVE / REJECT)
const updateStatus = async (bookingId: string, newStatus: string) => {
try {
const token = localStorage.getItem("token");
// Mengirim UUID booking_id ke backend Golang
await axios.put(
`http://172.17.110.6:8080/api/bookings/${bookingId}/status`,
{ status: newStatus },
{ headers: { Authorization: `Bearer ${token}` } }
);
alert(`Peminjaman berhasil di-${newStatus}!`);
fetchBookings(); // Refresh tabel setelah update
} catch (err: any) {
console.error("Gagal update status:", err.response?.data || err.message);
alert("Error: " + (err.response?.data?.error || "Gagal mengubah status"));
}
};
const filteredBookings = bookings.filter(b => b.status === activeTab);
return (
<div className="space-y-6">
<div className="flex items-center gap-3 mb-6">
<div className="bg-blue-100 p-3 rounded-lg text-blue-600">
<ShieldCheck size={28} />
</div>
<div>
<h2 className="text-2xl font-bold text-gray-800">Manajemen Persetujuan</h2>
<p className="text-gray-500 text-sm mt-1">Kelola permohonan peminjaman ruangan S-CLASS.</p>
</div>
</div>
{/* TABS */}
<div className="flex gap-4 border-b border-gray-200">
{["Pending", "Approved", "Rejected"].map((tab) => (
<button
key={tab}
onClick={() => setActiveTab(tab)}
className={`pb-3 px-4 font-bold text-sm transition-all ${
activeTab === tab
? "border-b-2 border-blue-600 text-blue-600"
: "text-gray-400 hover:text-gray-600"
}`}
>
{tab}
</button>
))}
</div>
{/* TABEL */}
<div className="bg-white rounded-xl shadow-sm border border-gray-100 overflow-hidden">
<div className="overflow-x-auto">
<table className="w-full text-left">
<thead className="bg-gray-50 text-gray-500 text-xs font-bold uppercase">
<tr>
<th className="p-4">Peminjam</th>
<th className="p-4">Ruangan</th>
<th className="p-4">Tanggal & Waktu</th>
<th className="p-4">Keperluan</th>
<th className="p-4 text-center">Status / Aksi</th>
</tr>
</thead>
<tbody className="divide-y divide-gray-50">
{filteredBookings.map((b) => (
// Pastikan key menggunakan booking_id (UUID)
<tr key={b.booking_id} className="hover:bg-gray-50/50 transition-colors">
{/* Gunakan huruf KECIL: b.user dan b.room sesuai JSON dari Golang */}
<td className="p-4 font-bold text-gray-800 text-sm">{b.user?.full_name || "Tanpa Nama"}</td>
<td className="p-4 text-sm font-medium text-gray-600">{b.room?.name || "Ruangan Tidak Diketahui"}</td>
{/* Menampilkan Waktu Mulai sampai Waktu Selesai */}
<td className="p-4 text-sm text-gray-600">
<div className="font-bold">
{new Date(b.start_time).toLocaleTimeString('id-ID', {hour: '2-digit', minute:'2-digit'})} - {new Date(b.end_time).toLocaleTimeString('id-ID', {hour: '2-digit', minute:'2-digit'})}
</div>
<div className="text-xs text-gray-400">
{new Date(b.start_time).toLocaleDateString('id-ID')}
</div>
</td>
<td className="p-4 text-sm text-gray-600">{b.purpose}</td>
<td className="p-4 text-center">
{b.status === "Pending" ? (
<div className="flex justify-center gap-2">
{/* 3. TOMBOL SETUJUI & TOLAK YANG SUDAH BERFUNGSI */}
<button
// KITA CEK SEMUA KEMUNGKINAN NAMA ID-NYA:
onClick={() => updateStatus(b.booking_id || b.BookingID || b.id, "Approved")}
className="flex items-center gap-1 px-3 py-1.5 bg-green-50 text-green-600 font-bold text-xs rounded-lg hover:bg-green-100 transition-colors"
>
<Check size={14} /> Setujui
</button>
<button
onClick={() => updateStatus(b.booking_id || b.BookingID || b.id, "Rejected")}
className="flex items-center gap-1 px-3 py-1.5 bg-red-50 text-red-600 font-bold text-xs rounded-lg hover:bg-red-100 transition-colors"
>
<X size={14} /> Tolak
</button>
</div>
) : (
<div className="flex flex-col items-center gap-1">
<span className={`px-3 py-1 rounded-full text-xs font-bold ${
b.status === 'Approved' ? 'bg-green-100 text-green-700' : 'bg-red-100 text-red-700'
}`}>
{b.status}
</span>
{/* Menampilkan Redeem Code jika sudah Approved */}
{b.status === 'Approved' && b.redeem_code && (
<span className="text-[10px] text-gray-500 font-mono bg-gray-100 px-2 py-0.5 rounded">
Code: {b.redeem_code}
</span>
)}
</div>
)}
</td>
</tr>
))}
{filteredBookings.length === 0 && (
<tr><td colSpan={5} className="p-8 text-center text-gray-400 font-medium">Tidak ada data di kategori ini.</td></tr>
)}
</tbody>
</table>
</div>
</div>
</div>
);
}