38428-vm/admin.php
2026-02-15 18:22:05 +00:00

153 lines
7.3 KiB
PHP

<?php
require_once __DIR__ . '/db/config.php';
$secret_token = 'lili_admin_2026';
if (($_GET['token'] ?? '') !== $secret_token) {
die('Acceso denegado. Se requiere un token válido.');
}
$db = db();
// Get active users (last 10 minutes)
$stmt = $db->query("SELECT COUNT(*) FROM visitor_logs WHERE last_activity > DATE_SUB(NOW(), INTERVAL 10 MINUTE)");
$active_users = $stmt->fetchColumn();
// Get active users with phone numbers
$stmt = $db->query("SELECT phone_number, country, last_activity FROM visitor_logs WHERE last_activity > DATE_SUB(NOW(), INTERVAL 10 MINUTE) AND phone_number IS NOT NULL AND phone_number != '' ORDER BY last_activity DESC");
$active_phones = $stmt->fetchAll();
// Get country distribution for active users
$stmt = $db->query("SELECT country, country_code, lat, lon, COUNT(*) as count FROM visitor_logs WHERE last_activity > DATE_SUB(NOW(), INTERVAL 10 MINUTE) GROUP BY country_code");
$locations = $stmt->fetchAll();
// Handle Video Password Update
if (isset($_POST['update_video_pass'])) {
$new_pass = trim($_POST['video_pass']);
$stmt = $db->prepare("INSERT INTO settings (setting_key, setting_value) VALUES ('video_room_password', ?) ON DUPLICATE KEY UPDATE setting_value = ?");
$stmt->execute([$new_pass, $new_pass]);
$success_msg = "Contraseña de la sala de video actualizada.";
}
// Get current video password
$stmt = $db->prepare("SELECT setting_value FROM settings WHERE setting_key = 'video_room_password'");
$stmt->execute();
$current_video_pass = $stmt->fetchColumn() ?: "lili123";
?>
<!DOCTYPE html>
<html lang="es">
<head>
<meta charset="UTF-8">
<title>Admin Dashboard - Lili Records</title>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.1/font/bootstrap-icons.css">
<link rel="stylesheet" href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css" />
<style>
body { background: #0f172a; color: white; font-family: 'Inter', sans-serif; }
.card { background: rgba(30, 41, 59, 0.7); border: 1px solid rgba(255,255,255,0.1); color: white; backdrop-filter: blur(10px); }
#map { height: 500px; border-radius: 15px; margin-top: 20px; }
.stat-value { font-size: 3rem; font-weight: bold; color: #00e676; }
.btn-success { background: #00e676; border: none; color: #0f172a; font-weight: bold; }
</style>
</head>
<body>
<div class="container py-5">
<div class="row mb-4">
<div class="col-md-12 text-center">
<img src="./assets/pasted-20260215-163754-def41f49.png" alt="Logo" style="width: 100px; height: 100px; border-radius: 50%; border: 4px solid #00e676; margin-bottom: 1rem; object-fit: cover; box-shadow: 0 0 25px rgba(0, 230, 118, 0.7);">
<h1>Panel de Administración Real-Time</h1>
<p class="text-secondary">Lili Records Radio Statistics</p>
</div>
</div>
<?php if (isset($success_msg)): ?>
<div class="alert alert-success bg-success text-white border-0 mb-4"><?= $success_msg ?></div>
<?php endif; ?>
<div class="row mb-4">
<div class="col-md-12">
<div class="card p-4">
<h5 class="mb-3"><i class="bi bi-camera-video-fill text-primary"></i> Control de Sala de Video</h5>
<form method="POST" class="row g-3">
<div class="col-auto">
<label class="form-label">Contraseña actual para los usuarios:</label>
<input type="text" name="video_pass" class="form-control bg-dark text-white border-secondary" value="<?= htmlspecialchars($current_video_pass) ?>" required>
</div>
<div class="col-auto d-flex align-items-end">
<button type="submit" name="update_video_pass" class="btn btn-success">ACTUALIZAR CONTRASEÑA</button>
</div>
<div class="col-12 mt-2">
<small class="text-secondary">Esta contraseña será solicitada a los usuarios que intenten activar su cámara en la web.</small>
</div>
</form>
</div>
</div>
</div>
<div class="row">
<div class="col-md-4">
<div class="card p-4 text-center">
<h5>Usuarios Conectados</h5>
<div class="stat-value"><?= $active_users ?></div>
<p class="text-secondary">En los últimos 10 minutos</p>
</div>
<div class="card p-4 mt-4">
<h5>Móviles Conectados</h5>
<ul class="list-group list-group-flush bg-transparent">
<?php if (empty($active_phones)): ?>
<li class="list-group-item bg-transparent text-secondary border-secondary">No hay móviles registrados</li>
<?php else: ?>
<?php foreach ($active_phones as $phone): ?>
<li class="list-group-item bg-transparent text-white border-secondary d-flex justify-content-between align-items-center">
<div>
<span class="fw-bold"><?= htmlspecialchars($phone['phone_number']) ?></span><br>
<small class="text-secondary"><?= htmlspecialchars($phone['country']) ?></small>
</div>
<span class="badge bg-success">Online</span>
</li>
<?php endforeach; ?>
<?php endif; ?>
</ul>
</div>
<div class="card p-4 mt-4">
<h5>Distribución por País</h5>
<ul class="list-group list-group-flush bg-transparent">
<?php foreach ($locations as $loc): ?>
<li class="list-group-item bg-transparent text-white border-secondary d-flex justify-content-between">
<span><?= htmlspecialchars($loc['country']) ?></span>
<span class="badge bg-primary"><?= $loc['count'] ?></span>
</li>
<?php endforeach; ?>
</ul>
</div>
</div>
<div class="col-md-8">
<div class="card p-3">
<h5>Mapa de Conexiones</h5>
<div id="map"></div>
</div>
</div>
</div>
</div>
<script src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js"></script>
<script>
const map = L.map('map').setView([20, 0], 2);
L.tileLayer('https://{s}.basemaps.cartocdn.com/dark_all/{z}/{x}/{y}{r}.png', {
attribution: '&copy; OpenStreetMap contributors'
}).addTo(map);
const locations = <?= json_encode($locations) ?>;
locations.forEach(loc => {
if (loc.lat && loc.lon) {
L.marker([loc.lat, loc.lon])
.addTo(map)
.bindPopup(`<b>${loc.country}</b><br>${loc.count} usuario(s)`);
}
});
</script>
</body>
</html>