HH/templates/analytics/kpi_report_list.html
ismail c5f76b3855
Some checks are pending
Build and Push Docker Image / build (push) Waiting to run
updates
2026-05-11 14:45:30 +03:00

427 lines
20 KiB
HTML

{% extends 'layouts/base.html' %}
{% load i18n %}
{% load analytics_extras %}
{% block title %}{% trans "KPI Reports" %} - PX360{% endblock %}
{% block extra_css %}
<style>
.page-header-gradient {
background: linear-gradient(135deg, #005696 0%, #0069a8 50%, #007bbd 100%);
color: white;
padding: 1.5rem 2rem;
border-radius: 1rem;
margin-bottom: 1.5rem;
box-shadow: 0 10px 15px -3px rgba(0, 86, 150, 0.2);
}
.year-card {
background: white;
border-radius: 1rem;
border: 2px solid #e2e8f0;
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1);
overflow: hidden;
transition: all 0.3s ease;
margin-bottom: 1.5rem;
}
.year-card:hover {
border-color: #005696;
box-shadow: 0 10px 25px -5px rgba(0, 86, 150, 0.15);
}
.year-header {
padding: 1rem 1.5rem;
border-bottom: 2px solid #e2e8f0;
background: linear-gradient(to right, #f8fafc, #f1f5f9);
display: flex;
align-items: center;
gap: 0.75rem;
}
.quarter-section {
padding: 1rem 1.5rem;
border-bottom: 1px solid #f1f5f9;
}
.quarter-section:last-child {
border-bottom: none;
}
.quarter-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.75rem;
}
.quarter-badge {
font-size: 0.65rem;
font-weight: 800;
text-transform: uppercase;
letter-spacing: 0.05em;
padding: 0.2rem 0.5rem;
border-radius: 0.375rem;
background: #e0f2fe;
color: #075985;
}
.quarter-months {
font-size: 0.7rem;
color: #94a3b8;
font-weight: 600;
}
.status-completed { background-color: #dcfce7; color: #166534; }
.status-pending { background-color: #fef9c3; color: #854d0e; }
.status-generating { background-color: #e0f2fe; color: #075985; }
.status-failed { background-color: #fee2e2; color: #991b1b; }
.filter-btn.active { background-color: #005696; color: white; }
.filter-btn:not(.active) { background-color: transparent; border: 1px solid #e2e8f0; color: #64748b; }
.filter-btn:not(.active):hover { background-color: #f8fafc; }
/* Compact report card styles */
.report-card {
background: white;
border: 1px solid #e2e8f0;
border-radius: 0.75rem;
padding: 0.875rem;
transition: all 0.2s ease;
cursor: pointer;
}
.report-card:hover {
border-color: #005696;
box-shadow: 0 4px 12px -2px rgba(0, 86, 150, 0.12);
transform: translateY(-2px);
}
.report-card .actions {
opacity: 0;
transition: opacity 0.2s ease;
}
.report-card:hover .actions {
opacity: 1;
}
</style>
{% endblock %}
{% block content %}
<!-- Page Header -->
<div class="page-header-gradient">
<div class="flex justify-between items-start">
<div>
<h1 class="text-2xl font-bold flex items-center gap-3">
<i data-lucide="bar-chart-3" class="w-7 h-7"></i>
{% trans "KPI Reports" %}
</h1>
<p class="text-sm mt-1 opacity-75">{% trans "Monthly automated reports for MOH and internal KPIs" %}</p>
</div>
<div class="flex items-center gap-3">
<div class="relative group">
<i data-lucide="search" class="w-4 h-4 absolute left-3 top-1/2 -translate-y-1/2 text-white/70 group-focus-within:text-white"></i>
<input type="text" id="searchInput" placeholder="{% trans 'Search KPI ID or indicator...' %}"
class="pl-10 pr-4 py-2.5 bg-white/20 border-transparent border focus:border-white/50 focus:bg-white/30 rounded-xl text-sm outline-none w-64 transition-all text-white placeholder-white/70">
</div>
<a href="{% url 'analytics:kpi_report_generate' %}"
class="bg-white text-[#005696] px-5 py-2.5 rounded-xl text-sm font-bold shadow-lg hover:bg-gray-100 flex items-center gap-2 transition">
<i data-lucide="plus" class="w-4 h-4"></i> {% trans "Generate Report" %}
</a>
</div>
</div>
</div>
<!-- Statistics Cards -->
<div class="grid grid-cols-4 gap-6 mb-6">
<div class="bg-white p-6 rounded-2xl border border-slate-200 shadow-sm flex items-center gap-4 hover:border-blue/30 hover:shadow-md transition-all">
<div class="p-3 bg-blue/10 rounded-xl">
<i data-lucide="bar-chart-3" class="text-blue w-5 h-5"></i>
</div>
<div>
<p class="text-[10px] font-bold text-slate uppercase tracking-wider">{% trans "Total Reports" %}</p>
<p class="text-xl font-black text-navy leading-tight">{{ stats.total }}</p>
</div>
</div>
<div class="bg-white p-6 rounded-2xl border border-slate-200 shadow-sm flex items-center gap-4 hover:border-green-400/50 hover:shadow-md transition-all">
<div class="p-3 bg-green-50 rounded-xl">
<i data-lucide="check-circle" class="text-green-600 w-5 h-5"></i>
</div>
<div>
<p class="text-[10px] font-bold text-slate uppercase tracking-wider">{% trans "Completed" %}</p>
<p class="text-xl font-black text-navy leading-tight">{{ stats.completed }}</p>
</div>
</div>
<div class="bg-white p-6 rounded-2xl border border-slate-200 shadow-sm flex items-center gap-4 hover:border-yellow-400/50 hover:shadow-md transition-all">
<div class="p-3 bg-yellow-50 rounded-xl">
<i data-lucide="clock" class="text-yellow-600 w-5 h-5"></i>
</div>
<div>
<p class="text-[10px] font-bold text-slate uppercase tracking-wider">{% trans "Pending" %}</p>
<p class="text-xl font-black text-navy leading-tight">{{ stats.pending }}</p>
</div>
</div>
<div class="bg-white p-6 rounded-2xl border border-slate-200 shadow-sm flex items-center gap-4 hover:border-red-400/50 hover:shadow-md transition-all">
<div class="p-3 bg-red-50 rounded-xl">
<i data-lucide="alert-triangle" class="text-red-500 w-5 h-5"></i>
</div>
<div>
<p class="text-[10px] font-bold text-slate uppercase tracking-wider">{% trans "Failed" %}</p>
<p class="text-xl font-black text-navy leading-tight">{{ stats.failed }}</p>
</div>
</div>
</div>
<!-- Filter Section -->
<div class="section-card mb-6" style="background: white; border-radius: 1rem; border: 2px solid #e2e8f0; box-shadow: 0 4px 6px -1px rgba(0,0,0,0.1); overflow: hidden;">
<div class="year-header">
<div style="width: 40px; height: 40px; border-radius: 0.75rem; display: flex; align-items: center; justify-content: center; background: linear-gradient(135deg, #005696, #007bbd);">
<i data-lucide="filter" style="width: 20px; height: 20px; color: white;"></i>
</div>
<h5 class="mb-0 fw-bold">{% trans "Filters" %}</h5>
<button onclick="toggleFilters()" class="ms-auto text-slate hover:text-navy transition p-2 rounded-lg hover:bg-white">
<i data-lucide="chevron-up" id="filterToggleIcon" class="w-5 h-5"></i>
</button>
</div>
<!-- Filter Tabs -->
<div class="px-6 py-4 border-b flex items-center justify-between">
<div class="flex items-center gap-3">
<a href="?" class="filter-btn px-4 py-1.5 rounded-full text-xs font-semibold transition {% if not filters.status %}active{% endif %}">
{% trans "All Reports" %}
</a>
<a href="?status=completed" class="filter-btn px-4 py-1.5 rounded-full text-xs font-semibold transition {% if filters.status == 'completed' %}active{% endif %}">
{% trans "Completed" %}
</a>
<a href="?status=pending" class="filter-btn px-4 py-1.5 rounded-full text-xs font-semibold transition {% if filters.status == 'pending' %}active{% endif %}">
{% trans "Pending" %}
</a>
<a href="?status=failed" class="filter-btn px-4 py-1.5 rounded-full text-xs font-semibold transition {% if filters.status == 'failed' %}active{% endif %}">
{% trans "Failed" %}
</a>
</div>
<p class="text-[10px] font-bold text-slate uppercase">
{% trans "Total:" %} <span class="text-navy">{{ stats.total }}</span>
</p>
</div>
<!-- Advanced Filters (Hidden by default) -->
<div id="advancedFilters" class="hidden px-6 py-4 bg-slate-50 border-b">
<form method="get" class="flex flex-wrap gap-4">
{% if filters.status %}
<input type="hidden" name="status" value="{{ filters.status }}">
{% endif %}
<div class="flex items-center gap-2">
<label class="text-xs font-bold text-slate uppercase">{% trans "Report Type" %}</label>
<select name="report_type" class="px-3 py-1.5 bg-white border rounded-lg text-xs">
<option value="">{% trans "All Types" %}</option>
{% for type_value, type_label in report_types %}
<option value="{{ type_value }}" {% if filters.report_type == type_value %}selected{% endif %}>
{{ type_label }}
</option>
{% endfor %}
</select>
</div>
<div class="flex items-center gap-2">
<label class="text-xs font-bold text-slate uppercase">{% trans "Year" %}</label>
<select name="year" class="px-3 py-1.5 bg-white border rounded-lg text-xs">
<option value="">{% trans "All Years" %}</option>
{% for y in years %}
<option value="{{ y }}" {% if filters.year == y|stringformat:'s' %}selected{% endif %}>
{{ y }}
</option>
{% endfor %}
</select>
</div>
<div class="flex items-center gap-2">
<label class="text-xs font-bold text-slate uppercase">{% trans "Month" %}</label>
<select name="month" class="px-3 py-1.5 bg-white border rounded-lg text-xs">
<option value="">{% trans "All Months" %}</option>
{% for m, m_label in months %}
<option value="{{ m }}" {% if filters.month == m|stringformat:'s' %}selected{% endif %}>
{{ m_label }}
</option>
{% endfor %}
</select>
</div>
<button type="submit" class="px-4 py-1.5 bg-navy text-white rounded-lg text-xs font-bold">{% trans "Apply" %}</button>
<a href="?" class="px-4 py-1.5 border rounded-lg text-xs font-semibold text-slate hover:bg-white">{% trans "Clear" %}</a>
</form>
</div>
</div>
<!-- Reports Grid - Grouped by Year and Quarter -->
{% if reports_by_year %}
{% for year, quarters in reports_by_year.items %}
<div class="year-card">
<!-- Year Header -->
<div class="year-header cursor-pointer" onclick="toggleYear('year-{{ year }}', this)">
<h2 class="text-xl font-black text-navy">{{ year }}</h2>
<span class="px-2.5 py-1 text-[10px] font-bold bg-navy/10 text-navy rounded-full">
{% with q1_count=quarters|get_item:"1"|length q2_count=quarters|get_item:"2"|length q3_count=quarters|get_item:"3"|length q4_count=quarters|get_item:"4"|length %}
{% with year_total=q1_count|add:q2_count|add:q3_count|add:q4_count %}
{{ year_total }} {% trans "reports" %}
{% endwith %}
{% endwith %}
</span>
<div class="flex-1 h-px bg-slate-200"></div>
<i data-lucide="chevron-right" class="w-5 h-5 text-slate transition-transform duration-200"></i>
</div>
<!-- Quarters: Q4, Q3, Q2, Q1 -->
<div id="year-{{ year }}" class="year-content hidden">
{% for quarter_num in "4321" %}
{% with quarter_reports=quarters|get_item:quarter_num %}
{% if quarter_reports %}
<div class="quarter-section">
<div class="quarter-header cursor-pointer" onclick="toggleQuarter('quarter-{{ year }}-{{ quarter_num }}', this)">
<span class="quarter-badge">Q{{ quarter_num }}</span>
<span class="quarter-months">
{% if quarter_num == "1" %}{% trans "Jan - Mar" %}
{% elif quarter_num == "2" %}{% trans "Apr - Jun" %}
{% elif quarter_num == "3" %}{% trans "Jul - Sep" %}
{% else %}{% trans "Oct - Dec" %}
{% endif %}
</span>
<span class="text-[10px] text-slate font-semibold ml-auto">{{ quarter_reports|length }} {% trans "reports" %}</span>
<i data-lucide="chevron-right" class="w-4 h-4 text-slate transition-transform duration-200"></i>
</div>
<!-- Report Cards Grid - Compact 4 columns -->
<div id="quarter-{{ year }}-{{ quarter_num }}" class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-3 hidden">
{% for report in quarter_reports %}
<div class="report-card group" onclick="window.location.href='{% url 'analytics:kpi_report_detail' report.id %}'">
<!-- Compact Header -->
<div class="flex items-start justify-between mb-2">
<div class="flex items-center gap-1.5 flex-wrap">
<span class="px-1.5 py-0.5 text-[10px] font-bold rounded bg-navy text-white">
{{ report.kpi_id }}
</span>
<span class="text-[10px] text-slate">{{ report.report_period_display }}</span>
</div>
<span class="px-1.5 py-0.5 text-[9px] rounded-full font-bold uppercase
{% if report.status == 'completed' %}status-completed
{% elif report.status == 'failed' %}status-failed
{% elif report.status == 'generating' %}status-generating
{% else %}status-pending{% endif %}">
{{ report.get_status_display }}
</span>
</div>
<!-- Title -->
<h3 class="text-xs font-semibold text-navy mb-1.5 line-clamp-2 group-hover:text-blue transition-colors leading-tight">
{{ report.indicator_title }}
</h3>
<!-- Hospital -->
<p class="text-[10px] text-slate mb-2 flex items-center gap-1">
<i data-lucide="building-2" class="w-2.5 h-2.5 inline"></i>
{{ report.hospital.name }}
</p>
<!-- Compact Metrics Row -->
<div class="flex gap-2 mb-2">
<div class="flex-1 bg-slate-50 rounded-md px-2 py-1.5 text-center">
<div class="text-[9px] font-bold text-slate uppercase">{% trans "Target" %}</div>
<div class="text-sm font-black text-navy">{{ report.target_percentage }}%</div>
</div>
<div class="flex-1 bg-slate-50 rounded-md px-2 py-1.5 text-center">
<div class="text-[9px] font-bold text-slate uppercase">{% trans "Result" %}</div>
<div class="text-sm font-black {% if report.overall_result >= report.target_percentage %}text-green-600{% else %}text-red-600{% endif %}">
{{ report.overall_result }}%
</div>
</div>
<div class="flex-1 bg-slate-50 rounded-md px-2 py-1.5 text-center">
<div class="text-[9px] font-bold text-slate uppercase">{% trans "Cases" %}</div>
<div class="text-sm font-black text-navy">{{ report.total_denominator }}</div>
</div>
</div>
<!-- Actions -->
<div class="actions flex gap-1.5 pt-2 border-t border-slate-100">
<button onclick="event.stopPropagation(); window.location.href='{% url 'analytics:kpi_report_detail' report.id %}'"
class="flex-1 bg-navy text-white text-center text-[10px] font-semibold py-1.5 rounded-md flex items-center justify-center gap-1 hover:bg-navy/90 transition">
<i data-lucide="eye" class="w-3 h-3"></i> {% trans "View" %}
</button>
<button onclick="event.stopPropagation(); window.location.href='{% url 'analytics:kpi_report_pdf' report.id %}'"
class="px-2 py-1.5 border border-slate-200 rounded-md hover:bg-slate-50 transition" title="{% trans 'Download PDF' %}">
<i data-lucide="file-down" class="w-3 h-3 text-slate"></i>
</button>
{% if report.status == 'failed' or report.status == 'pending' %}
<form method="post" action="{% url 'analytics:kpi_report_regenerate' report.id %}"
class="inline" onclick="event.stopPropagation(); return confirm('{% trans "Regenerate this report?" %}')" data-loading data-loading-text="{% trans '...' %}">
{% csrf_token %}
<button type="submit" class="px-2 py-1.5 border border-slate-200 rounded-md hover:bg-slate-50 transition" title="{% trans 'Regenerate' %}">
<i data-lucide="refresh-cw" class="w-3 h-3 text-slate"></i>
</button>
</form>
{% endif %}
</div>
</div>
{% endfor %}
</div>
</div>
{% endif %}
{% endwith %}
{% endfor %}
</div>
</div>
{% endfor %}
{% else %}
<div class="section-card text-center py-16" style="background: white; border-radius: 1rem; border: 2px solid #e2e8f0;">
<i data-lucide="bar-chart-3" class="w-20 h-20 mx-auto text-slate-300 mb-4"></i>
<h3 class="text-lg font-bold text-navy mb-2">{% trans "No KPI Reports Found" %}</h3>
<p class="text-slate mb-6">{% trans "Generate your first KPI report to get started." %}</p>
<a href="{% url 'analytics:kpi_report_generate' %}" class="hh-btn hh-btn-primary inline-flex items-center gap-2">
<i data-lucide="plus" class="w-4 h-4"></i>
{% trans "Generate Report" %}
</a>
</div>
{% endif %}
{% endblock %}
{% block extra_js %}
<script>
function toggleFilters() {
const filters = document.getElementById('advancedFilters');
const icon = document.getElementById('filterToggleIcon');
filters.classList.toggle('hidden');
if (filters.classList.contains('hidden')) {
icon.setAttribute('data-lucide', 'chevron-down');
} else {
icon.setAttribute('data-lucide', 'chevron-up');
}
lucide.createIcons();
}
function toggleYear(yearId, headerEl) {
const content = document.getElementById(yearId);
const icon = headerEl.querySelector('[data-lucide="chevron-right"], [data-lucide="chevron-down"]');
if (content.classList.contains('hidden')) {
content.classList.remove('hidden');
icon.setAttribute('data-lucide', 'chevron-down');
icon.style.transform = 'rotate(0deg)';
} else {
content.classList.add('hidden');
icon.setAttribute('data-lucide', 'chevron-right');
icon.style.transform = 'rotate(0deg)';
}
lucide.createIcons();
}
function toggleQuarter(quarterId, headerEl) {
const content = document.getElementById(quarterId);
const icon = headerEl.querySelector('[data-lucide="chevron-right"], [data-lucide="chevron-down"]');
if (content.classList.contains('hidden')) {
content.classList.remove('hidden');
icon.setAttribute('data-lucide', 'chevron-down');
icon.style.transform = 'rotate(0deg)';
} else {
content.classList.add('hidden');
icon.setAttribute('data-lucide', 'chevron-right');
icon.style.transform = 'rotate(0deg)';
}
lucide.createIcons();
}
// Search functionality
document.getElementById('searchInput')?.addEventListener('keypress', function(e) {
if (e.key === 'Enter') {
const value = this.value;
if (value) {
window.location.href = '?search=' + encodeURIComponent(value);
}
}
});
</script>
{% endblock %}