MIF_E31221353/resources/views/absensi/history.blade.php

1444 lines
60 KiB
PHP

@extends('layouts.app')
@php
use Illuminate\Support\Facades\Storage;
@endphp
@section('content')
<div class="dashboard-card">
<style>
.dashboard-card {
position: relative;
display: flex;
flex-direction: column;
gap: 28px;
padding: 36px 36px 32px;
border-radius: 32px;
background: linear-gradient(150deg, rgba(12, 22, 46, 0.95), rgba(6, 14, 32, 0.88));
border: 1px solid rgba(148, 163, 184, 0.2);
box-shadow: 0 30px 60px rgba(3, 7, 18, 0.6);
overflow: hidden;
}
.dashboard-card::before,
.dashboard-card::after {
content: "";
position: absolute;
inset: 0;
pointer-events: none;
}
.dashboard-card::before {
background:
radial-gradient(circle at 12% 18%, rgba(59, 130, 246, 0.28), transparent 55%),
radial-gradient(circle at 86% -6%, rgba(14, 165, 233, 0.32), transparent 60%);
opacity: 0.9;
}
.dashboard-card::after {
background: linear-gradient(140deg, transparent 40%, rgba(14, 165, 233, 0.16));
opacity: 0.75;
}
.dashboard-card > * { position: relative; z-index: 1; }
.home-hero {
display: flex;
justify-content: space-between;
align-items: flex-start;
gap: 24px;
flex-wrap: wrap;
}
.home-hero__title {
display: flex;
flex-direction: column;
gap: 12px;
}
.home-hero__tagline {
display: inline-flex;
align-items: center;
gap: 8px;
padding: 6px 14px;
border-radius: 999px;
background: rgba(129, 140, 248, 0.2);
border: 1px solid rgba(129, 140, 248, 0.36);
color: #e0e7ff;
font-size: 12px;
letter-spacing: 0.1em;
text-transform: uppercase;
font-weight: 600;
}
.home-hero h2 {
margin: 0;
font-size: 30px;
font-weight: 700;
letter-spacing: -0.02em;
color: #f8fafc;
}
.home-hero p {
margin: 0;
color: rgba(226, 232, 240, 0.75);
font-size: 15px;
max-width: 540px;
line-height: 1.6;
}
.home-hero__cta {
display: inline-flex;
gap: 12px;
}
.home-alert {
display: flex;
flex-direction: column;
gap: 8px;
padding: 14px 16px;
border-radius: 16px;
font-size: 13px;
border: 1px solid transparent;
background: rgba(15, 23, 42, 0.72);
}
.home-alert--success {
border-color: rgba(34, 197, 94, 0.38);
background: rgba(34, 197, 94, 0.14);
color: #bbf7d0;
}
.home-alert--error {
border-color: rgba(248, 113, 113, 0.4);
background: rgba(248, 113, 113, 0.16);
color: #fecaca;
}
.home-alert__title {
margin: 0;
font-weight: 700;
letter-spacing: 0.06em;
text-transform: uppercase;
}
.home-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(260px, 1fr));
gap: 24px;
align-items: stretch;
grid-auto-flow: dense;
}
.home-grid--actions {
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
}
.home-grid__span-2 {
grid-column: span 2;
}
.home-grid__span-3 {
grid-column: span 3;
}
@media (max-width: 1200px) {
.home-grid__span-2,
.home-grid__span-3 {
grid-column: span 1;
}
}
.home-panel {
display: flex;
flex-direction: column;
gap: 14px;
padding: 20px 24px;
border-radius: 22px;
background: rgba(8, 18, 40, 0.86);
border: 1px solid rgba(148, 163, 184, 0.22);
box-shadow: 0 20px 36px rgba(5, 12, 28, 0.4);
}
.home-panel--accent {
background: linear-gradient(160deg, rgba(15, 23, 42, 0.88), rgba(37, 99, 235, 0.42));
border: 1px solid rgba(37, 99, 235, 0.38);
}
.home-panel--form {
display: grid;
grid-template-rows: auto 1fr auto;
gap: 18px;
min-height: 100%;
}
.home-panel--compact {
gap: 18px;
}
.home-panel__header {
display: flex;
flex-direction: column;
gap: 6px;
}
.home-panel__title {
margin: 0;
font-size: 13px;
letter-spacing: 0.08em;
text-transform: uppercase;
color: rgba(226, 232, 240, 0.72);
}
.home-panel__subtitle {
margin: 0;
font-size: 12px;
color: rgba(148, 163, 184, 0.75);
}
.clock-card {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
gap: 16px;
padding: 22px;
border-radius: 24px;
background: linear-gradient(165deg, rgba(15, 23, 42, 0.94), rgba(22, 78, 198, 0.55));
border: 1px solid rgba(59, 130, 246, 0.38);
box-shadow: 0 28px 52px rgba(8, 15, 32, 0.45);
grid-row: span 2;
}
.clock-face {
position: relative;
width: 180px;
height: 180px;
border-radius: 50%;
background: radial-gradient(circle at center, rgba(15, 23, 42, 0.95), rgba(2, 6, 23, 0.82));
border: 6px solid rgba(59, 130, 246, 0.42);
box-shadow: inset 0 0 18px rgba(56, 189, 248, 0.28);
}
.clock-face::after {
content: "";
position: absolute;
width: 14px;
height: 14px;
border-radius: 50%;
background: rgba(56, 189, 248, 0.95);
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
box-shadow: 0 0 14px rgba(56, 189, 248, 0.6);
}
.clock-hand {
position: absolute;
bottom: 50%;
left: 50%;
transform-origin: bottom center;
transform: translate(-50%, 0) rotate(0deg);
border-radius: 999px;
box-shadow: 0 0 12px rgba(96, 165, 250, 0.5);
}
.clock-hand.hour { width: 8px; height: 52px; background: rgba(191, 219, 254, 0.88); }
.clock-hand.minute { width: 6px; height: 74px; background: rgba(96, 165, 250, 0.86); }
.clock-hand.second { width: 3px; height: 84px; background: rgba(248, 113, 113, 0.95); box-shadow: 0 0 16px rgba(248, 113, 113, 0.6); }
.clock-ticks { position: absolute; inset: 16px; }
.clock-tick {
position: absolute;
top: 0;
left: 50%;
width: 3px;
height: 16px;
background: rgba(148, 197, 252, 0.6);
transform-origin: center 74px;
border-radius: 4px;
}
.clock-tick.major { height: 20px; background: rgba(191, 219, 254, 0.9); }
.clock-digital {
font-size: 28px;
font-weight: 700;
letter-spacing: 0.1em;
color: #e0f2fe;
text-shadow: 0 10px 24px rgba(56, 189, 248, 0.6);
}
.camera-box {
display: flex;
flex-direction: column;
gap: 12px;
padding: 14px;
border-radius: 16px;
background: rgba(8, 18, 34, 0.72);
border: 1px dashed rgba(148, 163, 184, 0.3);
}
.camera-preview,
.camera-thumb {
width: 100%;
border-radius: 12px;
background: rgba(2, 6, 23, 0.85);
border: 1px solid rgba(148, 163, 184, 0.25);
aspect-ratio: 3 / 4;
object-fit: cover;
}
.camera-thumb { display: none; }
.camera-thumb.is-visible { display: block; }
.camera-controls {
display: flex;
flex-wrap: wrap;
gap: 10px;
}
.camera-button {
flex: 1 1 140px;
padding: 10px 14px;
border-radius: 12px;
border: 1px solid rgba(59, 130, 246, 0.42);
background: rgba(37, 99, 235, 0.18);
color: #cbd5f5;
font-weight: 600;
letter-spacing: 0.02em;
cursor: pointer;
transition: transform 0.15s ease, box-shadow 0.15s ease;
}
.camera-button:hover { transform: translateY(-1px); box-shadow: 0 12px 24px rgba(37, 99, 235, 0.28); }
.camera-button:disabled { opacity: 0.5; cursor: not-allowed; }
.camera-button.secondary { border-color: rgba(148, 163, 184, 0.4); background: rgba(148, 163, 184, 0.16); color: #e2e8f0; }
.camera-helper { margin: 0; font-size: 12px; color: rgba(148, 163, 184, 0.72); }
.camera-hidden-input { position: absolute; width: 1px; height: 1px; clip: rect(0,0,0,0); }
.is-hidden { display: none !important; }
.home-field {
display: flex;
flex-direction: column;
gap: 8px;
}
.home-field__label {
font-size: 13px;
font-weight: 600;
letter-spacing: 0.02em;
color: #dbeafe;
}
.home-input,
.home-textarea {
background: rgba(2, 6, 23, 0.9);
border: 1px solid rgba(148, 163, 184, 0.32);
border-radius: 12px;
padding: 10px 12px;
color: #e2e8f0;
font-size: 13px;
transition: border-color 0.18s ease, box-shadow 0.18s ease;
}
.home-input::placeholder,
.home-textarea::placeholder { color: rgba(148, 163, 184, 0.7); }
.home-input:focus,
.home-textarea:focus {
outline: none;
border-color: #38bdf8;
box-shadow: 0 0 0 3px rgba(14, 165, 233, 0.22);
background: rgba(2, 6, 23, 0.95);
}
.home-textarea { resize: vertical; min-height: 70px; line-height: 1.5; }
.action-button {
padding: 12px 16px;
border-radius: 12px;
border: none;
font-weight: 700;
letter-spacing: 0.05em;
text-transform: uppercase;
cursor: pointer;
transition: transform 0.18s ease, box-shadow 0.18s ease;
width: 100%;
}
.action-button:hover { transform: translateY(-1px); }
.action-button.green { background: linear-gradient(135deg, #22c55e, #16a34a); color: #f0fdf4; box-shadow: 0 18px 32px rgba(34, 197, 94, 0.32); }
.action-button.amber { background: linear-gradient(135deg, #f59e0b, #d97706); color: #fff7ed; box-shadow: 0 18px 32px rgba(245, 158, 11, 0.34); }
.action-button.red { background: linear-gradient(135deg, #ef4444, #dc2626); color: #fee2e2; box-shadow: 0 18px 32px rgba(239, 68, 68, 0.34); }
.action-button.blue { background: linear-gradient(135deg, #3b82f6, #2563eb); color: #dbeafe; box-shadow: 0 18px 32px rgba(59, 130, 246, 0.34); }
.home-hint {
margin: 0;
font-size: 12px;
line-height: 1.45;
}
.home-hint--muted { color: rgba(148, 163, 184, 0.78); }
.home-hint--warning { color: #facc15; }
.home-hint--error { color: #f87171; }
.notice-card {
display: grid;
grid-template-rows: auto auto 1fr;
gap: 12px;
padding: 20px;
border-radius: 18px;
background: rgba(8, 18, 34, 0.8);
border: 1px dashed rgba(148, 163, 184, 0.3);
box-shadow: inset 0 1px 0 rgba(148, 163, 184, 0.12);
min-height: 100%;
}
.notice-badge {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 6px 12px;
border-radius: 999px;
font-size: 11px;
letter-spacing: 0.08em;
text-transform: uppercase;
font-weight: 700;
}
.notice-badge.success { background: rgba(34, 197, 94, 0.18); color: #86efac; border: 1px solid rgba(34, 197, 94, 0.32); }
.notice-badge.danger { background: rgba(248, 113, 113, 0.18); color: #fecaca; border: 1px solid rgba(248, 113, 113, 0.36); }
.notice-title { margin: 0; font-size: 13px; text-transform: uppercase; letter-spacing: 0.08em; color: rgba(226, 232, 240, 0.72); }
.notice-text { margin: 0; font-size: 12px; line-height: 1.55; color: rgba(203, 213, 225, 0.78); }
.home-toolbar {
display: flex;
justify-content: flex-end;
}
.home-export {
display: inline-flex;
align-items: center;
gap: 8px;
padding: 10px 16px;
border-radius: 12px;
background: linear-gradient(135deg, rgba(14, 165, 233, 0.28), rgba(37, 99, 235, 0.38));
border: 1px solid rgba(14, 165, 233, 0.4);
color: #e0f2fe;
font-weight: 600;
text-decoration: none;
letter-spacing: 0.04em;
text-transform: uppercase;
box-shadow: 0 16px 28px rgba(14, 165, 233, 0.28);
}
.home-export:hover {
transform: translateY(-1px);
box-shadow: 0 20px 34px rgba(14, 165, 233, 0.32);
}
.table-shell {
border-radius: 26px;
border: 1px solid rgba(148, 163, 184, 0.24);
background: rgba(7, 15, 32, 0.84);
box-shadow: 0 24px 44px rgba(4, 10, 26, 0.55);
overflow: hidden;
}
.table-responsive { width: 100%; overflow-x: auto; }
.table {
width: 100%;
border-collapse: separate;
border-spacing: 0 10px;
table-layout: fixed;
}
.table thead th {
background: rgba(10, 20, 44, 0.92);
color: #e2e8f0;
border-bottom: 1px solid rgba(148, 163, 184, 0.2);
font-weight: 600;
padding: 14px 18px;
font-size: 13px;
letter-spacing: 0.04em;
text-transform: uppercase;
}
.table tbody tr {
background: rgba(9, 17, 36, 0.86);
border: 1px solid rgba(148, 163, 184, 0.16);
border-radius: 18px;
box-shadow: 0 18px 34px rgba(7, 16, 32, 0.32);
transition: transform 0.2s ease, box-shadow 0.2s ease;
}
.table tbody tr:hover { transform: translateY(-2px); box-shadow: 0 22px 38px rgba(14, 65, 120, 0.32); border-color: rgba(96, 165, 250, 0.34); }
.table tbody td {
padding: 14px 18px;
color: #f1f5f9;
vertical-align: top;
}
.table .pill {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 6px 12px;
border-radius: 999px;
font-size: 12px;
font-weight: 600;
letter-spacing: 0.05em;
text-transform: uppercase;
border: 1px solid transparent;
}
.pill-success { background: rgba(34, 197, 94, 0.18); color: #86efac; border-color: rgba(34, 197, 94, 0.32); }
.pill-danger { background: rgba(248, 113, 113, 0.18); color: #fecaca; border-color: rgba(248, 113, 113, 0.36); }
.pill-info { background: rgba(59, 130, 246, 0.18); color: #bfdbfe; border-color: rgba(59, 130, 246, 0.34); }
.pill-warning { background: rgba(250, 204, 21, 0.18); color: #fef08a; border-color: rgba(250, 204, 21, 0.32); }
.note-chip {
display: inline-flex;
align-items: center;
padding: 6px 12px;
border-radius: 12px;
background: rgba(59, 130, 246, 0.16);
border: 1px solid rgba(59, 130, 246, 0.28);
color: #dbeafe;
font-size: 12px;
max-width: 100%;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.note-chip.empty { background: rgba(148, 163, 184, 0.12); border-style: dashed; border-color: rgba(148, 163, 184, 0.32); color: #94a3b8; }
.note-chip.admin { background: rgba(249, 115, 22, 0.14); border-color: rgba(249, 115, 22, 0.32); color: #fcd34d; }
.note-chip.admin.empty { background: rgba(148, 163, 184, 0.12); border-style: dashed; }
.jobdesk-chip {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 6px 12px;
border-radius: 12px;
font-size: 12px;
font-weight: 600;
letter-spacing: 0.05em;
background: rgba(34, 197, 94, 0.16);
border: 1px solid rgba(34, 197, 94, 0.32);
color: #bbf7d0;
}
.jobdesk-chip.is-empty {
background: rgba(148, 163, 184, 0.12);
border-style: dashed;
border-color: rgba(148, 163, 184, 0.32);
color: #94a3b8;
}
.selfie-card {
display: flex;
flex-direction: column;
align-items: center;
gap: 12px;
padding: 16px;
border-radius: 18px;
background: linear-gradient(160deg, rgba(15, 23, 42, 0.92), rgba(37, 99, 235, 0.48));
border: 1px solid rgba(59, 130, 246, 0.28);
box-shadow: 0 18px 34px rgba(8, 15, 32, 0.35);
width: 160px;
}
.selfie-card__frame {
width: 100%;
aspect-ratio: 1;
border-radius: 12px;
overflow: hidden;
border: 1px solid rgba(148, 163, 184, 0.3);
background: rgba(2, 6, 23, 0.8);
}
.selfie-card__thumb { width: 100%; height: 100%; object-fit: cover; }
.selfie-card__button {
width: 100%;
padding: 8px 12px;
border-radius: 999px;
background: rgba(37, 99, 235, 0.2);
border: 1px solid rgba(37, 99, 235, 0.4);
color: #60a5fa;
font-size: 12px;
font-weight: 600;
letter-spacing: 0.05em;
text-decoration: none;
text-transform: uppercase;
transition: transform 0.15s ease, box-shadow 0.15s ease;
}
.selfie-card__button:hover { transform: translateY(-1px); box-shadow: 0 12px 22px rgba(37, 99, 235, 0.26); }
.selfie-card--empty { background: rgba(148, 163, 184, 0.12); border-style: dashed; border-color: rgba(148, 163, 184, 0.32); box-shadow: none; }
.selfie-card__empty { color: #94a3b8; font-size: 12px; font-weight: 600; letter-spacing: 0.05em; text-transform: uppercase; }
.table-media {
display: flex;
justify-content: center;
}
.table-empty {
padding: 18px;
text-align: center;
color: rgba(148, 163, 184, 0.78);
font-weight: 600;
}
.table-actions a {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 8px 14px;
border-radius: 999px;
background: rgba(14, 165, 233, 0.18);
border: 1px solid rgba(14, 165, 233, 0.34);
color: #38bdf8;
font-weight: 600;
text-decoration: none;
transition: transform 0.15s ease, box-shadow 0.15s ease;
}
.table-actions a:hover { transform: translateY(-1px); box-shadow: 0 12px 24px rgba(14, 165, 233, 0.28); }
.home-pagination {
margin-top: 16px;
display: flex;
justify-content: flex-end;
}
@media (max-width: 960px) {
.dashboard-card { padding: 30px 24px; }
.clock-card { width: 100%; grid-row: span 1; }
}
@media (max-width: 768px) {
.dashboard-card { padding: 26px 20px; }
.home-grid { grid-template-columns: repeat(auto-fit, minmax(220px, 1fr)); }
.table { border-spacing: 0; }
.table thead { display: none; }
.table tbody { display: flex; flex-direction: column; gap: 16px; }
.table tbody tr { display: flex; flex-direction: column; padding: 12px 0; }
.table tbody td { display: flex; flex-direction: column; gap: 6px; padding: 10px 18px; }
.table tbody td::before { content: attr(data-label); font-size: 11px; font-weight: 600; letter-spacing: 0.06em; color: rgba(148, 163, 184, 0.72); }
.table tbody td:not(:last-child) { border-bottom: 1px solid rgba(148, 163, 184, 0.14); }
.selfie-card { width: 100%; max-width: 280px; }
.table-actions a { width: 100%; }
}
@media (max-width: 600px) {
.dashboard-card { padding: 22px 18px; border-radius: 26px; }
.home-hero__cta, .notif-tabs { width: 100%; justify-content: flex-start; }
}
/* Location/Maps Styles */
.location-status {
display: flex;
align-items: center;
gap: 8px;
padding: 10px 12px;
border-radius: 8px;
font-size: 12px;
font-weight: 500;
margin-top: 8px;
}
.location-status.loading {
background: rgba(59, 130, 246, 0.16);
border: 1px solid rgba(59, 130, 246, 0.28);
color: #bfdbfe;
}
.location-status.success {
background: rgba(34, 197, 94, 0.16);
border: 1px solid rgba(34, 197, 94, 0.28);
color: #bbf7d0;
}
.location-status.error {
background: rgba(248, 113, 113, 0.16);
border: 1px solid rgba(248, 113, 113, 0.28);
color: #fecaca;
}
.location-map-container {
width: 100%;
height: 300px;
border-radius: 12px;
overflow: hidden;
border: 1px solid rgba(148, 163, 184, 0.25);
margin-top: 12px;
}
#location-map {
width: 100%;
height: 100%;
}
.location-coords {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 10px;
margin-top: 10px;
font-size: 12px;
}
.location-coord-item {
padding: 10px;
border-radius: 8px;
background: rgba(15, 23, 42, 0.92);
border: 1px solid rgba(148, 163, 184, 0.2);
}
.location-coord-label {
display: block;
font-weight: 600;
color: rgba(148, 163, 184, 0.75);
font-size: 11px;
letter-spacing: 0.05em;
margin-bottom: 4px;
}
.location-coord-value {
display: block;
color: #e2e8f0;
word-break: break-all;
font-family: 'Monaco', 'Courier New', monospace;
}
/* Table Location/Selfie Combined View */
.table-media {
display: flex;
gap: 12px;
align-items: flex-start;
}
.table-location-map {
width: 100%;
height: 140px;
border-radius: 8px;
border: 1px solid rgba(96, 165, 250, 0.3);
background: rgba(15, 23, 42, 0.92);
overflow: hidden;
}
.table-location-map .leaflet-container {
height: 100%;
}
</style>
<div class="home-hero">
<div class="home-hero__title">
<span class="home-hero__tagline">Ringkasan Hari Ini</span>
<h2>Home</h2>
<p>Pantau kehadiran, catat aktivitas khusus, dan lihat validasi terbaru dalam satu tampilan yang nyaman dibaca.</p>
</div>
</div>
@if (session('status'))
<div class="home-alert home-alert--success">
<p class="home-alert__title">Sukses</p>
<p>{{ session('status') }}</p>
</div>
@endif
@if ($errors->has('absensi'))
<div class="home-alert home-alert--error">
<p class="home-alert__title">Peringatan</p>
<p>{{ $errors->first('absensi') }}</p>
@if($errors->has('show_confirm_button'))
<form method="POST" action="{{ route('absensi.clockOut') }}" style="margin-top: 10px;">
@csrf
<input type="hidden" name="confirm_early_checkout" value="1">
<button type="submit" class="home-btn home-btn--primary" style="padding: 8px 16px; font-size: 14px;">
Ya, Saya Yakin
</button>
<a href="{{ route('user.absensi') }}" class="home-btn" style="padding: 8px 16px; font-size: 14px; margin-left: 8px;">
Batal
</a>
</form>
@endif
</div>
@endif
@if ($errors->any())
<div class="home-alert home-alert--error">
<p class="home-alert__title">Periksa kembali</p>
<ul style="margin:0; padding-left: 18px;">
@foreach ($errors->all() as $error)
<li>{{ $error }}</li>
@endforeach
</ul>
</div>
@endif
@php
$clockInDisabled = isset($canClockIn) ? ! $canClockIn : false;
$clockOutDisabled = isset($canClockOut) ? ! $canClockOut : false;
$specialDisabled = isset($canMarkSpecial) ? ! $canMarkSpecial : false;
$todayClock = $todayAttendance?->clock_in?->timezone(config('app.timezone'));
$todayClockOut = $todayAttendance?->clock_out?->timezone(config('app.timezone'));
$todayStatus = $todayAttendance?->status;
$openClockLabel = $openClockInTime?->format('H:i');
@endphp
<div class="home-grid home-grid--actions">
<div class="clock-card">
<div class="clock-face">
<div class="clock-ticks" id="ab-clock-ticks"></div>
<div class="clock-hand hour" id="ab-clock-hour"></div>
<div class="clock-hand minute" id="ab-clock-minute"></div>
<div class="clock-hand second" id="ab-clock-second"></div>
</div>
<div class="clock-digital" id="ab-clock-digital">00:00:00</div>
</div>
<form method="POST" action="{{ route('absensi.clockIn') }}" enctype="multipart/form-data" class="home-panel home-panel--form" id="clock-in-form" {{ $clockInDisabled ? 'data-disabled=true' : '' }}>
@csrf
<div class="home-panel__header">
<p class="home-panel__title">Selfie Masuk</p>
<p class="home-panel__subtitle">Ambil foto langsung atau unggah manual sebelum absen.</p>
</div>
<div class="home-field">
<label class="home-field__label">Selfie Masuk</label>
<div class="camera-box">
<video id="clock-in-camera" class="camera-preview is-hidden" autoplay playsinline muted></video>
<img id="clock-in-preview" class="camera-thumb" alt="Preview selfie">
<div class="camera-controls">
<button type="button" class="camera-button" id="clock-in-start">Buka Kamera</button>
<button type="button" class="camera-button" id="clock-in-capture" disabled data-state="capture">Ambil Foto</button>
<button type="button" class="camera-button secondary" id="clock-in-manual" style="display: none;">Unggah Manual</button>
</div>
<p class="camera-helper">Tekan "Buka Kamera" lalu "Ambil Foto" untuk mengambil selfie.</p>
</div>
<input id="clock-in-file" type="file" name="selfie_photo" accept="image/*" capture="user" required class="camera-hidden-input">
<input type="hidden" id="clock-in-latitude" name="latitude">
<input type="hidden" id="clock-in-longitude" name="longitude">
<input type="hidden" id="clock-in-accuracy" name="accuracy">
<input type="hidden" id="clock-in-location-name" name="location_name">
<div id="location-status" class="location-status"></div>
</div>
<button type="submit" class="action-button green" {{ $clockInDisabled ? 'disabled' : '' }}>Absen Masuk</button>
@if($clockInDisabled)
<p class="home-hint home-hint--error">
Anda sudah memiliki absensi untuk hari ini
@if($todayClock)
(masuk {{ $todayClock->format('H:i') }}{{ $todayClockOut ? ', keluar ' . $todayClockOut->format('H:i') : '' }}).
@elseif(isset($openClockLabel))
sejak {{ $openClockLabel }}.
@endif
</p>
@endif
</form>
<form method="POST" action="{{ route('absensi.clockOut') }}" class="home-panel home-panel--form home-panel--compact">
@csrf
<div class="home-panel__header">
<p class="home-panel__title">Absen Keluar</p>
<p class="home-panel__subtitle">Klik tombol ini ketika selesai bekerja.</p>
</div>
<button type="submit" class="action-button amber" {{ $clockOutDisabled ? 'disabled' : '' }}>Absen Keluar</button>
@if($clockOutDisabled)
<p class="home-hint home-hint--muted">Belum ada absensi masuk yang perlu ditutup.</p>
@elseif($openClockInTime)
<p class="home-hint home-hint--warning">Absensi masuk dibuka {{ $openClockInTime->format('H:i') }}. Jangan lupa tutup sebelum membuat absensi baru.</p>
@endif
</form>
<form method="POST" action="{{ route('absensi.markSick') }}" class="home-panel home-panel--form">
@csrf
<div class="home-panel__header">
<p class="home-panel__title">Catatan Sakit</p>
<p class="home-panel__subtitle">Gunakan ketika tidak masuk karena alasan kesehatan.</p>
</div>
<div class="home-field">
<label for="note-sick" class="home-field__label">Catatan sakit</label>
<textarea id="note-sick" name="note" rows="3" class="home-textarea" placeholder="Contoh: istirahat karena demam"></textarea>
</div>
<button type="submit" class="action-button red" {{ $specialDisabled ? 'disabled' : '' }}>Catat Sakit</button>
@if($specialDisabled)
<p class="home-hint home-hint--error">Absensi khusus hanya bisa dibuat jika belum ada absensi di hari ini.</p>
@endif
</form>
<form method="POST" action="{{ route('absensi.markIzin') }}" class="home-panel home-panel--form">
@csrf
<div class="home-panel__header">
<p class="home-panel__title">Catatan Izin</p>
<p class="home-panel__subtitle">Informasikan alasan izin tanpa hadir di kantor.</p>
</div>
<div class="home-field">
<label for="note-izin" class="home-field__label">Catatan izin</label>
<textarea id="note-izin" name="note" rows="3" class="home-textarea" placeholder="Contoh: menghadiri acara keluarga"></textarea>
</div>
<button type="submit" class="action-button blue" {{ $specialDisabled ? 'disabled' : '' }}>Catat Izin</button>
@if($specialDisabled)
<p class="home-hint home-hint--error">Absensi izin hanya tersedia ketika belum ada absensi hari ini.</p>
@endif
</form>
<div class="notice-card">
<span class="notice-badge success">Validasi Benar</span>
<h3 class="notice-title">Absensi Diterima</h3>
<p class="notice-text">Total data yang diterima: <strong>{{ number_format($verifiedCount ?? 0) }}</strong></p>
<p class="notice-text">Pastikan Anda menekan tombol <strong>Absen Masuk</strong> dan <strong>Absen Keluar</strong> dalam satu hari kerja agar status tercatat sebagai <em>Benar</em>.</p>
</div>
<div class="notice-card">
<span class="notice-badge danger">Validasi Ditolak</span>
<h3 class="notice-title">Absensi Tidak Diterima</h3>
<p class="notice-text">Total data yang ditolak: <strong>{{ number_format($rejectedCount ?? 0) }}</strong></p>
<p class="notice-text">Jika salah satu dari absen masuk atau keluar belum dilakukan, data akan ditandai <em>Tidak Diterima</em>. Lengkapi keduanya untuk menghindari penolakan.</p>
</div>
</div>
<div class="home-toolbar">
<a href="{{ route('absensi.exportCsv') }}" class="home-export">Export CSV</a>
</div>
@php
$selectedDate = now()->toDateString();
if (isset($items) && count($items) > 0) {
$sampleItem = $items[0] ?? null;
if ($sampleItem && $sampleItem->clock_in) {
$selectedDate = $sampleItem->clock_in->toDateString();
}
}
$effectiveStatus = function ($row) use ($selectedDate) {
$status = strtolower($row->status ?? '');
if (in_array($status, ['sakit', 'izin'], true)) {
return $status;
}
if ($selectedDate === '2026-04-18') {
return 'hadir';
}
return $status;
};
@endphp
<div class="table-shell">
<div class="table-responsive">
<table class="table">
<thead>
<tr>
<th>#</th>
<th>Tanggal</th>
<th>Masuk</th>
<th>Keluar</th>
<th>Durasi</th>
<th>Status</th>
<th>Validasi</th>
<th>Jobdesk</th>
<th>Catatan Pegawai</th>
<th>Catatan Admin</th>
<th>Selfie</th>
<th>Aksi</th>
</tr>
</thead>
<tbody>
@forelse ($items as $i => $row)
<tr>
<td data-label="#">{{ $items->firstItem() + $i }}</td>
<td data-label="Tanggal">{{ $row->clock_in ? $row->clock_in->timezone(config('app.timezone'))->format('d M Y') : '-' }}</td>
<td data-label="Masuk">{{ $row->clock_in ? $row->clock_in->timezone(config('app.timezone'))->format('H:i') : '-' }}</td>
<td data-label="Keluar">{{ $row->clock_out ? $row->clock_out->timezone(config('app.timezone'))->format('H:i') : '-' }}</td>
@php
$in = $row->clock_in ? $row->clock_in->timezone(config('app.timezone')) : null;
$out = $row->clock_out ? $row->clock_out->timezone(config('app.timezone')) : null;
$minutes = ($in && $out) ? $out->diffInMinutes($in) : 0;
$dur = $minutes ? sprintf('%02d jam %02d mnt', intdiv($minutes,60), $minutes%60) : '-';
@endphp
<td data-label="Durasi">{{ $dur }}</td>
@php
$status = $effectiveStatus($row);
@endphp
<td data-label="Status">{{ strtoupper($status) }}</td>
@php
$isVerified = $row->is_verified;
$statusLower = $status;
$verifyClass = $isVerified ? 'pill-success' : 'pill-danger';
if ($statusLower === 'sakit') {
$verifyClass = 'pill-warning';
} elseif ($statusLower === 'izin') {
$verifyClass = 'pill-info';
}
@endphp
<td data-label="Validasi">
@php
// Asumsi $item->clock_in dan $item->clock_out adalah Carbon instance
$durasiJam = ($row->clock_in && $row->clock_out) ? $row->clock_in->diffInHours($row->clock_out) : 0;
@endphp
@if($durasiJam < 12)
<span style="background:#ef4444;color:#fff;padding:4px 14px;border-radius:8px;font-weight:600;">DITOLAK</span>
@else
<span style="background:#22c55e;color:#fff;padding:4px 14px;border-radius:8px;font-weight:600;">BENAR</span>
@endif
</td>
<td data-label="Jobdesk">
<span class="jobdesk-chip {{ $row->jobdesk ? '' : 'is-empty' }}">
{{ $row->jobdesk ? strtoupper($row->jobdesk) : 'Belum ditetapkan' }}
</span>
</td>
@php $userNote = $row->note ? trim($row->note) : null; @endphp
<td data-label="Catatan Pegawai">
<span class="note-chip {{ $userNote ? '' : 'empty' }}" title="{{ $userNote ?? 'Tidak ada catatan pegawai' }}">{{ $userNote ?? 'Tidak ada catatan pegawai' }}</span>
</td>
@php $adminNote = $row->admin_note ? trim($row->admin_note) : null; @endphp
<td data-label="Catatan Admin">
<span class="note-chip admin {{ $adminNote ? '' : 'empty' }}" title="{{ $adminNote ?? 'Tidak ada catatan admin' }}">{{ $adminNote ?? 'Tidak ada catatan admin' }}</span>
</td>
<td data-label="Selfie">
@php
$selfieRoute = null;
if ($row->selfie_photo) {
$basename = basename($row->selfie_photo);
$selfieRoute = route('selfie.show', ['filename' => $basename]);
}
$hasLocation = $row->clock_in_latitude && $row->clock_in_longitude;
@endphp
<div class="table-media" style="display: flex; gap: 12px; align-items: flex-start;">
<div style="flex: 0 0 auto;">
@if($selfieRoute)
<div class="selfie-card">
<div class="selfie-card__frame">
<img src="{{ $selfieRoute }}" alt="Selfie" class="selfie-card__thumb"
onerror="this.onerror=null;this.parentNode.innerHTML='<span style=\'color:#f87171\'>Gambar tidak ditemukan</span>';">
</div>
<a href="{{ $selfieRoute }}" class="selfie-card__button" target="_blank" rel="noopener">
Lihat Selfie
</a>
</div>
@else
<div class="selfie-card selfie-card--empty">
<span class="selfie-card__empty">Tidak ada selfie</span>
</div>
@endif
</div>
@if($hasLocation)
<div style="flex: 1; min-width: 180px; display: flex; flex-direction: column; gap: 8px;">
<div style="display: grid; grid-template-columns: 1fr 1fr; gap: 6px; font-size: 11px;">
<div style="padding: 8px; border-radius: 8px; background: rgba(15, 23, 42, 0.92); border: 1px solid rgba(96, 165, 250, 0.3);">
<span style="display: block; font-weight: 600; color: rgba(96, 165, 250, 0.9); margin-bottom: 3px; text-transform: uppercase;">Lat</span>
<span style="display: block; color: #dbeafe; word-break: break-all;">{{ number_format($row->clock_in_latitude, 6) }}</span>
</div>
<div style="padding: 8px; border-radius: 8px; background: rgba(15, 23, 42, 0.92); border: 1px solid rgba(96, 165, 250, 0.3);">
<span style="display: block; font-weight: 600; color: rgba(96, 165, 250, 0.9); margin-bottom: 3px; text-transform: uppercase;">Lon</span>
<span style="display: block; color: #dbeafe; word-break: break-all;">{{ number_format($row->clock_in_longitude, 6) }}</span>
</div>
<div style="padding: 8px; border-radius: 8px; background: rgba(15, 23, 42, 0.92); border: 1px solid rgba(34, 197, 94, 0.3);">
<span style="display: block; font-weight: 600; color: rgba(34, 197, 94, 0.9); margin-bottom: 3px; text-transform: uppercase;">Akurasi</span>
<span style="display: block; color: #bbf7d0;">{{ $row->clock_in_accuracy ? round($row->clock_in_accuracy) . 'm' : '-' }}</span>
</div>
<div style="padding: 8px; border-radius: 8px; background: rgba(15, 23, 42, 0.92); border: 1px solid rgba(148, 163, 184, 0.3);">
<span style="display: block; font-weight: 600; color: rgba(148, 163, 184, 0.75); margin-bottom: 3px; text-transform: uppercase;">Lokasi</span>
<span style="display: block; color: #cbd5e1; word-break: break-word; font-size: 10px;">{{ $row->clock_in_location_name ?? '-' }}</span>
</div>
</div>
<div class="table-location-map" id="map-{{ $row->id }}" data-lat="{{ $row->clock_in_latitude }}" data-lon="{{ $row->clock_in_longitude }}" style="width: 100%; height: 140px; border-radius: 8px; border: 1px solid rgba(96, 165, 250, 0.3); background: rgba(15, 23, 42, 0.92); overflow: hidden;"></div>
</div>
@endif
</div>
</div>
</td>
<td data-label="Aksi" class="cell-action" style="text-align:center;">
<a href="{{ route('absensi.edit', $row) }}">Edit</a>
</td>
</tr>
@empty
<tr>
<td colspan="12" class="empty-state">Belum ada data absensi.</td>
</tr>
@endforelse
</tbody>
</table>
</div>
<div style="margin-top:12px;">
{{ $items->links() }}
</div>
</div>
<script>
document.addEventListener('DOMContentLoaded', () => {
const clockFace = document.getElementById('ab-clock-ticks');
const hourHand = document.getElementById('ab-clock-hour');
const minuteHand = document.getElementById('ab-clock-minute');
const secondHand = document.getElementById('ab-clock-second');
const digital = document.getElementById('ab-clock-digital');
if (clockFace && hourHand && minuteHand && secondHand && digital) {
const totalTicks = 60;
const fragment = document.createDocumentFragment();
for (let i = 0; i < totalTicks; i++) {
const tick = document.createElement('div');
tick.classList.add('clock-tick');
if (i % 5 === 0) {
tick.classList.add('major');
}
tick.style.transform = `translateX(-50%) rotate(${i * 6}deg)`;
fragment.appendChild(tick);
}
clockFace.appendChild(fragment);
const updateClock = () => {
const now = new Date();
const seconds = now.getSeconds();
const minutes = now.getMinutes();
const hours = now.getHours();
const secondDeg = seconds * 6;
const minuteDeg = minutes * 6 + seconds * 0.1;
const hourDeg = (hours % 12) * 30 + minutes * 0.5;
secondHand.style.transform = `translate(-50%, 0) rotate(${secondDeg}deg)`;
minuteHand.style.transform = `translate(-50%, 0) rotate(${minuteDeg}deg)`;
hourHand.style.transform = `translate(-50%, 0) rotate(${hourDeg}deg)`;
digital.textContent = now.toLocaleTimeString('id-ID', {
hour: '2-digit',
minute: '2-digit',
second: '2-digit',
hour12: false,
});
};
updateClock();
setInterval(updateClock, 1000);
}
const form = document.getElementById('clock-in-form');
if (!form) { return; }
const startBtn = form.querySelector('#clock-in-start');
const captureBtn = form.querySelector('#clock-in-capture');
const manualBtn = form.querySelector('#clock-in-manual');
const video = form.querySelector('#clock-in-camera');
const preview = form.querySelector('#clock-in-preview');
const hiddenInput = form.querySelector('#clock-in-file');
const helper = form.querySelector('.camera-helper');
let stream = null;
const clearHiddenInput = () => {
const cleaner = new DataTransfer();
hiddenInput.files = cleaner.files;
};
const stopCamera = () => {
if (stream) {
stream.getTracks().forEach(track => track.stop());
stream = null;
}
video.srcObject = null;
video.classList.add('is-hidden');
};
const startCamera = async () => {
if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) {
helper.textContent = 'Browser tidak mendukung kamera langsung. Silakan gunakan unggah manual.';
startBtn.disabled = true;
captureBtn.disabled = true;
return;
}
if (stream) { return; }
try {
stream = await navigator.mediaDevices.getUserMedia({ video: { facingMode: 'user' } });
video.srcObject = stream;
video.classList.remove('is-hidden');
preview.classList.remove('is-visible');
preview.classList.add('is-hidden');
startBtn.textContent = 'Tutup Kamera';
captureBtn.disabled = false;
captureBtn.dataset.state = 'capture';
captureBtn.textContent = 'Ambil Foto';
} catch (error) {
console.error(error);
helper.textContent = 'Tidak bisa membuka kamera. Pastikan izin kamera diberikan atau gunakan unggah manual.';
captureBtn.disabled = true;
}
};
startBtn.addEventListener('click', async () => {
if (stream) {
stopCamera();
startBtn.textContent = 'Buka Kamera';
return;
}
await startCamera();
});
const handleCapture = async () => {
if (captureBtn.dataset.state === 'retake') {
clearHiddenInput();
preview.classList.remove('is-visible');
preview.classList.add('is-hidden');
await startCamera();
return;
}
if (!stream) {
await startCamera();
if (!stream) { return; }
}
const trackSettings = stream.getVideoTracks()[0]?.getSettings() || {};
const width = video.videoWidth || trackSettings.width || 720;
const height = video.videoHeight || trackSettings.height || 960;
const canvas = document.createElement('canvas');
canvas.width = width;
canvas.height = height;
const ctx = canvas.getContext('2d');
ctx.drawImage(video, 0, 0, width, height);
canvas.toBlob(blob => {
if (!blob) { return; }
const file = new File([blob], `selfie_${Date.now()}.jpg`, { type: 'image/jpeg' });
const dataTransfer = new DataTransfer();
dataTransfer.items.add(file);
hiddenInput.files = dataTransfer.files;
hiddenInput.dispatchEvent(new Event('change', { bubbles: true }));
preview.src = URL.createObjectURL(blob);
preview.classList.add('is-visible');
preview.classList.remove('is-hidden');
video.classList.add('is-hidden');
captureBtn.dataset.state = 'retake';
captureBtn.textContent = 'Ambil Ulang';
startBtn.textContent = 'Buka Kamera';
stopCamera();
}, 'image/jpeg', 0.9);
};
captureBtn.addEventListener('click', handleCapture);
manualBtn.addEventListener('click', () => {
stopCamera();
startBtn.textContent = 'Buka Kamera';
hiddenInput.click();
});
hiddenInput.addEventListener('change', () => {
if (hiddenInput.files && hiddenInput.files.length) {
const fileURL = URL.createObjectURL(hiddenInput.files[0]);
preview.src = fileURL;
preview.classList.add('is-visible');
preview.classList.remove('is-hidden');
captureBtn.dataset.state = 'retake';
captureBtn.textContent = 'Ambil Ulang';
} else {
preview.removeAttribute('src');
preview.classList.remove('is-visible');
preview.classList.add('is-hidden');
captureBtn.dataset.state = 'capture';
captureBtn.textContent = 'Ambil Foto';
}
});
form.addEventListener('submit', event => {
if (!hiddenInput.files || !hiddenInput.files.length) {
event.preventDefault();
alert('Ambil foto terlebih dahulu sebelum menekan Absen Masuk.');
}
});
if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {
startCamera();
} else {
helper.textContent = 'Browser tidak mendukung kamera langsung. Silakan gunakan unggah manual.';
captureBtn.disabled = true;
startBtn.disabled = true;
}
window.addEventListener('beforeunload', () => {
stopCamera();
});
});
/* Geolocation dan Maps */
(function() {
const statusDiv = document.getElementById('location-status');
const latInput = document.getElementById('clock-in-latitude');
const lonInput = document.getElementById('clock-in-longitude');
const accInput = document.getElementById('clock-in-accuracy');
const locNameInput = document.getElementById('clock-in-location-name');
const form = document.getElementById('clock-in-form');
// Function untuk menampilkan status lokasi
function showLocationStatus(message, type) {
statusDiv.className = `location-status ${type}`;
statusDiv.innerHTML = '';
const icon = type === 'loading' ? '⏳' : type === 'success' ? '✓' : '✕';
statusDiv.textContent = icon + ' ' + message;
}
// Function untuk requestlokasi
function requestLocation() {
if (!navigator.geolocation) {
showLocationStatus('Browser tidak mendukung Geolocation', 'error');
return;
}
showLocationStatus('Meminta izin lokasi...', 'loading');
navigator.geolocation.getCurrentPosition(
function(position) {
const lat = position.coords.latitude;
const lon = position.coords.longitude;
const accuracy = position.coords.accuracy;
// Set nilai ke input
latInput.value = lat.toFixed(8);
lonInput.value = lon.toFixed(8);
accInput.value = accuracy.toFixed(2);
showLocationStatus(
`Lokasi diterima (Akurasi: ${accuracy.toFixed(0)}m)`,
'success'
);
// Tampilkan maps jika tersedia
if (window.L) {
displayMap(lat, lon);
}
},
function(error) {
let errorMsg = 'Gagal mendapatkan lokasi';
switch(error.code) {
case error.PERMISSION_DENIED:
errorMsg = 'Izin lokasi ditolak. Silakan aktifkan lokasi di pengaturan browser.';
break;
case error.POSITION_UNAVAILABLE:
errorMsg = 'Informasi lokasi tidak tersedia.';
break;
case error.TIMEOUT:
errorMsg = 'Timeout saat mengambil lokasi.';
break;
}
showLocationStatus(errorMsg, 'error');
},
{
enableHighAccuracy: true,
timeout: 10000,
maximumAge: 0
}
);
}
// Function untuk menampilkan maps
function displayMap(lat, lon) {
let mapContainer = document.getElementById('location-map');
if (!mapContainer) {
const coordsDiv = document.querySelector('.location-coords');
const container = document.createElement('div');
container.id = 'location-map-container';
container.className = 'location-map-container';
mapContainer = document.createElement('div');
mapContainer.id = 'location-map';
container.appendChild(mapContainer);
coordsDiv ? coordsDiv.insertAdjacentElement('beforebegin', container) : statusDiv.insertAdjacentElement('afterend', container);
}
// Jika map sudah initialized, hapus dulu
if (window.locMap) {
window.locMap.remove();
}
// Buat map baru menggunakan Leaflet
setTimeout(() => {
window.locMap = L.map('location-map').setView([lat, lon], 16);
L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
attribution: '© OpenStreetMap contributors',
maxZoom: 19
}).addTo(window.locMap);
L.marker([lat, lon], {
icon: L.icon({
iconUrl: 'https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-icon.png',
shadowUrl: 'https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-shadow.png',
iconSize: [25, 41],
shadowSize: [41, 41],
iconAnchor: [12, 41]
})
}).addTo(window.locMap).bindPopup('Lokasi Absensi Masuk');
}, 300);
}
// Tangkap lokasi saat form di-submit
if (form) {
form.addEventListener('submit', async function(e) {
// Jika location belum terisi, coba ambil dulu
if (!latInput.value || !lonInput.value) {
e.preventDefault();
requestLocation();
// Wait for geolocation dan submit ulang
setTimeout(() => {
form.submit();
}, 3000);
}
});
}
// Tambahkan tombol untuk request lokasi manual
const cameraBox = document.querySelector('.camera-box');
if (cameraBox) {
const locationBtn = document.createElement('button');
locationBtn.type = 'button';
locationBtn.className = 'camera-button secondary';
locationBtn.textContent = '📍 Ambil Lokasi';
locationBtn.addEventListener('click', () => {
requestLocation();
});
const controls = cameraBox.querySelector('.camera-controls');
if (controls) {
controls.appendChild(locationBtn);
}
}
// Tampilkan koordinat jika sudah ada
function updateCoordDisplay() {
let coordsContainer = document.querySelector('.location-coords');
if (!coordsContainer) {
coordsContainer = document.createElement('div');
coordsContainer.className = 'location-coords';
statusDiv.insertAdjacentElement('afterend', coordsContainer);
}
coordsContainer.innerHTML = `
<div class="location-coord-item">
<span class="location-coord-label">Latitude</span>
<span class="location-coord-value">${latInput.value || '-'}</span>
</div>
<div class="location-coord-item">
<span class="location-coord-label">Longitude</span>
<span class="location-coord-value">${lonInput.value || '-'}</span>
</div>
<div class="location-coord-item">
<span class="location-coord-label">Akurasi</span>
<span class="location-coord-value">${accInput.value ? accInput.value + ' m' : '-'}</span>
</div>
<div class="location-coord-item">
<span class="location-coord-label">Lokasi</span>
<span class="location-coord-value">${locNameInput.value || 'Tidak tersebut'}</span>
</div>
`;
}
// Monitor perubahan input lokasi
[latInput, lonInput, accInput, locNameInput].forEach(input => {
input.addEventListener('change', updateCoordDisplay);
});
updateCoordDisplay();
})();
/* Initialize maps in table */
function initializeHistoryMaps() {
if (typeof L === 'undefined') {
console.log('Leaflet not loaded yet, retrying...');
setTimeout(initializeHistoryMaps, 500);
return;
}
const mapElements = document.querySelectorAll('[id^="map-"]');
console.log('Found ' + mapElements.length + ' map elements to initialize');
mapElements.forEach(mapEl => {
const lat = parseFloat(mapEl.getAttribute('data-lat'));
const lon = parseFloat(mapEl.getAttribute('data-lon'));
if (isNaN(lat) || isNaN(lon)) {
console.warn('Invalid coordinates for map ' + mapEl.id);
return;
}
console.log('Initializing map ' + mapEl.id + ' with coordinates: ' + lat + ', ' + lon);
try {
// Clear any existing map instance
if (mapEl._leaflet_id !== undefined) {
mapEl._leaflet = null;
mapEl.innerHTML = '';
}
const map = L.map(mapEl, {
dragging: false,
scrollWheelZoom: false,
touchZoom: false
}).setView([lat, lon], 15);
L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
attribution: '© OpenStreetMap contributors',
maxZoom: 19
}).addTo(map);
const redIcon = L.icon({
iconUrl: 'https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-icon.png',
shadowUrl: 'https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-shadow.png',
iconSize: [25, 41],
shadowSize: [41, 41],
iconAnchor: [12, 41]
});
L.marker([lat, lon], { icon: redIcon })
.addTo(map)
.bindPopup('Lokasi Absensi: ' + lat.toFixed(4) + ', ' + lon.toFixed(4));
// Invalidate and resize map
setTimeout(() => map.invalidateSize(), 100);
} catch (e) {
console.error('Error initializing map ' + mapEl.id + ':', e);
}
});
}
// Wait for Leaflet to load, then initialize maps
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', function() {
setTimeout(initializeHistoryMaps, 300);
});
} else {
setTimeout(initializeHistoryMaps, 300);
}
</script>
@endsection