This commit is contained in:
[Valentino Heman Budiarto] 2026-06-05 20:16:32 +07:00
parent 2bedb35863
commit 097fa72e51

View File

@ -1,50 +1,74 @@
"use client"; "use client";
import { useState, useEffect } from "react"; import { useState, useEffect } from "react";
// Import ikon yang diperlukan
import { Activity, Power, ZapOff, AlertTriangle, Lightbulb, Wind, Projector } from "lucide-react"; import { Activity, Power, ZapOff, AlertTriangle, Lightbulb, Wind, Projector } from "lucide-react";
export default function PowerMonitoringPage() { export default function PowerMonitoringPage() {
// =========================================================================
const fetchPowerStatus = async () => { // 1. DEKLARASI STATE
try { // =========================================================================
const response = await fetch("http://172.17.172.17:8080/api/hardware/power-status");
const data = await response.json();
// Asumsikan data dari HA berupa string angka, kita update ke room D101
setRooms(prev => prev.map(room =>
room.name === "Kelas D101" ? { ...room, power: parseFloat(data.power) || 0 } : room
));
} catch (err) {
console.error("Gagal mengambil data daya:", err);
}
};
useEffect(() => {
// Ambil data awal
fetchPowerStatus();
// Set interval untuk update setiap 5 detik agar real-time sesuai HA
const interval = setInterval(fetchPowerStatus, 5000);
return () => clearInterval(interval);
}, []);
const [rooms, setRooms] = useState<any[]>([]); const [rooms, setRooms] = useState<any[]>([]);
// 1. STATE BARU: Dipisahkan berdasarkan ID Ruangan agar tidak bentrok
const [roomDeviceStatus, setRoomDeviceStatus] = useState<{ [roomId: string]: { [deviceName: string]: boolean } }>({}); const [roomDeviceStatus, setRoomDeviceStatus] = useState<{ [roomId: string]: { [deviceName: string]: boolean } }>({});
// =========================================================================
// 2. FUNGSI FETCH DATA DARI BACKEND
// =========================================================================
// A. Tarik Data Daya (Power / Watt)
const fetchPowerStatus = async () => {
try {
const response = await fetch("http://172.17.172.17:8080/api/hardware/power-status");
const data = await response.json();
setRooms(prev => prev.map(room =>
room.name === "Kelas D101" ? { ...room, power: parseFloat(data.power) || 0 } : room
));
} catch (err) {
console.error("Gagal mengambil data daya:", err);
}
};
// B. Tarik Data Status Perangkat (Sinkronisasi Real-time Antar Admin)
const fetchDeviceStatus = async () => {
try {
const response = await fetch("http://172.17.172.17:8080/api/hardware/status");
const result = await response.json();
if (result.status === "success") {
const backendData = result.data;
// Kita terjemahkan data Golang ("on"/"off") menjadi boolean (true/false)
// dan masukkan khusus untuk Kelas D101 (id: 1)
setRoomDeviceStatus(prev => ({
...prev,
"room_1": {
...prev["room_1"],
"Lampu 1": backendData.lampu1 === "on",
"Lampu 2": backendData.lampu2 === "on",
"AC 1": backendData.ac === "on",
"Proyektor": backendData.projector === "on",
}
}));
}
} catch (err) {
console.error("Gagal sinkronisasi status perangkat:", err);
}
};
// =========================================================================
// 3. INISIALISASI & POLLING (AUTO-REFRESH)
// =========================================================================
useEffect(() => { useEffect(() => {
// Simulasi Data Ruangan dari IoT // A. Simulasi Data Ruangan dari DB/IoT
const dummyRooms = [ const dummyRooms = [
{ id: 1, name: "Kelas D101", power: 1250, isRelayOn: true, lastUpdate: "Baru saja" }, { id: 1, name: "Kelas D101", power: 0, isRelayOn: true, lastUpdate: "Real-time" },
{ id: 2, name: "Kelas D102", power: 0, isRelayOn: false, lastUpdate: "2 mnt lalu" }, { id: 2, name: "Kelas D102", power: 0, isRelayOn: false, lastUpdate: "2 mnt lalu" },
{ id: 3, name: "Kelas D103", power: 45, isRelayOn: true, lastUpdate: "Baru saja" }, { id: 3, name: "Kelas D103", power: 45, isRelayOn: true, lastUpdate: "Baru saja" },
{ id: 4, name: "Kelas D104", power: 0, isRelayOn: false, lastUpdate: "10 mnt lalu" }, { id: 4, name: "Kelas D104", power: 0, isRelayOn: false, lastUpdate: "10 mnt lalu" },
]; ];
setRooms(dummyRooms); setRooms(dummyRooms);
// Inisialisasi status default (semua OFF) untuk setiap ruangan // B. Inisialisasi status default semua device OFF
const initialStatus: { [roomId: string]: { [deviceName: string]: boolean } } = {}; const initialStatus: { [roomId: string]: { [deviceName: string]: boolean } } = {};
dummyRooms.forEach(room => { dummyRooms.forEach(room => {
initialStatus[`room_${room.id}`] = { initialStatus[`room_${room.id}`] = {
@ -55,9 +79,24 @@ useEffect(() => {
}; };
}); });
setRoomDeviceStatus(initialStatus); setRoomDeviceStatus(initialStatus);
// C. Tarik data pertama kali saat halaman dibuka
fetchPowerStatus();
fetchDeviceStatus();
// D. Jalankan Polling setiap 2 detik (2000 ms)
const interval = setInterval(() => {
fetchPowerStatus();
fetchDeviceStatus();
}, 2000);
// E. Bersihkan interval jika admin pindah halaman
return () => clearInterval(interval);
}, []); }, []);
// 2. FUNGSI HANDLE TOGGLE YANG DIPERBAIKI (Menerima roomId) // =========================================================================
// 4. FUNGSI KONTROL DEVICE (TOGGLE ON/OFF)
// =========================================================================
const handleDeviceToggle = async (roomId: number, roomName: string, deviceName: string) => { const handleDeviceToggle = async (roomId: number, roomName: string, deviceName: string) => {
const roomIdKey = `room_${roomId}`; const roomIdKey = `room_${roomId}`;
const currentStatus = roomDeviceStatus[roomIdKey]?.[deviceName] || false; const currentStatus = roomDeviceStatus[roomIdKey]?.[deviceName] || false;
@ -72,29 +111,44 @@ useEffect(() => {
else if (deviceName === 'Lampu 1') backendDevice = "lampu1"; else if (deviceName === 'Lampu 1') backendDevice = "lampu1";
else if (deviceName === 'Lampu 2') backendDevice = "lampu2"; else if (deviceName === 'Lampu 2') backendDevice = "lampu2";
// Update UI seketika agar terasa responsif bagi admin yang menekan tombol
setRoomDeviceStatus(prev => ({
...prev,
[roomIdKey]: {
...prev[roomIdKey],
[deviceName]: !currentStatus,
},
}));
try { try {
// Tembak API Golang (Ganti dengan endpoint aslimu jika berbeda)
const response = await fetch("http://172.17.172.17:8080/api/hardware/control", { const response = await fetch("http://172.17.172.17:8080/api/hardware/control", {
method: "POST", method: "POST",
headers: { "Content-Type": "application/json" }, headers: { "Content-Type": "application/json" },
body: JSON.stringify({ device: backendDevice, action: actionType }), body: JSON.stringify({ device: backendDevice, action: actionType }),
}); });
if (response.ok) { if (!response.ok) {
// 3. UPDATE STATE HANYA UNTUK RUANGAN YANG DIKLIK // Jika gagal, kembalikan posisi tombol ke keadaan semula
setRoomDeviceStatus(prev => ({ setRoomDeviceStatus(prev => ({
...prev, ...prev,
[roomIdKey]: { [roomIdKey]: {
...prev[roomIdKey], ...prev[roomIdKey],
[deviceName]: !currentStatus, [deviceName]: currentStatus,
}, },
})); }));
} else {
const errorData = await response.json(); const errorData = await response.json();
alert(`GAGAL: ${errorData.error || response.statusText}`); alert(`GAGAL: ${errorData.error || response.statusText}`);
} }
} catch (error) { } catch (error) {
console.error("Error API:", error); console.error("Error API:", error);
// Jika server mati, kembalikan posisi tombol
setRoomDeviceStatus(prev => ({
...prev,
[roomIdKey]: {
...prev[roomIdKey],
[deviceName]: currentStatus,
},
}));
alert("GAGAL: Tidak dapat terhubung ke Server Golang."); alert("GAGAL: Tidak dapat terhubung ke Server Golang.");
} }
}; };
@ -105,23 +159,25 @@ useEffect(() => {
} }
}; };
// =========================================================================
// 5. TAMPILAN UI (RENDER)
// =========================================================================
return ( return (
<div className="space-y-6"> <div className="space-y-6">
{/* Header section... */} {/* Header section */}
<div className="flex items-center gap-3 mb-6"> <div className="flex items-center gap-3 mb-6">
<div className="bg-blue-100 p-3 rounded-lg text-blue-600"> <div className="bg-blue-100 p-3 rounded-lg text-blue-600">
<Activity size={28} /> <Activity size={28} />
</div> </div>
<div> <div>
<h2 className="text-2xl font-bold text-gray-800">Power Monitoring & Control</h2> <h2 className="text-2xl font-bold text-gray-800">Power Monitoring & Control</h2>
<p className="text-gray-500 text-sm mt-1">Pantau konsumsi daya kWh meter dan kendalikan *relay* sirkuit ruangan.</p> <p className="text-gray-500 text-sm mt-1">Pantau konsumsi daya kWh meter dan kendalikan relay sirkuit ruangan.</p>
</div> </div>
</div> </div>
<div className="grid grid-cols-1 md:grid-cols-2 xl:grid-cols-3 gap-6"> <div className="grid grid-cols-1 md:grid-cols-2 xl:grid-cols-3 gap-6">
{[...rooms].sort((a, b) => a.name.localeCompare(b.name)).map((room) => { {[...rooms].sort((a, b) => a.name.localeCompare(b.name)).map((room) => {
// Ambil status spesifik untuk ruangan ini
const roomIdKey = `room_${room.id}`; const roomIdKey = `room_${room.id}`;
const currentRoomStatus = roomDeviceStatus[roomIdKey] || {}; const currentRoomStatus = roomDeviceStatus[roomIdKey] || {};
@ -136,7 +192,7 @@ useEffect(() => {
<h3 className="text-lg font-bold text-gray-800 mb-4">{room.name}</h3> <h3 className="text-lg font-bold text-gray-800 mb-4">{room.name}</h3>
{/* Power display... */} {/* Power display */}
<div className="flex items-end gap-2 mb-6"> <div className="flex items-end gap-2 mb-6">
<span className={`text-5xl font-black tracking-tight ${room.power > 1000 ? 'text-orange-500' : 'text-gray-800'}`}> <span className={`text-5xl font-black tracking-tight ${room.power > 1000 ? 'text-orange-500' : 'text-gray-800'}`}>
{room.power} {room.power}
@ -150,7 +206,7 @@ useEffect(() => {
</div> </div>
)} )}
{/* 3. PANEL KONTROL IoT BARU DENGAN LOGIKA WARNA DIPERBAIKI */} {/* PANEL KONTROL IoT */}
<div className="mt-2 mb-6 pt-4 border-t border-gray-100"> <div className="mt-2 mb-6 pt-4 border-t border-gray-100">
<p className="text-[10px] font-bold text-gray-400 uppercase tracking-wider mb-3">IoT Device Control</p> <p className="text-[10px] font-bold text-gray-400 uppercase tracking-wider mb-3">IoT Device Control</p>
<div className="grid grid-cols-2 gap-2"> <div className="grid grid-cols-2 gap-2">
@ -200,13 +256,13 @@ useEffect(() => {
? 'bg-purple-100 text-purple-600 border border-purple-200' ? 'bg-purple-100 text-purple-600 border border-purple-200'
: 'bg-gray-50 text-gray-400 border border-transparent'}`} : 'bg-gray-50 text-gray-400 border border-transparent'}`}
> >
<Projector size={14} /> Proyektor <Projector size={14} className={currentRoomStatus['Proyektor'] ? "fill-purple-500" : ""} /> Proyektor
</button> </button>
</div> </div>
</div> </div>
{/* Footer... */} {/* Footer */}
<div className="flex items-center justify-between mt-auto pt-4 border-t border-gray-100"> <div className="flex items-center justify-between mt-auto pt-4 border-t border-gray-100">
<span className="text-xs text-gray-400 font-medium">Update: {room.lastUpdate}</span> <span className="text-xs text-gray-400 font-medium">Update: {room.lastUpdate}</span>
<button <button