MIF_E31232303/resources/views/admin/rule-basis/index.blade.php

348 lines
16 KiB
PHP

@extends('layouts.admin-app')
@section('page-title', 'Manajemen Rule Basis')
@section('page-subtitle', 'Kelola aturan diagnosa sistem pakar')
@push('styles')
<link href="https://fonts.googleapis.com/css2?family=Plus+Jakarta+Sans:wght@400;500;600;700;800&display=swap" rel="stylesheet">
<style>
:root {
--green-deep: #1a4d2e;
--green-mid: #2d7a4f;
--green-bright:#4ade80;
--bg-base: #f0f5f1;
--bg-card: #ffffff;
--text-primary:#0f1f14;
--text-muted: #6b7f72;
--shadow: 0 4px 24px rgba(29,77,46,.08), 0 1px 4px rgba(29,77,46,.05);
--radius: 20px;
--input-border:#d1d5db;
--input-focus: #2d7a4f;
}
body, .index-root { font-family: 'Plus Jakarta Sans', sans-serif; }
@keyframes fadeUp {
from { opacity:0; transform:translateY(12px); }
to { opacity:1; transform:translateY(0); }
}
.anim { animation: fadeUp .5s cubic-bezier(.22,1,.36,1) both; }
.a1 { animation-delay:.04s; }
.a2 { animation-delay:.12s; }
.a3 { animation-delay:.20s; }
/* ── alert ── */
.alert-success {
display:flex; align-items:center; gap:.75rem;
padding:.9rem 1.2rem; margin-bottom:1.25rem;
background:#f0fdf4; border:1.5px solid #bbf7d0;
border-radius:14px; font-size:.85rem; font-weight:600; color:#166534;
}
/* ── main card ── */
.index-card {
background:var(--bg-card);
border-radius:var(--radius);
border:1.5px solid rgba(29,77,46,.07);
box-shadow:var(--shadow);
overflow:hidden;
}
/* ── toolbar ── */
.toolbar {
display:flex; align-items:center; justify-content:space-between;
padding:1.4rem 1.75rem;
border-bottom:1.5px solid rgba(29,77,46,.07);
flex-wrap:wrap; gap:.75rem;
}
.toolbar-title-bar { display:inline-block; width:3px; height:16px; background:linear-gradient(180deg,var(--green-mid),var(--green-bright)); border-radius:2px; vertical-align:middle; margin-right:.4rem; }
.toolbar-title { font-size:1rem; font-weight:700; color:var(--text-primary); letter-spacing:-.01em; }
.btn-add {
display:inline-flex; align-items:center; gap:.45rem;
padding:.6rem 1.25rem;
background:linear-gradient(135deg,#2d7a4f,#16a34a);
color:#fff; font-size:.82rem; font-weight:700;
border-radius:50px; text-decoration:none;
box-shadow:0 4px 14px rgba(22,163,74,.25); transition:all .2s;
}
.btn-add:hover { transform:translateY(-1px); box-shadow:0 6px 20px rgba(22,163,74,.35); }
/* ── filter bar ── */
.filter-bar {
padding:1.1rem 1.75rem;
background:#fafcfb;
border-bottom:1.5px solid rgba(29,77,46,.07);
display:flex; gap:1rem; flex-wrap:wrap; align-items:flex-end;
}
.filter-group { display:flex; flex-direction:column; gap:.35rem; flex:1; min-width:160px; }
.filter-label { font-size:.72rem; font-weight:700; text-transform:uppercase; letter-spacing:.07em; color:var(--text-muted); }
.filter-select, .filter-input {
padding:.6rem .9rem;
border:1.5px solid var(--input-border); border-radius:10px;
font-size:.82rem; font-family:'Plus Jakarta Sans',sans-serif;
color:var(--text-primary); background:#fff; outline:none;
transition:border-color .15s, box-shadow .15s;
}
.filter-select:focus, .filter-input:focus {
border-color:var(--input-focus);
box-shadow:0 0 0 3px rgba(45,122,79,.1);
}
.filter-actions { display:flex; gap:.5rem; align-items:flex-end; }
.btn-filter {
display:inline-flex; align-items:center; gap:.35rem;
padding:.6rem 1.1rem;
background:#f0fdf4; border:1.5px solid #bbf7d0;
color:var(--green-mid); font-size:.8rem; font-weight:700;
border-radius:10px; cursor:pointer; transition:all .15s;
}
.btn-filter:hover { background:#dcfce7; border-color:var(--green-mid); }
.btn-reset {
display:inline-flex; align-items:center; gap:.35rem;
padding:.6rem 1rem;
background:#f9fafb; border:1.5px solid #e5e7eb;
color:var(--text-muted); font-size:.8rem; font-weight:600;
border-radius:10px; text-decoration:none; transition:all .15s;
}
.btn-reset:hover { background:#f3f4f6; }
/* ── table ── */
.table-wrap { overflow-x:auto; }
table { width:100%; border-collapse:collapse; }
thead tr { border-bottom:1.5px solid rgba(29,77,46,.08); }
thead th {
padding:.75rem 1.1rem;
text-align:left; font-size:.67rem; font-weight:700;
text-transform:uppercase; letter-spacing:.08em;
color:var(--text-muted); background:#fafcfb; white-space:nowrap;
}
thead th.center { text-align:center; }
tbody tr { border-bottom:1px solid rgba(29,77,46,.05); transition:background .15s; }
tbody tr:last-child { border-bottom:none; }
tbody tr:hover { background:#fafcfb; }
td { padding:.85rem 1.1rem; vertical-align:middle; }
.td-no { font-size:.78rem; color:var(--text-muted); font-weight:500; }
/* id badge */
.id-pill {
display:inline-block; padding:.18rem .6rem;
background:#f0fdf4; border:1px solid #bbf7d0;
color:var(--green-mid); font-size:.68rem; font-weight:700;
border-radius:6px; font-family:'DM Mono',monospace; letter-spacing:.02em;
margin-bottom:.3rem;
}
.cell-name { font-size:.82rem; font-weight:600; color:var(--text-primary); }
.cell-sub { font-size:.75rem; color:var(--text-muted); }
/* numeric badge */
.num-badge {
display:inline-block; padding:.25rem .65rem;
background:#f9fafb; border:1.5px solid #e5e7eb;
color:var(--text-primary); font-size:.8rem; font-weight:700;
border-radius:8px; font-family:'DM Mono',monospace;
min-width:52px; text-align:center;
}
/* cf badge color by value */
.cf-high { background:#f0fdf4; border-color:#bbf7d0; color:#166534; }
.cf-mid { background:#fefce8; border-color:#fde68a; color:#a16207; }
.cf-low { background:#f9fafb; border-color:#e5e7eb; color:#6b7280; }
.td-keterangan { font-size:.78rem; color:var(--text-muted); max-width:180px; }
/* action buttons */
.actions { display:flex; align-items:center; justify-content:center; gap:.4rem; }
.btn-edit, .btn-del {
display:inline-flex; align-items:center; gap:.3rem;
padding:.38rem .85rem; border-radius:8px;
font-size:.75rem; font-weight:700;
text-decoration:none; border:1.5px solid transparent;
cursor:pointer; transition:all .15s; white-space:nowrap;
font-family:'Plus Jakarta Sans',sans-serif;
}
.btn-edit { background:#f0fdf4; color:#2d7a4f; border-color:#bbf7d0; }
.btn-edit:hover { background:#dcfce7; border-color:#2d7a4f; }
.btn-del { background:#fff5f5; color:#dc2626; border-color:#fecaca; }
.btn-del:hover { background:#fee2e2; border-color:#dc2626; }
/* empty state */
.empty-state { padding:4rem 2rem; text-align:center; }
.empty-icon { width:48px; height:48px; margin:0 auto 1rem; opacity:.22; }
.empty-state h4 { font-size:.95rem; font-weight:700; color:var(--text-primary); margin-bottom:.35rem; }
.empty-state p { font-size:.82rem; color:var(--text-muted); }
/* pagination */
.pagination-wrap {
padding:1.1rem 1.75rem;
border-top:1.5px solid rgba(29,77,46,.07);
}
</style>
@endpush
@section('content')
<div class="index-root">
{{-- Alert --}}
@if(session('success'))
<div class="alert-success anim a1">
<svg width="16" height="16" fill="none" stroke="#16a34a" viewBox="0 0 24 24" stroke-width="2.2">
<path stroke-linecap="round" stroke-linejoin="round" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
{{ session('success') }}
</div>
@endif
<div class="index-card anim a2">
{{-- Toolbar --}}
<div class="toolbar">
<div class="toolbar-title">
<span class="toolbar-title-bar"></span>
Daftar Rule Basis
</div>
<a href="{{ route('admin.rule-basis.create') }}" class="btn-add">
<svg width="13" height="13" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2.5">
<path stroke-linecap="round" stroke-linejoin="round" d="M12 4v16m8-8H4"/>
</svg>
Tambah Rule
</a>
</div>
{{-- Filter --}}
<div class="filter-bar anim a2">
<form method="GET" action="{{ route('admin.rule-basis.index') }}" style="display:contents;">
<div class="filter-group">
<label class="filter-label">Filter Penyakit</label>
<select name="penyakit" class="filter-select">
<option value="">Semua Penyakit</option>
@foreach($penyakits as $penyakit)
<option value="{{ $penyakit->id_penyakit }}" {{ request('penyakit') == $penyakit->id_penyakit ? 'selected' : '' }}>
{{ $penyakit->id_penyakit }} {{ $penyakit->nama_penyakit }}
</option>
@endforeach
</select>
</div>
<div class="filter-group">
<label class="filter-label">Filter Gejala</label>
<select name="gejala" class="filter-select">
<option value="">Semua Gejala</option>
@foreach($gejalas as $gejala)
<option value="{{ $gejala->id_gejala }}" {{ request('gejala') == $gejala->id_gejala ? 'selected' : '' }}>
{{ $gejala->id_gejala }} {{ $gejala->nama_gejala }}
</option>
@endforeach
</select>
</div>
<div class="filter-actions">
<button type="submit" class="btn-filter">
<svg width="12" height="12" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2.2">
<path stroke-linecap="round" stroke-linejoin="round" d="M3 4a1 1 0 011-1h16a1 1 0 011 1v2a1 1 0 01-.293.707L13 13.414V19a1 1 0 01-.553.894l-4 2A1 1 0 017 21v-7.586L3.293 6.707A1 1 0 013 6V4z"/>
</svg>
Filter
</button>
@if(request('penyakit') || request('gejala'))
<a href="{{ route('admin.rule-basis.index') }}" class="btn-reset">
<svg width="12" height="12" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2.2">
<path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12"/>
</svg>
Reset
</a>
@endif
</div>
</form>
</div>
{{-- Table --}}
<div class="table-wrap anim a3">
<table>
<thead>
<tr>
<th>No</th>
<th>Penyakit</th>
<th>Gejala</th>
<th class="center">MB</th>
<th class="center">MD</th>
<th class="center">CF</th>
<th>Keterangan</th>
<th class="center">Aksi</th>
</tr>
</thead>
<tbody>
@forelse($rules as $index => $rule)
<tr>
<td class="td-no">{{ $rules->firstItem() + $index }}</td>
<td>
<div class="id-pill">{{ $rule->id_penyakit }}</div>
<div class="cell-name">{{ $rule->penyakit->nama_penyakit ?? '-' }}</div>
</td>
<td>
<div class="id-pill">{{ $rule->id_gejala }}</div>
<div class="cell-sub">{{ $rule->gejala->nama_gejala ?? '-' }}</div>
</td>
<td style="text-align:center;">
<span class="num-badge">{{ number_format($rule->mb, 2) }}</span>
</td>
<td style="text-align:center;">
<span class="num-badge">{{ number_format($rule->md, 2) }}</span>
</td>
<td style="text-align:center;">
@php $cf = $rule->cf_pakar; @endphp
<span class="num-badge {{ $cf >= 0.7 ? 'cf-high' : ($cf >= 0.4 ? 'cf-mid' : 'cf-low') }}">
{{ number_format($cf, 2) }}
</span>
</td>
<td class="td-keterangan">{{ Str::limit($rule->keterangan ?? '-', 50) }}</td>
<td>
<div class="actions">
<a href="{{ route('admin.rule-basis.edit', $rule->id_rule) }}" class="btn-edit">
<svg width="12" height="12" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2.2">
<path stroke-linecap="round" stroke-linejoin="round" d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z"/>
</svg>
Edit
</a>
<form action="{{ route('admin.rule-basis.destroy', $rule->id_rule) }}" method="POST"
onsubmit="return confirm('Yakin ingin menghapus rule ini?')" style="display:contents;">
@csrf
@method('DELETE')
<button type="submit" class="btn-del">
<svg width="12" height="12" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2.2">
<path stroke-linecap="round" stroke-linejoin="round" d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16"/>
</svg>
Hapus
</button>
</form>
</div>
</td>
</tr>
@empty
<tr>
<td colspan="8">
<div class="empty-state">
<svg class="empty-icon" fill="none" stroke="#2d7a4f" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="1.5" 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"/>
</svg>
<h4>Belum ada rule basis</h4>
<p>Klik <strong>Tambah Rule</strong> untuk menambahkan aturan diagnosa.</p>
</div>
</td>
</tr>
@endforelse
</tbody>
</table>
</div>
{{-- Pagination --}}
@if($rules->hasPages())
<div class="pagination-wrap">
{{ $rules->links() }}
</div>
@endif
</div>
</div>
@endsection