admin monitoring

This commit is contained in:
[Valentino Heman Budiarto] 2026-06-15 17:40:51 +07:00
parent c85f4c6b2d
commit 1cf7a02ba7

View File

@ -1,48 +1,47 @@
"use client";
import { useState, useEffect } from "react";
import { Activity, Power, AlertTriangle, Lightbulb, Wind, Projector, Zap } from "lucide-react";
import { Activity, Power, ZapOff, AlertTriangle, Lightbulb, Wind, Projector } from "lucide-react";
export default function PowerMonitoringPage() {
// =========================================================================
// 1. DEKLARASI STATE UNTUK KELAS D101
// 1. DEKLARASI STATE
// =========================================================================
const [powerData, setPowerData] = useState({ umum: 0, ac1: 0, ac2: 0 });
const [deviceStatus, setDeviceStatus] = useState({
"Lampu 1": false,
"Lampu 2": false,
"AC": false,
"Proyektor": false,
});
const [isRelayOn, setIsRelayOn] = useState(true); // Status apakah ruangan terkunci/terbuka
const [lastUpdate, setLastUpdate] = useState("Menunggu data...");
const [rooms, setRooms] = useState<any[]>([]);
const [roomDeviceStatus, setRoomDeviceStatus] = useState<{ [roomId: string]: { [deviceName: string]: boolean } }>({});
// State khusus untuk menampung rincian 3 MCB di D101
const [powerDataD101, setPowerDataD101] = useState({ umum: 0, ac1: 0, ac2: 0 });
// =========================================================================
// 2. FUNGSI FETCH DATA DARI BACKEND GOLANG
// 2. FUNGSI FETCH DATA DARI BACKEND
// =========================================================================
// A. Tarik Data Daya dari 3 MCB (Umum, AC1, AC2)
// A. Tarik Data Daya (Power / Watt)
const fetchPowerStatus = async () => {
try {
// Pastikan route ini sesuai dengan router.GET di main.go Golang kamu
const response = await fetch("http://172.17.172.17:8080/api/hardware/power-status");
if (response.ok) {
const data = await response.json();
setPowerData({
umum: parseFloat(data.umum) || 0,
ac1: parseFloat(data.ac1) || 0,
ac2: parseFloat(data.ac2) || 0,
});
const now = new Date();
setLastUpdate(`${now.getHours().toString().padStart(2, '0')}:${now.getMinutes().toString().padStart(2, '0')}:${now.getSeconds().toString().padStart(2, '0')}`);
const umum = parseFloat(data.umum) || 0;
const ac1 = parseFloat(data.ac1) || 0;
const ac2 = parseFloat(data.ac2) || 0;
const totalD101 = umum + ac1 + ac2;
setPowerDataD101({ umum, ac1, ac2 });
// Update nilai power D101 di dalam array rooms
setRooms(prev => prev.map(room =>
room.name === "Kelas D101" ? { ...room, power: totalD101, lastUpdate: "Real-time" } : room
));
}
} catch (err) {
console.error("Gagal mengambil data daya:", err);
}
};
// B. Tarik Data Status Perangkat IoT
// B. Tarik Data Status Perangkat (Sinkronisasi Real-time)
const fetchDeviceStatus = async () => {
try {
const response = await fetch("http://172.17.172.17:8080/api/hardware/status");
@ -50,14 +49,17 @@ export default function PowerMonitoringPage() {
if (result.status === "success") {
const backendData = result.data;
setDeviceStatus({
"Lampu 1": backendData.lampu1 === "on",
"Lampu 2": backendData.lampu2 === "on",
"AC": backendData.ac === "on",
"Proyektor": backendData.projector === "on",
});
// Jika semua mati dan gemboknya aktif, bisa kita asumsikan ruangan off
// Tergantung bagaimana logika "auth" kamu di frontend
setRoomDeviceStatus(prev => ({
...prev,
"room_1": { // room_1 diasumsikan sebagai D101
...prev["room_1"],
"Lampu 1": backendData.lampu1 === "on",
"Lampu 2": backendData.lampu2 === "on",
"AC": backendData.ac === "on",
"Proyektor": backendData.projector === "on",
}
}));
}
} catch (err) {
console.error("Gagal sinkronisasi status perangkat:", err);
@ -65,16 +67,37 @@ export default function PowerMonitoringPage() {
};
// =========================================================================
// 3. INISIALISASI & POLLING (AUTO-REFRESH 3 DETIK)
// 3. INISIALISASI & POLLING (AUTO-REFRESH)
// =========================================================================
useEffect(() => {
// Simulasi Data Ruangan
const dummyRooms = [
{ id: 1, name: "Kelas D101", power: 0, isRelayOn: true, lastUpdate: "Menunggu..." },
{ 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: 4, name: "Kelas D104", power: 0, isRelayOn: false, lastUpdate: "10 mnt lalu" },
];
setRooms(dummyRooms);
// Inisialisasi status default semua device OFF
const initialStatus: { [roomId: string]: { [deviceName: string]: boolean } } = {};
dummyRooms.forEach(room => {
initialStatus[`room_${room.id}`] = {
"Lampu 1": false,
"Lampu 2": false,
"AC": false,
"Proyektor": false,
};
});
setRoomDeviceStatus(initialStatus);
fetchPowerStatus();
fetchDeviceStatus();
const interval = setInterval(() => {
fetchPowerStatus();
fetchDeviceStatus();
}, 3000);
}, 3000); // Polling 3 detik
return () => clearInterval(interval);
}, []);
@ -82,24 +105,30 @@ export default function PowerMonitoringPage() {
// =========================================================================
// 4. FUNGSI KONTROL DEVICE & CUT OFF
// =========================================================================
const handleDeviceToggle = async (deviceName: string) => {
// Tentukan kunci array state yang benar
const key = deviceName as keyof typeof deviceStatus;
const currentStatus = deviceStatus[key];
const handleDeviceToggle = async (roomId: number, roomName: string, deviceName: string) => {
const roomIdKey = `room_${roomId}`;
const currentStatus = roomDeviceStatus[roomIdKey]?.[deviceName] || false;
const actionType = currentStatus ? "off" : "on";
if (currentStatus) {
if (!window.confirm(`Apakah Anda yakin ingin mematikan ${deviceName}?`)) return;
const confirmMsg = `Apakah Anda yakin ingin mematikan ${deviceName} di ${roomName}?`;
if (!window.confirm(confirmMsg)) return;
}
// Jika yang ditekan bukan D101, ubah UI saja (simulasi)
if (roomName !== "Kelas D101") {
setRoomDeviceStatus(prev => ({ ...prev, [roomIdKey]: { ...prev[roomIdKey], [deviceName]: !currentStatus } }));
return;
}
// Logika khusus D101 (Tembak ke Golang)
let backendDevice = "";
if (deviceName === 'AC') backendDevice = "ac";
else if (deviceName === 'Proyektor') backendDevice = "projector";
else if (deviceName === 'Lampu 1') backendDevice = "lampu1";
else if (deviceName === 'Lampu 2') backendDevice = "lampu2";
// Optimistic UI (Berubah cepat di layar)
setDeviceStatus(prev => ({ ...prev, [key]: !currentStatus }));
setRoomDeviceStatus(prev => ({ ...prev, [roomIdKey]: { ...prev[roomIdKey], [deviceName]: !currentStatus } }));
try {
const response = await fetch("http://172.17.172.17:8080/api/hardware/control", {
@ -109,167 +138,156 @@ export default function PowerMonitoringPage() {
});
if (!response.ok) {
setDeviceStatus(prev => ({ ...prev, [key]: currentStatus }));
const errorData = await response.json();
alert(`GAGAL: ${errorData.error || "Server menolak perintah"}`);
setRoomDeviceStatus(prev => ({ ...prev, [roomIdKey]: { ...prev[roomIdKey], [deviceName]: currentStatus } }));
alert("Gagal mengontrol perangkat.");
}
} catch (error) {
console.error("Error API:", error);
setDeviceStatus(prev => ({ ...prev, [key]: currentStatus }));
setRoomDeviceStatus(prev => ({ ...prev, [roomIdKey]: { ...prev[roomIdKey], [deviceName]: currentStatus } }));
alert("GAGAL: Tidak dapat terhubung ke Server Golang.");
}
};
// FUNGSI CUT OFF 3 MCB SEKALIGUS
const handleCutOff = async () => {
if (!window.confirm(`PERINGATAN FATAL: Anda yakin ingin memutus 3 MCB Daya di Kelas D101 secara paksa?`)) return;
try {
const response = await fetch("http://172.17.172.17:8080/api/cutoff", {
method: "POST",
});
if (response.ok) {
alert(`Daya di Kelas D101 berhasil diputus!`);
setIsRelayOn(false);
fetchPowerStatus(); // Paksa update layar agar jadi 0 Watt
fetchDeviceStatus(); // Paksa update icon perangkat mati
} else {
alert("Gagal memutus daya. Cek koneksi server.");
// FUNGSI CUT OFF DAYA KHUSUS D101
const handleCutOff = async (roomName: string, roomId: number) => {
if (roomName === "Kelas D101") {
if (!window.confirm(`PERINGATAN FATAL: Anda yakin ingin memutus 3 MCB Daya di ${roomName} secara paksa?`)) return;
try {
const response = await fetch("http://172.17.172.17:8080/api/cutoff", { method: "POST" });
if (response.ok) {
alert(`Daya di ${roomName} berhasil diputus!`);
setRooms(prev => prev.map(r => r.name === roomName ? { ...r, isRelayOn: false, power: 0 } : r));
fetchPowerStatus();
} else {
alert("Gagal memutus daya. Cek koneksi server.");
}
} catch (error) {
alert("Terjadi kesalahan jaringan saat memutus daya.");
}
} else {
// Dummy untuk kelas lain
if (window.confirm(`Simulasi mematikan daya di ${roomName}?`)) {
setRooms(prev => prev.map(r => r.name === roomName ? { ...r, isRelayOn: false, power: 0 } : r));
}
} catch (error) {
console.error(error);
alert("Terjadi kesalahan jaringan saat memutus daya.");
}
};
// Hitung total daya
const totalPower = powerData.umum + powerData.ac1 + powerData.ac2;
// =========================================================================
// 5. TAMPILAN UI (RENDER)
// =========================================================================
return (
<div className="space-y-6 max-w-6xl mx-auto">
{/* HEADER SECTION */}
<div className="flex flex-col md:flex-row justify-between items-start md:items-center gap-4 mb-6">
<div className="flex items-center gap-3">
<div className="bg-blue-100 p-3 rounded-lg text-blue-600">
<Activity size={28} />
</div>
<div>
<h2 className="text-2xl font-bold text-gray-800">S-CLASS Power Monitoring</h2>
<p className="text-gray-500 text-sm mt-1">Pantau beban 3 MCB utama & kendalikan perangkat Kelas D101.</p>
</div>
<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">
<Activity size={28} />
</div>
<button
onClick={handleCutOff}
className="flex items-center gap-2 bg-red-600 hover:bg-red-700 text-white px-6 py-3 rounded-xl font-bold shadow-lg transition-all duration-200"
>
<Power size={20} /> GLOBAL CUT OFF
</button>
</div>
{/* TOTAL POWER SUMMARY */}
<div className="bg-gray-900 text-white p-6 rounded-2xl shadow-lg flex flex-col md:flex-row justify-between items-center gap-4">
<div className="flex items-center gap-4">
<div className="bg-gray-800 p-4 rounded-full">
<Zap size={32} className="text-yellow-400" />
</div>
<div>
<h3 className="text-lg text-gray-300 font-semibold">Total Konsumsi Daya (D101)</h3>
<p className="text-sm text-gray-400">Update terakhir: {lastUpdate}</p>
</div>
</div>
<div className="text-right">
<span className={`text-5xl font-black ${totalPower > 3000 ? 'text-red-400' : 'text-yellow-400'}`}>
{totalPower.toFixed(1)}
</span>
<span className="text-2xl text-gray-400 ml-2">Watts</span>
<div>
<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>
</div>
</div>
{/* 3 MCB MONITORING CARDS */}
<div className="grid grid-cols-1 md:grid-cols-3 gap-6">
{/* MCB UMUM */}
<div className="bg-white p-6 rounded-xl border border-gray-100 shadow-sm relative overflow-hidden">
<div className="absolute top-0 right-0 px-3 py-1 text-[10px] font-black uppercase bg-blue-100 text-blue-700 rounded-bl-xl">MCB 1</div>
<h3 className="text-sm font-bold text-gray-500 uppercase tracking-wider mb-1">Jalur Umum</h3>
<p className="text-xs text-gray-400 mb-4">Lampu, Proyektor & Stop Kontak</p>
<div className="flex items-end gap-2">
<span className="text-4xl font-black text-gray-800">{powerData.umum.toFixed(1)}</span>
<span className="text-gray-500 font-bold mb-1">W</span>
</div>
</div>
{/* MCB AC 1 */}
<div className="bg-white p-6 rounded-xl border border-gray-100 shadow-sm relative overflow-hidden">
<div className="absolute top-0 right-0 px-3 py-1 text-[10px] font-black uppercase bg-teal-100 text-teal-700 rounded-bl-xl">MCB 2</div>
<h3 className="text-sm font-bold text-gray-500 uppercase tracking-wider mb-1">Pendingin 1</h3>
<p className="text-xs text-gray-400 mb-4">Air Conditioner Unit 1</p>
<div className="flex items-end gap-2">
<span className="text-4xl font-black text-gray-800">{powerData.ac1.toFixed(1)}</span>
<span className="text-gray-500 font-bold mb-1">W</span>
</div>
</div>
{/* MCB AC 2 */}
<div className="bg-white p-6 rounded-xl border border-gray-100 shadow-sm relative overflow-hidden">
<div className="absolute top-0 right-0 px-3 py-1 text-[10px] font-black uppercase bg-teal-100 text-teal-700 rounded-bl-xl">MCB 3</div>
<h3 className="text-sm font-bold text-gray-500 uppercase tracking-wider mb-1">Pendingin 2</h3>
<p className="text-xs text-gray-400 mb-4">Air Conditioner Unit 2</p>
<div className="flex items-end gap-2">
<span className="text-4xl font-black text-gray-800">{powerData.ac2.toFixed(1)}</span>
<span className="text-gray-500 font-bold mb-1">W</span>
</div>
</div>
</div>
{totalPower > 3500 && (
<div className="flex items-center gap-3 bg-red-50 border border-red-200 text-red-700 p-4 rounded-xl font-bold animate-pulse">
<AlertTriangle size={24} /> PERINGATAN: Beban Kelas D101 melebihi batas aman (3500W)!
</div>
)}
{/* IOT DEVICE CONTROLS */}
<div className="bg-white p-6 rounded-xl border border-gray-100 shadow-sm mt-6">
<h3 className="text-lg font-bold text-gray-800 mb-4">Kontrol Perangkat Kelas D101</h3>
<div className="grid grid-cols-2 md:grid-cols-4 gap-4">
<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) => {
<button
onClick={() => handleDeviceToggle('Lampu 1')}
className={`flex flex-col items-center justify-center gap-3 p-4 rounded-xl font-bold transition-all border-2
${deviceStatus['Lampu 1'] ? 'bg-yellow-50 border-yellow-400 text-yellow-700' : 'bg-gray-50 border-transparent text-gray-400 hover:bg-gray-100'}`}
>
<Lightbulb size={28} className={deviceStatus['Lampu 1'] ? "fill-yellow-400" : ""} /> Lampu 1
</button>
const roomIdKey = `room_${room.id}`;
const currentRoomStatus = roomDeviceStatus[roomIdKey] || {};
<button
onClick={() => handleDeviceToggle('Lampu 2')}
className={`flex flex-col items-center justify-center gap-3 p-4 rounded-xl font-bold transition-all border-2
${deviceStatus['Lampu 2'] ? 'bg-yellow-50 border-yellow-400 text-yellow-700' : 'bg-gray-50 border-transparent text-gray-400 hover:bg-gray-100'}`}
>
<Lightbulb size={28} className={deviceStatus['Lampu 2'] ? "fill-yellow-400" : ""} /> Lampu 2
</button>
return (
<div key={room.id} className={`bg-white p-6 rounded-xl border-2 shadow-sm relative overflow-hidden flex flex-col transition-all
${room.name === "Kelas D101" ? "border-blue-300" : "border-gray-100"}`}>
<div className={`absolute top-0 right-0 px-4 py-1.5 text-[10px] font-black uppercase tracking-wider rounded-bl-xl
${room.isRelayOn ? 'bg-green-100 text-green-700' : 'bg-red-100 text-red-700'}`}>
{room.isRelayOn ? 'Sirkuit Aktif' : 'Sirkuit Terputus'}
</div>
<button
onClick={() => handleDeviceToggle('AC')}
className={`flex flex-col items-center justify-center gap-3 p-4 rounded-xl font-bold transition-all border-2
${deviceStatus['AC'] ? 'bg-blue-50 border-blue-400 text-blue-700' : 'bg-gray-50 border-transparent text-gray-400 hover:bg-gray-100'}`}
>
<Wind size={28} className={deviceStatus['AC'] ? "animate-pulse" : ""} /> AC Ruangan
</button>
<h3 className="text-lg font-bold text-gray-800 mb-1">{room.name}</h3>
{room.name === "Kelas D101" && <p className="text-[10px] text-blue-500 font-bold mb-3">(Live Connected API)</p>}
{room.name !== "Kelas D101" && <p className="text-[10px] text-gray-400 font-bold mb-3">(Simulasi / Offline)</p>}
<div className="flex items-end gap-2 mb-2">
<span className={`text-5xl font-black tracking-tight ${room.power > 3000 ? 'text-red-500' : room.power > 1000 ? 'text-orange-500' : 'text-gray-800'}`}>
{typeof room.power === 'number' ? room.power.toFixed(1) : room.power}
</span>
<span className="text-gray-500 font-bold mb-1.5">Watts</span>
</div>
<button
onClick={() => handleDeviceToggle('Proyektor')}
className={`flex flex-col items-center justify-center gap-3 p-4 rounded-xl font-bold transition-all border-2
${deviceStatus['Proyektor'] ? 'bg-purple-50 border-purple-400 text-purple-700' : 'bg-gray-50 border-transparent text-gray-400 hover:bg-gray-100'}`}
>
<Projector size={28} className={deviceStatus['Proyektor'] ? "fill-purple-400" : ""} /> Proyektor
</button>
{/* RINCIAN KHUSUS D101 */}
{room.name === "Kelas D101" && (
<div className="bg-gray-50 p-2 rounded-lg flex justify-between text-[10px] text-gray-600 font-medium mb-4 border border-gray-100">
<span>Umum: <b className="text-gray-800">{powerDataD101.umum.toFixed(0)}W</b></span>
<span>AC1: <b className="text-gray-800">{powerDataD101.ac1.toFixed(0)}W</b></span>
<span>AC2: <b className="text-gray-800">{powerDataD101.ac2.toFixed(0)}W</b></span>
</div>
)}
{room.name !== "Kelas D101" && <div className="mb-4 h-8"></div>}
</div>
{room.power > 3000 && (
<div className="flex items-center gap-2 text-xs font-bold text-red-600 bg-red-50 p-2 rounded-lg mb-4">
<AlertTriangle size={14} /> Batas Daya Terlampaui!
</div>
)}
<div className="mt-auto 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>
<div className="grid grid-cols-2 gap-2">
<button
onClick={() => handleDeviceToggle(room.id, room.name, 'Lampu 1')}
disabled={!room.isRelayOn}
className={`flex justify-center items-center gap-2 p-2 rounded-lg text-xs font-bold transition-all
${currentRoomStatus['Lampu 1'] ? 'bg-yellow-100 text-yellow-600 border border-yellow-200' : 'bg-gray-50 text-gray-400 border border-transparent'}`}
>
<Lightbulb size={14} className={currentRoomStatus['Lampu 1'] ? "fill-yellow-500" : ""} /> Lampu 1
</button>
<button
onClick={() => handleDeviceToggle(room.id, room.name, 'Lampu 2')}
disabled={!room.isRelayOn}
className={`flex justify-center items-center gap-2 p-2 rounded-lg text-xs font-bold transition-all
${currentRoomStatus['Lampu 2'] ? 'bg-yellow-100 text-yellow-600 border border-yellow-200' : 'bg-gray-50 text-gray-400 border border-transparent'}`}
>
<Lightbulb size={14} className={currentRoomStatus['Lampu 2'] ? "fill-yellow-500" : ""} /> Lampu 2
</button>
<button
onClick={() => handleDeviceToggle(room.id, room.name, 'AC')}
disabled={!room.isRelayOn}
className={`flex justify-center items-center gap-2 p-2 rounded-lg text-xs font-bold transition-all
${currentRoomStatus['AC'] ? 'bg-blue-100 text-blue-600 border border-blue-200' : 'bg-gray-50 text-gray-400 border border-transparent'}`}
>
<Wind size={14} className={currentRoomStatus['AC'] ? "animate-pulse" : ""} /> AC
</button>
<button
onClick={() => handleDeviceToggle(room.id, room.name, 'Proyektor')}
disabled={!room.isRelayOn}
className={`flex justify-center items-center gap-2 p-2 rounded-lg text-xs font-bold transition-all
${currentRoomStatus['Proyektor'] ? 'bg-purple-100 text-purple-600 border border-purple-200' : 'bg-gray-50 text-gray-400 border border-transparent'}`}
>
<Projector size={14} className={currentRoomStatus['Proyektor'] ? "fill-purple-500" : ""} /> Proyektor
</button>
</div>
</div>
<div className="flex items-center justify-between mt-6 pt-4 border-t border-gray-100">
<span className="text-xs text-gray-400 font-medium">Update: {room.lastUpdate}</span>
<button
onClick={() => handleCutOff(room.name, room.id)}
disabled={!room.isRelayOn}
className={`flex items-center gap-2 px-4 py-2 rounded-lg text-xs font-bold transition-all
${room.isRelayOn
? 'bg-red-50 text-red-600 hover:bg-red-600 hover:text-white border border-red-200 hover:border-red-600'
: 'bg-gray-100 text-gray-400 cursor-not-allowed'}`}
>
{room.isRelayOn ? <><Power size={14} /> Cut Off Power</> : <><ZapOff size={14} /> Offline</>}
</button>
</div>
</div>
);
})}
</div>
</div>
);