HH/templates/analytics/kpi_report_list.html
2026-02-22 08:35:53 +03:00

341 lines
18 KiB
HTML

{% extends 'layouts/base.html' %}
{% load i18n %}
{% block title %}{% trans "KPI Reports" %} - PX360{% endblock %}
{% block extra_css %}
<style>
.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; }
</style>
{% endblock %}
{% block content %}
<!-- Header -->
<header class="mb-6">
<div class="flex justify-between items-start">
<div>
<h1 class="text-2xl font-bold text-navy 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 text-slate mt-1">{% 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-slate group-focus-within:text-navy"></i>
<input type="text" id="searchInput" placeholder="{% trans 'Search KPI ID or indicator...' %}"
class="pl-10 pr-4 py-2.5 bg-slate-100 border-transparent border focus:border-navy/30 focus:bg-white rounded-xl text-sm outline-none w-64 transition-all">
</div>
<a href="{% url 'analytics:kpi_report_generate' %}"
class="bg-navy text-white px-5 py-2.5 rounded-xl text-sm font-bold shadow-lg shadow-navy/20 hover:bg-blue flex items-center gap-2 transition">
<i data-lucide="plus" class="w-4 h-4"></i> {% trans "Generate Report" %}
</a>
</div>
</div>
</header>
<!-- Statistics Cards -->
<div class="grid grid-cols-4 gap-6 mb-6">
<div class="bg-white p-4 rounded-2xl border shadow-sm flex items-center gap-4">
<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-4 rounded-2xl border shadow-sm flex items-center gap-4">
<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-4 rounded-2xl border shadow-sm flex items-center gap-4">
<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-4 rounded-2xl border shadow-sm flex items-center gap-4">
<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 Tabs -->
<div class="bg-white px-6 py-4 rounded-t-2xl 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 class="h-4 w-[1px] bg-slate-200 mx-2"></div>
<button onclick="toggleFilters()" class="flex items-center gap-2 text-xs font-bold text-blue uppercase tracking-tight hover:underline">
<i data-lucide="filter" class="w-3 h-3"></i> {% trans "Advanced Filters" %}
</button>
</div>
<p class="text-[10px] font-bold text-slate uppercase">
{% trans "Showing:" %} <span class="text-navy">{{ page_obj.start_index|default:0 }}-{{ page_obj.end_index|default:0 }} {% trans "of" %} {{ page_obj.paginator.count|default:0 }}</span>
</p>
</div>
<!-- Advanced Filters (Hidden by default) -->
<div id="advancedFilters" class="hidden bg-slate-50 px-6 py-4 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>
{% if request.user.is_px_admin %}
<div class="flex items-center gap-2">
<label class="text-xs font-bold text-slate uppercase">{% trans "Hospital" %}</label>
<select name="hospital" class="px-3 py-1.5 bg-white border rounded-lg text-xs">
<option value="">{% trans "All Hospitals" %}</option>
{% for hospital in hospitals %}
<option value="{{ hospital.id }}" {% if filters.hospital == hospital.id|stringformat:'s' %}selected{% endif %}>
{{ hospital.name }}
</option>
{% endfor %}
</select>
</div>
{% endif %}
<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>
<!-- Reports Grid -->
{% if reports %}
<div class="bg-white rounded-b-2xl shadow-sm border p-6">
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
{% for report in reports %}
<div class="card hover:shadow-lg hover:-translate-y-1 transition-all duration-200 cursor-pointer group"
onclick="window.location.href='{% url 'analytics:kpi_report_detail' report.id %}'">
<!-- Header -->
<div class="flex items-start justify-between mb-3">
<div class="flex items-center gap-2">
<span class="px-2 py-1 text-xs font-bold rounded bg-navy text-white">
{{ report.kpi_id }}
</span>
<span class="text-xs text-slate">{{ report.report_period_display }}</span>
</div>
<span class="px-2 py-0.5 text-xs rounded-full font-semibold 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="font-semibold text-navy mb-2 line-clamp-2 group-hover:text-blue transition-colors">{{ report.indicator_title }}</h3>
<!-- Hospital -->
<p class="text-sm text-slate mb-4 flex items-center gap-1">
<i data-lucide="building-2" class="w-3 h-3 inline"></i>
{{ report.hospital.name }}
</p>
<!-- Results -->
<div class="grid grid-cols-3 gap-2 mb-4">
<div class="bg-light rounded-lg p-3 text-center">
<div class="text-[10px] font-bold text-slate uppercase mb-1">{% trans "Target" %}</div>
<div class="text-lg font-black text-navy">{{ report.target_percentage }}%</div>
</div>
<div class="bg-light rounded-lg p-3 text-center">
<div class="text-[10px] font-bold text-slate uppercase mb-1">{% trans "Result" %}</div>
<div class="text-lg font-black {% if report.overall_result >= report.target_percentage %}text-green-600{% else %}text-red-600{% endif %}">
{{ report.overall_result }}%
</div>
</div>
<div class="bg-light rounded-lg p-3 text-center">
<div class="text-[10px] font-bold text-slate uppercase mb-1">{% trans "Cases" %}</div>
<div class="text-lg font-black text-navy">{{ report.total_denominator }}</div>
</div>
</div>
<!-- Actions -->
<div class="flex gap-2 pt-3 border-t opacity-0 group-hover:opacity-100 transition-opacity">
<button onclick="event.stopPropagation(); window.location.href='{% url 'analytics:kpi_report_detail' report.id %}'"
class="flex-1 btn-primary text-center text-sm flex items-center justify-center gap-2">
<i data-lucide="eye" class="w-4 h-4"></i> {% trans "View" %}
</button>
<button onclick="event.stopPropagation(); window.location.href='{% url 'analytics:kpi_report_pdf' report.id %}'"
class="btn-secondary px-3" title="{% trans 'Download PDF' %}">
<i data-lucide="file-down" class="w-4 h-4"></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?" %}')">
{% csrf_token %}
<button type="submit" class="btn-secondary px-3" title="{% trans 'Regenerate' %}">
<i data-lucide="refresh-cw" class="w-4 h-4"></i>
</button>
</form>
{% endif %}
</div>
</div>
{% endfor %}
</div>
<!-- Pagination -->
{% if page_obj.has_other_pages %}
<div class="bg-slate-50 px-8 py-4 flex items-center justify-between border-t mt-6 rounded-lg">
<div class="flex items-center gap-4">
<span class="text-xs text-slate font-medium">
{% trans "Showing" %} <span class="font-bold text-navy">{{ page_obj.start_index }}-{{ page_obj.end_index }}</span> {% trans "of" %} <span class="font-bold text-navy">{{ page_obj.paginator.count }}</span> {% trans "entries" %}
</span>
<!-- Page Size Selector -->
<form method="get" class="flex items-center gap-2" id="pageSizeForm">
{% for key, value in request.GET.items %}
{% if key != 'page_size' and key != 'page' %}
<input type="hidden" name="{{ key }}" value="{{ value }}">
{% endif %}
{% endfor %}
<label class="text-xs text-slate">{% trans "Show" %}</label>
<select name="page_size" onchange="document.getElementById('pageSizeForm').submit()" class="px-2 py-1 bg-white border border-slate-200 rounded-lg text-xs focus:outline-none focus:ring-2 focus:ring-navy">
<option value="6" {% if page_obj.paginator.per_page == 6 %}selected{% endif %}>6</option>
<option value="12" {% if page_obj.paginator.per_page == 12 %}selected{% endif %}>12</option>
<option value="24" {% if page_obj.paginator.per_page == 24 %}selected{% endif %}>24</option>
<option value="48" {% if page_obj.paginator.per_page == 48 %}selected{% endif %}>48</option>
</select>
</form>
</div>
<div class="flex gap-2">
{% if page_obj.has_previous %}
<a href="?page={{ page_obj.previous_page_number }}{% for key, value in request.GET.items %}{% if key != 'page' %}&{{ key }}={{ value }}{% endif %}{% endfor %}"
class="w-8 h-8 flex items-center justify-center rounded-lg border bg-white hover:bg-slate-50 transition">
<i data-lucide="chevron-left" class="w-4 h-4 text-slate"></i>
</a>
{% else %}
<span class="w-8 h-8 flex items-center justify-center rounded-lg border bg-slate-100 text-slate-300 cursor-not-allowed">
<i data-lucide="chevron-left" class="w-4 h-4"></i>
</span>
{% endif %}
{% for num in page_obj.paginator.page_range %}
{% if num == page_obj.number %}
<span class="w-8 h-8 flex items-center justify-center rounded-lg bg-navy text-white text-xs font-bold">{{ num }}</span>
{% elif num > page_obj.number|add:'-3' and num < page_obj.number|add:'3' %}
<a href="?page={{ num }}{% for key, value in request.GET.items %}{% if key != 'page' %}&{{ key }}={{ value }}{% endif %}{% endfor %}"
class="w-8 h-8 flex items-center justify-center rounded-lg border bg-white hover:bg-slate-50 text-xs font-bold text-slate transition">
{{ num }}
</a>
{% elif num == 1 or num == page_obj.paginator.num_pages %}
<a href="?page={{ num }}{% for key, value in request.GET.items %}{% if key != 'page' %}&{{ key }}={{ value }}{% endif %}{% endfor %}"
class="w-8 h-8 flex items-center justify-center rounded-lg border bg-white hover:bg-slate-50 text-xs font-bold text-slate transition">
{{ num }}
</a>
{% elif num == page_obj.number|add:'-3' or num == page_obj.number|add:'3' %}
<span class="w-8 h-8 flex items-center justify-center text-xs text-slate">...</span>
{% endif %}
{% endfor %}
{% if page_obj.has_next %}
<a href="?page={{ page_obj.next_page_number }}{% for key, value in request.GET.items %}{% if key != 'page' %}&{{ key }}={{ value }}{% endif %}{% endfor %}"
class="w-8 h-8 flex items-center justify-center rounded-lg border bg-white hover:bg-slate-50 transition">
<i data-lucide="chevron-right" class="w-4 h-4 text-slate"></i>
</a>
{% else %}
<span class="w-8 h-8 flex items-center justify-center rounded-lg border bg-slate-100 text-slate-300 cursor-not-allowed">
<i data-lucide="chevron-right" class="w-4 h-4"></i>
</span>
{% endif %}
</div>
</div>
{% endif %}
</div>
{% else %}
<div class="card text-center py-16">
<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="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');
filters.classList.toggle('hidden');
}
// 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 %}