This commit is contained in:
Halbe Bruno
2025-12-05 19:40:39 -03:00
commit f37bc712e6
4312 changed files with 359196 additions and 0 deletions

View File

@@ -0,0 +1,202 @@
<!DOCTYPE html>
<html lang="pt-BR">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title><?= $title ?? 'Dashboard' ?> - DNSBlock Admin</title>
<script src="https://cdn.tailwindcss.com"></script>
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700&display=swap" rel="stylesheet">
<!-- Alpine.js for interactions -->
<script defer src="https://cdn.jsdelivr.net/npm/alpinejs@3.x.x/dist/cdn.min.js"></script>
<style>
body {
font-family: 'Inter', sans-serif;
}
[x-cloak] {
display: none !important;
}
</style>
<script>
tailwind.config = {
theme: {
extend: {
colors: {
primary: {
50: '#fef2f2',
100: '#fee2e2',
500: '#ef4444',
600: '#dc2626',
700: '#b91c1c',
}
}
}
}
}
</script>
</head>
<body class="bg-gray-50 text-gray-800 font-sans antialiased h-screen overflow-hidden" x-data="{ sidebarOpen: false }">
<!-- Mobile Header -->
<div class="md:hidden flex items-center justify-between bg-white border-b border-gray-200 p-4 sticky top-0 z-30">
<a href="/admin/dashboard" class="font-bold text-xl tracking-tight">DNS<span
class="text-primary-600">Block</span></a>
<button @click="sidebarOpen = !sidebarOpen" class="text-gray-500 focus:outline-none">
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h16">
</path>
</svg>
</button>
</div>
<div class="flex h-screen overflow-hidden">
<!-- Sidebar -->
<aside :class="sidebarOpen ? 'translate-x-0' : '-translate-x-full'"
class="fixed inset-y-0 left-0 z-40 w-64 bg-white border-r border-gray-200 transition-transform duration-300 md:relative md:translate-x-0 flex flex-col">
<div class="h-16 flex items-center px-6 border-b border-gray-100">
<a href="/admin/dashboard" class="font-bold text-2xl tracking-tight text-gray-900">DNS<span
class="text-primary-600">Block</span></a>
</div>
<nav class="flex-1 px-4 py-6 space-y-1 overflow-y-auto">
<a href="/admin/dashboard"
class="flex items-center px-4 py-3 text-gray-700 rounded-lg hover:bg-primary-50 hover:text-primary-700 transition-colors group <?= ($_SERVER['REQUEST_URI'] == '/admin/dashboard') ? 'bg-primary-50 text-primary-700' : '' ?>">
<svg class="w-5 h-5 mr-3 text-gray-400 group-hover:text-primary-500" fill="none"
stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M4 6a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2H6a2 2 0 01-2-2V6zM14 6a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2h-2a2 2 0 01-2-2V6zM4 16a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2H6a2 2 0 01-2-2v-2zM14 16a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2h-2a2 2 0 01-2-2v-2z">
</path>
</svg>
Dashboard
</a>
<div class="pt-4 pb-2">
<p class="px-4 text-xs font-semibold text-gray-400 uppercase tracking-wider">Gestão</p>
</div>
<a href="/admin/clients"
class="flex items-center px-4 py-3 text-gray-700 rounded-lg hover:bg-primary-50 hover:text-primary-700 transition-colors group <?= (strpos($_SERVER['REQUEST_URI'], '/admin/clients') !== false) ? 'bg-primary-50 text-primary-700' : '' ?>">
<svg class="w-5 h-5 mr-3 text-gray-400 group-hover:text-primary-500" fill="none"
stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z">
</path>
</svg>
Clientes
</a>
<a href="/admin/servers"
class="flex items-center px-4 py-3 text-gray-700 rounded-lg hover:bg-primary-50 hover:text-primary-700 transition-colors group <?= (strpos($_SERVER['REQUEST_URI'], '/admin/servers') !== false) ? 'bg-primary-50 text-primary-700' : '' ?>">
<svg class="w-5 h-5 mr-3 text-gray-400 group-hover:text-primary-500" fill="none"
stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M5 12h14M5 12a2 2 0 01-2-2V6a2 2 0 012-2h14a2 2 0 012 2v4a2 2 0 01-2 2M5 12a2 2 0 00-2 2v4a2 2 0 002 2h14a2 2 0 002-2v-4a2 2 0 00-2-2m-2-4h.01M17 16h.01">
</path>
</svg>
Servidores
</a>
<a href="/admin/orders"
class="flex items-center px-4 py-3 text-gray-700 rounded-lg hover:bg-primary-50 hover:text-primary-700 transition-colors group <?= (strpos($_SERVER['REQUEST_URI'], '/admin/orders') !== false) ? 'bg-primary-50 text-primary-700' : '' ?>">
<svg class="w-5 h-5 mr-3 text-gray-400 group-hover:text-primary-500" fill="none"
stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z">
</path>
</svg>
Ordens Judiciais
</a>
<a href="/admin/logs"
class="flex items-center px-4 py-3 text-gray-700 rounded-lg hover:bg-primary-50 hover:text-primary-700 transition-colors group <?= (strpos($_SERVER['REQUEST_URI'], '/admin/logs') !== false) ? 'bg-primary-50 text-primary-700' : '' ?>">
<svg class="w-5 h-5 mr-3 text-gray-400 group-hover:text-primary-500" fill="none"
stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-3 7h3m-3 4h3m-6-4h.01M9 16h.01">
</path>
</svg>
Logs do Sistema
</a>
</nav>
<div class="p-4 border-t border-gray-100">
<a href="/logout"
class="flex items-center px-4 py-3 text-gray-600 rounded-lg hover:bg-red-50 hover:text-red-700 transition-colors">
<svg class="w-5 h-5 mr-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M17 16l4-4m0 0l-4-4m4 4H7m6 4v1a3 3 0 01-3 3H6a3 3 0 01-3-3V7a3 3 0 013-3h4a3 3 0 013 3v1">
</path>
</svg>
Sair
</a>
</div>
</aside>
<!-- Main Content Wrapper -->
<div class="flex-1 flex flex-col overflow-hidden">
<!-- Topbar (Fixed) -->
<header class="bg-white shadow-sm z-20 flex-none">
<div class="max-w-7xl mx-auto py-4 px-4 sm:px-6 lg:px-8 flex justify-between items-center">
<h2 class="font-semibold text-xl text-gray-800 leading-tight">
<?= $title ?? 'Dashboard' ?>
</h2>
<div class="flex items-center space-x-4">
<form action="/admin/search" method="GET" class="relative">
<span class="absolute inset-y-0 left-0 pl-3 flex items-center pointer-events-none">
<svg class="h-5 w-5 text-gray-400" fill="none" stroke="currentColor"
viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"></path>
</svg>
</span>
<input type="text" name="q"
class="pl-10 pr-4 py-2 border border-gray-300 rounded-full text-sm focus:outline-none focus:ring-2 focus:ring-primary-500 focus:border-transparent"
placeholder="Buscar..." value="<?= $_GET['q'] ?? '' ?>">
</form>
<a href="/admin/profile"
class="flex items-center hover:bg-gray-50 rounded-full pr-3 py-1 transition-colors">
<div
class="h-8 w-8 rounded-full bg-primary-100 flex items-center justify-center text-primary-700 font-bold">
<?= substr($_SESSION['user_name'] ?? 'A', 0, 1) ?>
</div>
<span
class="ml-2 text-sm font-medium text-gray-700 hidden md:block"><?= $_SESSION['user_name'] ?? 'Admin' ?></span>
</a>
</div>
</div>
</header>
<!-- Scrollable Content -->
<main class="flex-1 overflow-x-hidden overflow-y-auto bg-gray-50">
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
<?php if (isset($_SESSION['flash_success'])): ?>
<div class="mb-4 bg-green-100 border border-green-400 text-green-700 px-4 py-3 rounded relative"
role="alert">
<span class="block sm:inline"><?= $_SESSION['flash_success'] ?></span>
<?php unset($_SESSION['flash_success']); ?>
</div>
<?php endif; ?>
<?php if (isset($_SESSION['flash_error'])): ?>
<div class="mb-4 bg-red-100 border border-red-400 text-red-700 px-4 py-3 rounded relative"
role="alert">
<span class="block sm:inline"><?= $_SESSION['flash_error'] ?></span>
<?php unset($_SESSION['flash_error']); ?>
</div>
<?php endif; ?>
<?php if (isset($content))
include $content; ?>
</div>
</main>
</div>
<!-- Overlay -->
<div x-show="sidebarOpen" @click="sidebarOpen = false"
class="fixed inset-0 z-30 bg-black bg-opacity-50 md:hidden" x-transition.opacity></div>
</div>
</body>
</html>