first commit

This commit is contained in:
2026-06-16 10:45:41 +07:00
commit ccecc0bc6b
144 changed files with 124547 additions and 0 deletions

View File

@@ -0,0 +1,381 @@
<!DOCTYPE html>
<html>
<head>
<title>NRML Dashboard</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" rel="stylesheet">
<script src="https://cdn.jsdelivr.net/npm/chart.js@3.9.1"></script>
<script src="https://cdn.jsdelivr.net/npm/chartjs-plugin-datalabels@2.2.0"></script>
<script src="https://cdn.jsdelivr.net/npm/html-to-image@1.11.11/dist/html-to-image.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js"></script>
<script src="https://html2canvas.hertzen.com/dist/html2canvas.min.js"></script>
<link rel="stylesheet" href="https://unpkg.com/leaflet/dist/leaflet.css" />
<script src="https://unpkg.com/leaflet/dist/leaflet.js"></script>
<script src="/js/dashboard/filter.js"></script>
<script src="/js/dashboard/charts.js"></script>
<script src="/js/dashboard/export.js"></script>
<style>
body {
margin: 0;
}
.top-navbar {
height: 60px;
background: #0B8F3C;
color: white;
display: flex;
align-items: center;
padding: 0 25px;
}
.brand-title {
font-weight: 600;
font-size: 18px;
color: #f8f9fa;
}
.nav-bar {
display: flex;
background: white;
border-bottom: 1px solid #dcdcdc;
padding: 0 20px;
position: sticky;
top: 0;
z-index: 1000;
}
.btn {
border-radius: 0 !important;
}
.btn-theme-outline {
background-color: #fff;
color: #0B8F3C;
border: 1px solid #0B8F3C;
}
.btn-theme-outline:hover {
background-color: #cce0d4;
color: #0B8F3C;
}
.nav-item {
padding: 12px 18px;
text-decoration: none;
color: #262626;
font-weight: 500;
border-bottom: 3px solid transparent;
font-size: 14px;
}
.nav-item:hover {
background: #cce0d4;
color: #0B8F3C;
}
.active-tab {
color: #0B8F3C;
border-bottom: 3px solid #0B8F3C;
background: #e5efe8;
}
.content-area {
padding: 20px;
background: #f8f9fa;
min-height: calc(100vh - 110px);
}
.card {
border-radius: 0px !important;
border: 1px solid #E5E7EB;
}
.chart-container {
position: relative;
height: 400px;
width: 100%;
}
.map-container {
height: 400px;
width: 100%;
position: relative;
}
#provinceMap {
height: 100%;
width: 100%;
}
.form-select {
border-radius: 0px !important;
}
.shadow-sm {
box-shadow: none !important;
}
.card h3 {
color: #0B8F3C;
}
.export-control {
position: relative;
}
#exportItems {
display: flex;
gap: 8px;
opacity: 0;
transform: translateX(-10px);
pointer-events: none;
width: 0;
overflow: hidden;
transition: all 0.2s ease;
}
#exportItems.show {
opacity: 1;
transform: translateX(0);
pointer-events: auto;
width: auto;
}
.export-modal {
display: none;
position: fixed;
inset: 0;
background: rgba(0, 0, 0, 0.4);
z-index: 10000;
}
.export-content {
background: white;
padding: 20px;
width: 400px;
margin: 10% auto;
border-radius: 10px;
}
/* SLIDE FEATURE (from master) */
.slide-wrapper {
position: relative;
overflow: hidden;
height: 100%;
min-height: 400px;
}
.slide {
position: absolute;
width: 100%;
top: 0;
left: 100%;
opacity: 0;
transition: all 0.5s ease-in-out;
}
.slide.active {
left: 0;
opacity: 1;
z-index: 2;
}
.slide.prev {
left: -100%;
opacity: 0;
}
.slide-btn {
position: absolute;
top: 10%;
transform: translateY(-50%);
background: #fff;
color: #0B8F3C;
border: 1px solid #0B8F3C;
padding: 8px 15px;
cursor: pointer;
z-index: 10;
}
.prev-btn {
right: 75px;
}
.next-btn {
right: 25px;
}
.slide-btn:hover {
background: rgba(7, 120, 24, 0.8);
color: #cee6d7;
}
@media print {
#floatingExport {
display: none !important;
}
.nav-bar,
.top-navbar {
display: none !important;
}
.card {
page-break-inside: avoid;
}
}
</style>
</head>
<body>
<div class="top-navbar">
<div class="brand-title">
National Reference Medical Laboratory Surveillance Dashboard
</div>
<div id="lastUpdated" class="ms-auto small">
Last update: --
</div>
</div>
<div class="nav-bar">
<a href="/dashboard" class="nav-item {{ request()->is('dashboard') ? 'active-tab' : '' }}">
Overview
</a>
@foreach($programs as $program)
@if($program->code === 'SEQ')
<a href="/dashboard/seq" class="nav-item {{ request()->is('dashboard/seq') ? 'active-tab' : '' }}">
SEQ
</a>
@else
<a href="/dashboard/{{ strtolower($program->code) }}"
class="nav-item {{ request()->is('dashboard/' . strtolower($program->code)) ? 'active-tab' : '' }}">
{{ $program->code }}
</a>
@endif
@endforeach
<div class="ms-auto d-flex align-items-center gap-2 pe-3">
<button onclick="reloadDataSource()" class="btn btn-sm btn-theme-outline">
Refresh Data
</button>
<div id="exportControl" class="d-flex align-items-center gap-2">
<button id="exportToggle" class="btn btn-sm btn-theme-outline">
Export
</button>
<div id="exportItems" class="align-items-center gap-2">
<button class="btn btn-sm btn-light" onclick="openChartSelector()">Charts</button>
<button class="btn btn-sm btn-light" onclick="exportFullDashboard()">Screen</button>
<button class="btn btn-sm btn-light" onclick="window.print()">Print</button>
<button class="btn btn-sm btn-outline-secondary" id="exportClose"></button>
</div>
<div id="chartModal" class="export-modal">
<div class="export-content">
<h5>Select Charts</h5>
<div id="chartList"></div>
<div class="mt-3 d-flex justify-content-end gap-2">
<button onclick="closeChartSelector()">Cancel</button>
<button onclick="exportSelectedCharts()">Download PDF</button>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="main-wrapper">
<div class="content-area">
@yield('content')
</div>
</div>
<div id="reloadOverlay" style="
display:none;
position:fixed;
inset:0;
background:rgba(255,255,255,0.7);
z-index:99999;
justify-content:center;
align-items:center;
flex-direction:column;
font-size:18px;
font-weight:600;
color:#333;
">
<div class="spinner-border text-primary mb-3"></div>
Updating dashboard data...
</div>
@yield('scripts')
<script>
document.addEventListener("DOMContentLoaded", () => {
updateLastUpdated();
});
window.addEventListener("click", (e) => {
const modal = document.getElementById("chartModal");
if (e.target === modal) modal.style.display = "none";
});
function updateLastUpdated() {
const now = new Date();
const time = now.toLocaleTimeString([], {
hour: '2-digit',
minute: '2-digit'
});
const date = now.toISOString().split('T')[0];
document.getElementById('lastUpdated').innerHTML =
`Last update: ${time} | ${date}`;
}
function reloadDataSource() {
const overlay = document.getElementById('reloadOverlay');
overlay.style.display = 'flex';
fetch('/api/dashboard/reload')
.then(res => res.json())
.then(() => {
updateLastUpdated();
location.reload();
})
.catch(err => {
console.error(err);
overlay.style.display = 'none';
alert('Failed to update dashboard data.');
});
}
</script>
</body>
</html>

View File

@@ -0,0 +1,30 @@
<!DOCTYPE html>
<html lang="{{ str_replace('_', '-', app()->getLocale()) }}">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="csrf-token" content="{{ csrf_token() }}">
<title>{{ config('app.name', 'Laravel') }}</title>
<!-- Fonts -->
<link rel="preconnect" href="https://fonts.bunny.net">
<link href="https://fonts.bunny.net/css?family=figtree:400,500,600&display=swap" rel="stylesheet" />
<!-- Scripts -->
@vite(['resources/css/app.css', 'resources/js/app.js'])
</head>
<body class="font-sans text-gray-900 antialiased">
<div class="min-h-screen flex flex-col sm:justify-center items-center pt-6 sm:pt-0 bg-gray-100 dark:bg-gray-900">
<div>
<a href="/">
<x-application-logo class="w-20 h-20 fill-current text-gray-500" />
</a>
</div>
<div class="w-full sm:max-w-md mt-6 px-6 py-4 bg-white dark:bg-gray-800 shadow-md overflow-hidden sm:rounded-lg">
{{ $slot }}
</div>
</div>
</body>
</html>

View File

@@ -0,0 +1,100 @@
<nav x-data="{ open: false }" class="bg-white dark:bg-gray-800 border-b border-gray-100 dark:border-gray-700">
<!-- Primary Navigation Menu -->
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
<div class="flex justify-between h-16">
<div class="flex">
<!-- Logo -->
<div class="shrink-0 flex items-center">
<a href="{{ route('dashboard') }}">
<x-application-logo class="block h-9 w-auto fill-current text-gray-800 dark:text-gray-200" />
</a>
</div>
<!-- Navigation Links -->
<div class="hidden space-x-8 sm:-my-px sm:ms-10 sm:flex">
<x-nav-link :href="route('dashboard')" :active="request()->routeIs('dashboard')">
{{ __('Dashboard') }}
</x-nav-link>
</div>
</div>
<!-- Settings Dropdown -->
<div class="hidden sm:flex sm:items-center sm:ms-6">
<x-dropdown align="right" width="48">
<x-slot name="trigger">
<button class="inline-flex items-center px-3 py-2 border border-transparent text-sm leading-4 font-medium rounded-md text-gray-500 dark:text-gray-400 bg-white dark:bg-gray-800 hover:text-gray-700 dark:hover:text-gray-300 focus:outline-none transition ease-in-out duration-150">
<div>{{ Auth::user()->name }}</div>
<div class="ms-1">
<svg class="fill-current h-4 w-4" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z" clip-rule="evenodd" />
</svg>
</div>
</button>
</x-slot>
<x-slot name="content">
<x-dropdown-link :href="route('profile.edit')">
{{ __('Profile') }}
</x-dropdown-link>
<!-- Authentication -->
<form method="POST" action="{{ route('logout') }}">
@csrf
<x-dropdown-link :href="route('logout')"
onclick="event.preventDefault();
this.closest('form').submit();">
{{ __('Log Out') }}
</x-dropdown-link>
</form>
</x-slot>
</x-dropdown>
</div>
<!-- Hamburger -->
<div class="-me-2 flex items-center sm:hidden">
<button @click="open = ! open" class="inline-flex items-center justify-center p-2 rounded-md text-gray-400 dark:text-gray-500 hover:text-gray-500 dark:hover:text-gray-400 hover:bg-gray-100 dark:hover:bg-gray-900 focus:outline-none focus:bg-gray-100 dark:focus:bg-gray-900 focus:text-gray-500 dark:focus:text-gray-400 transition duration-150 ease-in-out">
<svg class="h-6 w-6" stroke="currentColor" fill="none" viewBox="0 0 24 24">
<path :class="{'hidden': open, 'inline-flex': ! open }" class="inline-flex" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h16" />
<path :class="{'hidden': ! open, 'inline-flex': open }" class="hidden" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
</svg>
</button>
</div>
</div>
</div>
<!-- Responsive Navigation Menu -->
<div :class="{'block': open, 'hidden': ! open}" class="hidden sm:hidden">
<div class="pt-2 pb-3 space-y-1">
<x-responsive-nav-link :href="route('dashboard')" :active="request()->routeIs('dashboard')">
{{ __('Dashboard') }}
</x-responsive-nav-link>
</div>
<!-- Responsive Settings Options -->
<div class="pt-4 pb-1 border-t border-gray-200 dark:border-gray-600">
<div class="px-4">
<div class="font-medium text-base text-gray-800 dark:text-gray-200">{{ Auth::user()->name }}</div>
<div class="font-medium text-sm text-gray-500">{{ Auth::user()->email }}</div>
</div>
<div class="mt-3 space-y-1">
<x-responsive-nav-link :href="route('profile.edit')">
{{ __('Profile') }}
</x-responsive-nav-link>
<!-- Authentication -->
<form method="POST" action="{{ route('logout') }}">
@csrf
<x-responsive-nav-link :href="route('logout')"
onclick="event.preventDefault();
this.closest('form').submit();">
{{ __('Log Out') }}
</x-responsive-nav-link>
</form>
</div>
</div>
</div>
</nav>