824 lines
47 KiB
HTML
824 lines
47 KiB
HTML
{% extends "layouts/base.html" %}
|
|
{% load i18n %}
|
|
{% load static %}
|
|
|
|
{% block title %}{{ observation.tracking_code }} - {% trans "Observation Detail" %} - PX360{% endblock %}
|
|
|
|
{% block extra_css %}
|
|
<style>
|
|
.timeline {
|
|
position: relative;
|
|
padding-left: 30px;
|
|
}
|
|
.timeline::before {
|
|
content: '';
|
|
position: absolute;
|
|
left: 8px;
|
|
top: 0;
|
|
bottom: 0;
|
|
width: 2px;
|
|
background: #e2e8f0;
|
|
}
|
|
.timeline-item {
|
|
position: relative;
|
|
padding-bottom: 24px;
|
|
}
|
|
.timeline-item::before {
|
|
content: '';
|
|
position: absolute;
|
|
left: -26px;
|
|
top: 5px;
|
|
width: 14px;
|
|
height: 14px;
|
|
border-radius: 50%;
|
|
background: #fff;
|
|
border: 3px solid #005696;
|
|
z-index: 1;
|
|
}
|
|
.timeline-item.status_change::before { border-color: #f97316; }
|
|
.timeline-item.note::before { border-color: #22c55e; }
|
|
</style>
|
|
{% endblock %}
|
|
|
|
{% block content %}
|
|
<!-- Back Link -->
|
|
<div class="mb-6">
|
|
<a href="{% url 'observations:observation_list' %}" class="inline-flex items-center gap-2 text-slate hover:text-navy transition font-medium">
|
|
<i data-lucide="arrow-left" class="w-4 h-4"></i> {% trans "Observations" %}
|
|
</a>
|
|
</div>
|
|
|
|
<!-- Observation Header -->
|
|
<div class="bg-white rounded-2xl shadow-sm border border-slate-100 p-6 mb-6">
|
|
<div class="flex flex-col lg:flex-row lg:items-start lg:justify-between gap-4">
|
|
<div class="flex-1">
|
|
<div class="flex flex-wrap items-center gap-3 mb-3">
|
|
<span class="font-mono text-lg font-bold text-blue tracking-wide">{{ observation.tracking_code }}</span>
|
|
<span class="px-2.5 py-1 rounded-full text-[10px] font-bold uppercase
|
|
{% if observation.status == 'new' %}bg-yellow-100 text-yellow-700
|
|
{% elif observation.status == 'triaged' %}bg-sky-100 text-sky-700
|
|
{% elif observation.status == 'assigned' %}bg-indigo-100 text-indigo-700
|
|
{% elif observation.status == 'in_progress' %}bg-blue-100 text-blue-700
|
|
{% elif observation.status == 'resolved' %}bg-green-100 text-green-700
|
|
{% elif observation.status == 'closed' %}bg-slate-100 text-slate-600
|
|
{% elif observation.status == 'rejected' %}bg-red-100 text-red-700
|
|
{% elif observation.status == 'duplicate' %}bg-slate-100 text-slate-600
|
|
{% elif observation.status == 'contacted' %}bg-purple-100 text-purple-700
|
|
{% elif observation.status == 'contacted_no_response' %}bg-slate-100 text-slate-600
|
|
{% else %}bg-slate-100 text-slate-600{% endif %}">
|
|
{{ observation.get_status_display }}
|
|
</span>
|
|
<span class="px-2.5 py-1 rounded-full text-[10px] font-bold uppercase
|
|
{% if observation.severity == 'low' %}bg-green-100 text-green-700
|
|
{% elif observation.severity == 'medium' %}bg-yellow-100 text-yellow-700
|
|
{% elif observation.severity == 'high' %}bg-orange-100 text-orange-700
|
|
{% elif observation.severity == 'critical' %}bg-red-500 text-white
|
|
{% else %}bg-slate-100 text-slate-600{% endif %}">
|
|
{{ observation.get_severity_display }}
|
|
</span>
|
|
{% if observation.is_anonymous %}
|
|
<span class="px-2.5 py-1 rounded-full text-xs font-medium bg-gray-100 text-gray-500 italic">
|
|
<i data-lucide="eye-off" class="w-3 h-3 inline me-1"></i>{% trans "Anonymous" %}
|
|
</span>
|
|
{% endif %}
|
|
{% if observation.is_overdue %}
|
|
<span class="px-2.5 py-1 rounded-full text-[10px] font-bold uppercase bg-red-500 text-white">
|
|
<i data-lucide="clock" class="w-3 h-3 inline me-1"></i>{% trans "Overdue" %}
|
|
</span>
|
|
{% endif %}
|
|
</div>
|
|
<h1 class="text-2xl font-bold text-navy">
|
|
{% if observation.title %}
|
|
{{ observation.title }}
|
|
{% else %}
|
|
{{ observation.description|truncatewords:10 }}
|
|
{% endif %}
|
|
</h1>
|
|
</div>
|
|
|
|
<div class="flex items-center gap-3">
|
|
{% if can_convert and not observation.action_id %}
|
|
<a href="{% url 'observations:observation_convert_to_action' observation.id %}"
|
|
class="bg-green-500 text-white px-4 py-2.5 rounded-xl text-sm font-semibold hover:bg-green-600 transition inline-flex items-center gap-2">
|
|
<i data-lucide="arrow-right-circle" class="w-4 h-4"></i> {% trans "Convert to Action" %}
|
|
</a>
|
|
{% endif %}
|
|
<a href="{% url 'rca:rca_create' %}?related_model=observation&related_id={{ observation.pk }}"
|
|
class="bg-purple-600 text-white px-4 py-2.5 rounded-xl text-sm font-semibold hover:bg-purple-700 transition inline-flex items-center gap-2">
|
|
<i data-lucide="search" class="w-4 h-4"></i> {% trans "Initiate RCA" %}
|
|
</a>
|
|
<a href="{% url 'projects:project_create' %}?related_model=observation&related_id={{ observation.pk }}"
|
|
class="bg-teal-600 text-white px-4 py-2.5 rounded-xl text-sm font-semibold hover:bg-teal-700 transition inline-flex items-center gap-2">
|
|
<i data-lucide="folder-plus" class="w-4 h-4"></i> {% trans "Create QI Project" %}
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="flex flex-col lg:flex-row gap-6">
|
|
<!-- Main Content -->
|
|
<div class="flex-1 space-y-6">
|
|
<!-- Description Card -->
|
|
<section class="bg-white rounded-2xl p-6 shadow-sm border border-slate-100">
|
|
<h3 class="font-bold text-navy mb-4 flex items-center gap-2">
|
|
<i data-lucide="file-text" class="w-5 h-5 text-blue"></i>
|
|
{% trans "Description" %}
|
|
</h3>
|
|
<div class="bg-light/40 p-4 rounded-xl border-l-4 border-blue">
|
|
<p class="text-sm leading-relaxed text-slate" style="white-space: pre-wrap;">{{ observation.description }}</p>
|
|
</div>
|
|
{% if observation.description_en %}
|
|
<div class="bg-light/40 p-4 rounded-xl border-l-4 border-green-500 mt-4">
|
|
<p class="text-[10px] font-semibold text-slate uppercase tracking-wide mb-1">English</p>
|
|
<p class="text-sm leading-relaxed text-slate" style="white-space: pre-wrap;">{{ observation.description_en }}</p>
|
|
</div>
|
|
{% endif %}
|
|
</section>
|
|
|
|
<!-- AI Analysis Panel -->
|
|
{% include "observations/partials/ai_panel.html" %}
|
|
|
|
<!-- Details Card -->
|
|
<section class="bg-white rounded-2xl p-6 shadow-sm border border-slate-100">
|
|
<h3 class="font-bold text-navy mb-4 flex items-center gap-2">
|
|
<i data-lucide="info" class="w-5 h-5 text-blue"></i>
|
|
{% trans "Details" %}
|
|
</h3>
|
|
<div class="grid grid-cols-2 gap-4">
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Category" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.category.get_localized_name|default:_("Not specified") }}</p>
|
|
</div>
|
|
{% if observation.sub_category %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Sub-Category" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.sub_category.get_localized_name }}</p>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.taxonomy_domain or observation.taxonomy_category or observation.taxonomy_subcategory or observation.taxonomy_classification %}
|
|
<div class="col-span-2">
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "SHCT Taxonomy" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">
|
|
{{ observation.taxonomy_domain.name_en|default:"" }}
|
|
{% if observation.taxonomy_category %} > {{ observation.taxonomy_category.name_en }}{% endif %}
|
|
{% if observation.taxonomy_subcategory %} > {{ observation.taxonomy_subcategory.name_en }}{% endif %}
|
|
{% if observation.taxonomy_classification %} > {{ observation.taxonomy_classification.name_en }}{% endif %}
|
|
</p>
|
|
</div>
|
|
{% endif %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Location" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">
|
|
{% if observation.location %}{{ observation.location.name_en }}{% elif observation.location_text %}{{ observation.location_text }}{% else %}{% trans "Not specified" %}{% endif %}
|
|
</p>
|
|
</div>
|
|
{% if observation.main_section %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Section" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.main_section.name_en }}</p>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.subsection %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Subsection" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.subsection.name_en }}</p>
|
|
</div>
|
|
{% endif %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Incident Date/Time" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.incident_datetime|date:"M d, Y H:i" }}</p>
|
|
</div>
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Submitted" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.created_at|date:"M d, Y H:i" }}</p>
|
|
</div>
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Last Updated" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.updated_at|date:"M d, Y H:i" }}</p>
|
|
</div>
|
|
{% if observation.due_at %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Response Deadline" %}</p>
|
|
<p class="text-sm font-medium {% if observation.is_overdue %}text-red-600 font-bold{% else %}text-gray-800{% endif %}">
|
|
{{ observation.due_at|date:"M d, Y H:i" }}
|
|
{% if observation.is_overdue %}<i data-lucide="alert-triangle" class="w-3 h-3 inline ms-1"></i>{% endif %}
|
|
</p>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.triaged_at %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Triaged" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">
|
|
{{ observation.triaged_at|date:"M d, Y H:i" }}
|
|
{% if observation.triaged_by %}({{ observation.triaged_by.get_full_name }}){% endif %}
|
|
</p>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
</section>
|
|
|
|
<!-- Reporter Information -->
|
|
<section class="bg-white rounded-2xl p-6 shadow-sm border border-slate-100">
|
|
<h3 class="font-bold text-navy mb-4 flex items-center gap-2">
|
|
<i data-lucide="user" class="w-5 h-5 text-blue"></i>
|
|
{% trans "Reporter Information" %}
|
|
</h3>
|
|
{% if observation.is_anonymous %}
|
|
<div class="text-center py-4">
|
|
<div class="w-12 h-12 bg-gray-100 rounded-full flex items-center justify-center mx-auto mb-3">
|
|
<i data-lucide="eye-off" class="w-6 h-6 text-gray-400"></i>
|
|
</div>
|
|
<p class="text-slate text-sm">{% trans "This observation was submitted anonymously" %}</p>
|
|
</div>
|
|
{% else %}
|
|
<div class="grid grid-cols-2 gap-4">
|
|
{% if observation.reporter_staff_id %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Staff ID" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.reporter_staff_id }}</p>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.reporter_name %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Name" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.reporter_name }}</p>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.reporter_phone %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Phone" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.reporter_phone }}</p>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.reporter_email %}
|
|
<div>
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Email" %}</p>
|
|
<p class="text-sm font-medium text-gray-800">{{ observation.reporter_email }}</p>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
</section>
|
|
|
|
<!-- Attachments -->
|
|
{% if attachments %}
|
|
<section class="bg-white rounded-2xl p-6 shadow-sm border border-slate-100">
|
|
<h3 class="font-bold text-navy mb-4 flex items-center gap-2">
|
|
<i data-lucide="paperclip" class="w-5 h-5 text-blue"></i>
|
|
{% trans "Attachments" %} ({{ attachments.count }})
|
|
</h3>
|
|
<div class="space-y-3">
|
|
{% for attachment in attachments %}
|
|
<div class="flex items-center justify-between p-4 bg-slate-50 border border-slate-100 rounded-xl">
|
|
<div class="flex items-center gap-3">
|
|
<i data-lucide="file" class="w-5 h-5 text-blue"></i>
|
|
<div>
|
|
<div class="font-semibold text-gray-800">{{ attachment.filename }}</div>
|
|
<div class="text-xs text-slate">{{ attachment.file_type }} - {{ attachment.file_size|filesizeformat }}</div>
|
|
</div>
|
|
</div>
|
|
<a href="{{ attachment.file.url }}" target="_blank" class="p-2 rounded-lg bg-navy text-white hover:bg-blue transition">
|
|
<i data-lucide="download" class="w-4 h-4"></i>
|
|
</a>
|
|
</div>
|
|
{% endfor %}
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
<!-- Timeline -->
|
|
<section class="bg-white rounded-2xl p-6 shadow-sm border border-slate-100">
|
|
<h3 class="font-bold text-navy mb-4 flex items-center gap-2">
|
|
<i data-lucide="clock" class="w-5 h-5 text-blue"></i>
|
|
{% trans "Timeline" %}
|
|
</h3>
|
|
{% if timeline %}
|
|
<div class="timeline">
|
|
{% for item in timeline %}
|
|
<div class="timeline-item {{ item.type }}">
|
|
<div class="bg-slate-50 rounded-xl p-4 border border-slate-100">
|
|
<div class="flex justify-between items-start mb-2">
|
|
<div class="flex items-center gap-2">
|
|
<span class="px-2.5 py-1 rounded-lg text-xs font-semibold bg-navy text-white">
|
|
{% if item.type == 'status_change' %}{% trans "Status Changed" %}
|
|
{% elif item.type == 'note' %}{% trans "Note" %}{% endif %}
|
|
</span>
|
|
{% if item.item.created_by %}
|
|
<span class="text-xs text-slate">{% trans "by" %} {{ item.item.created_by.get_full_name }}</span>
|
|
{% elif item.item.changed_by %}
|
|
<span class="text-xs text-slate">{% trans "by" %} {{ item.item.changed_by.get_full_name }}</span>
|
|
{% endif %}
|
|
</div>
|
|
<span class="text-xs text-slate">{{ item.created_at|date:"M d, Y H:i" }}</span>
|
|
</div>
|
|
{% if item.type == 'status_change' %}
|
|
<div class="flex items-center gap-2">
|
|
{% if item.item.from_status %}
|
|
<span class="px-2 py-0.5 rounded text-xs font-medium bg-slate-100 text-slate-600">{{ item.item.from_status }}</span>
|
|
<i data-lucide="arrow-right" class="w-3 h-3 text-slate"></i>
|
|
{% endif %}
|
|
<span class="px-2 py-0.5 rounded text-xs font-medium bg-blue-100 text-blue-600">{{ item.item.to_status }}</span>
|
|
</div>
|
|
{% elif item.type == 'note' %}
|
|
<p class="text-gray-700 text-sm mt-1">{{ item.item.note }}</p>
|
|
{% endif %}
|
|
{% if item.item.comment %}
|
|
<p class="text-gray-600 text-sm mt-2">{{ item.item.comment }}</p>
|
|
{% endif %}
|
|
</div>
|
|
</div>
|
|
{% endfor %}
|
|
</div>
|
|
{% else %}
|
|
<div class="text-center py-8">
|
|
<i data-lucide="clock" class="w-12 h-12 mx-auto mb-3 text-gray-300"></i>
|
|
<p class="text-slate text-sm">{% trans "No timeline entries yet" %}</p>
|
|
</div>
|
|
{% endif %}
|
|
</section>
|
|
</div>
|
|
|
|
<!-- Sidebar -->
|
|
<div class="lg:w-80 space-y-4">
|
|
<!-- Status Timeline -->
|
|
{% include "partials/stage_timeline.html" with stage_timeline=stage_timeline %}
|
|
|
|
<!-- Assignment Info -->
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 border-b border-slate-100">
|
|
<h3 class="font-bold text-navy flex items-center gap-2">
|
|
<i data-lucide="users" class="w-4 h-4"></i> {% trans "Assignment" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4 space-y-4">
|
|
<div>
|
|
<div class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Department" %}</div>
|
|
<div class="text-sm font-medium text-gray-800">{{ observation.assigned_department.name|default:_("Not assigned") }}</div>
|
|
</div>
|
|
<div>
|
|
<div class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Assigned To" %}</div>
|
|
<div class="text-sm font-medium text-gray-800">{{ observation.assigned_to.get_full_name|default:_("Not assigned") }}</div>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
|
|
<!-- Communication Info -->
|
|
{% if observation.person_noted or observation.department_noted or observation.communication_method or observation.communication_datetime or observation.patient_file_number %}
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 border-b border-slate-100">
|
|
<h3 class="font-bold text-navy flex items-center gap-2">
|
|
<i data-lucide="phone-call" class="w-4 h-4"></i> {% trans "Communication" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4 space-y-3">
|
|
{% if observation.patient_file_number %}
|
|
<div>
|
|
<div class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "File Number" %}</div>
|
|
<div class="text-sm font-medium text-gray-800">{{ observation.patient_file_number }}</div>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.person_noted %}
|
|
<div>
|
|
<div class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Person Noted" %}</div>
|
|
<div class="text-sm font-medium text-gray-800">{{ observation.person_noted }}</div>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.department_noted %}
|
|
<div>
|
|
<div class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Department Noted" %}</div>
|
|
<div class="text-sm font-medium text-gray-800">{{ observation.department_noted.name }}</div>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.communication_method %}
|
|
<div>
|
|
<div class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Via" %}</div>
|
|
<div class="text-sm font-medium text-gray-800">{{ observation.communication_method }}</div>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.communication_datetime %}
|
|
<div>
|
|
<div class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">{% trans "Contacted At" %}</div>
|
|
<div class="text-sm font-medium text-gray-800">{{ observation.communication_datetime|date:"M d, Y H:i" }}</div>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
<!-- Department Response -->
|
|
{% if observation.assigned_department %}
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 border-b border-slate-100 {% if observation.department_responded_at %}bg-blue-50{% elif observation.dept_response_is_overdue %}bg-red-50{% else %}bg-amber-50{% endif %}">
|
|
<h3 class="font-bold {% if observation.department_responded_at %}text-navy{% elif observation.dept_response_is_overdue %}text-red-700{% else %}text-amber-800{% endif %} flex items-center gap-2">
|
|
<i data-lucide="building-2" class="w-4 h-4"></i>
|
|
{% blocktrans with dept=observation.assigned_department.get_localized_name|default:observation.assigned_department.name %}Response from {{ dept }}{% endblocktrans %}
|
|
{% if observation.department_responded_at %}
|
|
<span class="text-xs font-normal text-navy ml-2">
|
|
<i data-lucide="check-circle" class="w-3 h-3 inline mr-1"></i>
|
|
{% trans "Received" %}
|
|
</span>
|
|
{% elif observation.dept_response_is_overdue %}
|
|
<span class="text-xs font-normal text-red-600 ml-2">
|
|
<i data-lucide="alert-triangle" class="w-3 h-3 inline mr-1"></i>
|
|
{% trans "OVERDUE" %}
|
|
</span>
|
|
{% else %}
|
|
<span class="text-xs font-normal text-amber-600 ml-2">
|
|
<i data-lucide="clock" class="w-3 h-3 inline mr-1"></i>
|
|
{% trans "Awaiting" %}
|
|
</span>
|
|
{% endif %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4">
|
|
{% if observation.forwarded_to_dept_at %}
|
|
<div class="flex items-center gap-3 mb-3 text-sm text-slate-600">
|
|
<span class="flex items-center gap-1">
|
|
<i data-lucide="send" class="w-4 h-4"></i>
|
|
{% trans "Sent:" %} {{ observation.forwarded_to_dept_at|date:"Y-m-d H:i" }}
|
|
</span>
|
|
{% if observation.dept_response_sla_due_at and not observation.department_responded_at %}
|
|
<span>•</span>
|
|
<span class="flex items-center gap-1 {% if observation.dept_response_is_overdue %}text-red-600 font-semibold{% endif %}">
|
|
<i data-lucide="timer" class="w-4 h-4"></i>
|
|
{% trans "Deadline:" %} {{ observation.dept_response_sla_due_at|date:"Y-m-d H:i" }}
|
|
</span>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if observation.department_responded_at %}
|
|
<div class="flex items-center gap-3 mb-3 text-sm text-slate-600">
|
|
<span class="flex items-center gap-1">
|
|
<i data-lucide="user" class="w-4 h-4"></i>
|
|
{{ observation.department_responded_by.get_full_name|default:"-" }}
|
|
</span>
|
|
<span>•</span>
|
|
<span class="flex items-center gap-1">
|
|
<i data-lucide="calendar" class="w-4 h-4"></i>
|
|
{{ observation.department_responded_at|date:"Y-m-d H:i" }}
|
|
</span>
|
|
</div>
|
|
{% if observation.department_response_en %}
|
|
<div class="mb-3">
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">English</p>
|
|
<div class="prose prose-sm max-w-none bg-slate-50 rounded-xl p-3">
|
|
<p class="text-sm text-slate-700 leading-relaxed">{{ observation.department_response_en|linebreaks }}</p>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.department_response_ar %}
|
|
<div class="mb-3">
|
|
<p class="text-xs font-semibold text-slate uppercase tracking-wide mb-1">العربية</p>
|
|
<div class="prose prose-sm max-w-none bg-slate-50 rounded-xl p-3">
|
|
<p class="text-sm text-slate-700 leading-relaxed" dir="rtl">{{ observation.department_response_ar|linebreaks }}</p>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
{% if observation.department_response_summary_en %}
|
|
<div class="mt-3 bg-light border border-blue-200 rounded-xl p-3">
|
|
<div class="flex items-center gap-2 mb-2">
|
|
<i data-lucide="sparkles" class="w-4 h-4 text-navy"></i>
|
|
<span class="text-sm font-bold text-navy">{% trans "AI Summary" %}</span>
|
|
</div>
|
|
<p class="text-sm text-slate-700 leading-relaxed">{{ observation.department_response_summary_en }}</p>
|
|
{% if observation.department_response_summary_ar %}
|
|
<p class="text-sm text-slate-700 leading-relaxed mt-2" dir="rtl">{{ observation.department_response_summary_ar }}</p>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
<!-- Acceptance Review -->
|
|
{% if observation.dept_response_acceptance_status %}
|
|
<div class="mt-3 border rounded-xl p-3
|
|
{% if observation.dept_response_acceptance_status == 'acceptable' %}border-green-200 bg-green-50
|
|
{% elif observation.dept_response_acceptance_status == 'not_acceptable' %}border-red-200 bg-red-50
|
|
{% else %}border-yellow-200 bg-yellow-50{% endif %}">
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center gap-2">
|
|
{% if observation.dept_response_acceptance_status == 'acceptable' %}
|
|
<i data-lucide="check-circle" class="w-4 h-4 text-green-600"></i>
|
|
<span class="text-sm font-semibold text-green-700">{% trans "Accepted" %}</span>
|
|
{% elif observation.dept_response_acceptance_status == 'not_acceptable' %}
|
|
<i data-lucide="x-circle" class="w-4 h-4 text-red-600"></i>
|
|
<span class="text-sm font-semibold text-red-700">{% trans "Not Acceptable" %}</span>
|
|
{% else %}
|
|
<i data-lucide="clock" class="w-4 h-4 text-yellow-600"></i>
|
|
<span class="text-sm font-semibold text-yellow-700">{% trans "Pending Review" %}</span>
|
|
{% endif %}
|
|
{% if observation.dept_response_accepted_by %}
|
|
<span class="text-xs text-slate-500">by {{ observation.dept_response_accepted_by.get_full_name }}</span>
|
|
{% endif %}
|
|
</div>
|
|
{% if can_review_dept_response and observation.dept_response_acceptance_status == 'pending' %}
|
|
<div class="flex gap-2">
|
|
<form method="post" action="{% url 'observations:observation_review_dept_response' observation.pk %}">
|
|
{% csrf_token %}
|
|
<input type="hidden" name="acceptance_status" value="acceptable">
|
|
<button type="submit" class="px-3 py-1.5 text-xs font-semibold text-white bg-green-600 rounded-lg hover:bg-green-700 transition flex items-center gap-1">
|
|
<i data-lucide="check" class="w-3 h-3"></i> {% trans "Acceptable" %}
|
|
</button>
|
|
</form>
|
|
<form method="post" action="{% url 'observations:observation_review_dept_response' observation.pk %}">
|
|
{% csrf_token %}
|
|
<input type="hidden" name="acceptance_status" value="not_acceptable">
|
|
<button type="submit" class="px-3 py-1.5 text-xs font-semibold text-white bg-red-600 rounded-lg hover:bg-red-700 transition flex items-center gap-1">
|
|
<i data-lucide="x" class="w-3 h-3"></i> {% trans "Not Acceptable" %}
|
|
</button>
|
|
</form>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
{% if observation.dept_response_acceptance_notes %}
|
|
<p class="text-xs text-slate-600 mt-2">{{ observation.dept_response_acceptance_notes }}</p>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% else %}
|
|
<div class="text-center py-4">
|
|
<div class="inline-block w-6 h-6 border-4 border-amber-200 border-t-amber-500 rounded-full animate-spin mb-2"></div>
|
|
<p class="text-slate text-sm">{% trans "Waiting for department response..." %}</p>
|
|
{% if observation.dept_response_sla_due_at %}
|
|
<p class="text-xs text-slate-400 mt-1">
|
|
{% trans "Deadline:" %} {{ observation.dept_response_sla_due_at|date:"Y-m-d H:i" }}
|
|
</p>
|
|
{% endif %}
|
|
</div>
|
|
{% if can_send_reminder %}
|
|
<div class="mt-3 flex gap-2">
|
|
<form method="post" action="{% url 'observations:observation_send_dept_response_reminder' observation.pk %}" class="flex-1">
|
|
{% csrf_token %}
|
|
<input type="hidden" name="reminder_type" value="first">
|
|
<button type="submit" class="w-full bg-amber-500 text-white px-4 py-2 rounded-xl font-semibold hover:bg-amber-600 transition flex items-center justify-center gap-2 text-sm">
|
|
<i data-lucide="bell" class="w-4 h-4"></i> {% trans "Send Reminder" %}
|
|
</button>
|
|
</form>
|
|
{% if observation.dept_response_reminder_sent_at %}
|
|
<form method="post" action="{% url 'observations:observation_send_dept_response_reminder' observation.pk %}" class="flex-1">
|
|
{% csrf_token %}
|
|
<input type="hidden" name="reminder_type" value="second">
|
|
<button type="submit" class="w-full bg-red-500 text-white px-4 py-2 rounded-xl font-semibold hover:bg-red-600 transition flex items-center justify-center gap-2 text-sm">
|
|
<i data-lucide="alert-triangle" class="w-4 h-4"></i> {% trans "Urgent Reminder" %}
|
|
</button>
|
|
</form>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
{% endif %}
|
|
{% if can_respond_to_department %}
|
|
<div class="mt-3">
|
|
<a href="{% url 'observations:observation_department_response' observation.pk %}"
|
|
class="w-full bg-navy text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-blue transition flex items-center justify-center gap-2 text-sm">
|
|
<i data-lucide="message-square" class="w-4 h-4"></i>
|
|
{% if observation.department_responded_at %}{% trans "Update Response" %}{% else %}{% trans "Submit Response" %}{% endif %}
|
|
</a>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
<!-- Linked Action -->
|
|
{% if px_action %}
|
|
<section class="bg-navy rounded-2xl p-5 shadow-lg text-white">
|
|
<h3 class="font-bold flex items-center gap-2 mb-3">
|
|
<i data-lucide="link" class="w-4 h-4"></i> {% trans "Linked PX Action" %}
|
|
</h3>
|
|
<p class="text-sm opacity-90 mb-3">{{ px_action.title|truncatewords:10 }}</p>
|
|
<a href="{% url 'actions:action_detail' px_action.id %}" class="inline-flex items-center gap-2 px-4 py-2 bg-white text-navy rounded-xl font-semibold text-sm hover:bg-gray-100 transition">
|
|
<i data-lucide="arrow-right" class="w-4 h-4"></i> {% trans "View Action" %}
|
|
</a>
|
|
</section>
|
|
{% endif %}
|
|
|
|
<!-- RCA Section -->
|
|
{% if linked_rcas %}
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 border-b border-slate-100 bg-purple-50">
|
|
<h3 class="font-bold text-purple-700 flex items-center gap-2">
|
|
<i data-lucide="search" class="w-4 h-4"></i> {% trans "Root Cause Analyses" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4 space-y-3">
|
|
{% for rca in linked_rcas %}
|
|
<a href="{% url 'rca:rca_detail' pk=rca.pk %}" class="block p-3 rounded-lg border border-slate-200 hover:border-purple-300 transition group">
|
|
<p class="text-sm font-bold text-navy group-hover:text-purple-700 truncate">{{ rca.title }}</p>
|
|
<div class="flex items-center gap-2 mt-1">
|
|
<span class="px-2 py-0.5 rounded-full text-[10px] font-bold uppercase
|
|
{% if rca.status == 'draft' %}bg-slate-100 text-slate-600
|
|
{% elif rca.status == 'in_progress' %}bg-blue-100 text-blue-700
|
|
{% elif rca.status == 'review' %}bg-yellow-100 text-yellow-700
|
|
{% elif rca.status == 'approved' %}bg-green-100 text-green-700
|
|
{% elif rca.status == 'closed' %}bg-gray-100 text-gray-600{% endif %}">
|
|
{{ rca.get_status_display }}
|
|
</span>
|
|
<span class="text-[10px] text-slate">{{ rca.created_at|date:"M d, Y" }}</span>
|
|
</div>
|
|
</a>
|
|
{% endfor %}
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
<!-- Triage Form -->
|
|
{% if can_triage %}
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 bg-navy text-white">
|
|
<h3 class="font-bold flex items-center gap-2">
|
|
<i data-lucide="sliders-horizontal" class="w-4 h-4"></i> {% trans "Triage" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4">
|
|
<form method="post" action="{% url 'observations:observation_triage' observation.id %}">
|
|
{% csrf_token %}
|
|
<div class="mb-3">
|
|
<label class="block text-xs font-semibold text-slate uppercase tracking-wide mb-2">{% trans "Department" %}</label>
|
|
{{ triage_form.assigned_department }}
|
|
</div>
|
|
<div class="mb-3">
|
|
<label class="block text-xs font-semibold text-slate uppercase tracking-wide mb-2">{% trans "Assign To" %}</label>
|
|
{{ triage_form.assigned_to }}
|
|
</div>
|
|
<div class="mb-3">
|
|
<label class="block text-xs font-semibold text-slate uppercase tracking-wide mb-2">{% trans "Status" %}</label>
|
|
{{ triage_form.status }}
|
|
</div>
|
|
<div class="mb-3">
|
|
<label class="block text-xs font-semibold text-slate uppercase tracking-wide mb-2">{% trans "Note" %}</label>
|
|
{{ triage_form.note }}
|
|
</div>
|
|
<button type="submit" class="w-full bg-navy text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-blue transition flex items-center justify-center gap-2">
|
|
<i data-lucide="check-circle" class="w-4 h-4"></i> {% trans "Update" %}
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
<!-- Add Note -->
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 bg-green-500 text-white">
|
|
<h3 class="font-bold flex items-center gap-2">
|
|
<i data-lucide="message-square" class="w-4 h-4"></i> {% trans "Add Note" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4">
|
|
<form method="post" action="{% url 'observations:observation_add_note' observation.id %}">
|
|
{% csrf_token %}
|
|
<div class="mb-3">
|
|
{{ note_form.note }}
|
|
</div>
|
|
<div class="mb-3">
|
|
<label class="flex items-center gap-2 cursor-pointer">
|
|
{{ note_form.is_internal }}
|
|
<span class="text-xs font-medium text-slate">{% trans "Internal note (not visible to public)" %}</span>
|
|
</label>
|
|
</div>
|
|
<button type="submit" class="w-full bg-green-500 text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-green-600 transition flex items-center justify-center gap-2">
|
|
<i data-lucide="plus-circle" class="w-4 h-4"></i> {% trans "Add Note" %}
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
|
|
<!-- Quick Status Change -->
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 bg-orange-500 text-white">
|
|
<h3 class="font-bold flex items-center gap-2">
|
|
<i data-lucide="refresh-cw" class="w-4 h-4"></i> {% trans "Quick Status Change" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4">
|
|
<form method="post" action="{% url 'observations:observation_change_status' observation.id %}">
|
|
{% csrf_token %}
|
|
<div class="mb-3">
|
|
{{ status_form.status }}
|
|
</div>
|
|
<div class="mb-3">
|
|
{{ status_form.comment }}
|
|
</div>
|
|
<button type="submit" class="w-full bg-orange-500 text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-orange-600 transition flex items-center justify-center gap-2">
|
|
<i data-lucide="check" class="w-4 h-4"></i> {% trans "Change Status" %}
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
|
|
{% if observation.status == 'resolved' or observation.status == 'closed' %}
|
|
<!-- Reopen -->
|
|
{% if can_triage %}
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 bg-amber-500 text-white">
|
|
<h3 class="font-bold flex items-center gap-2">
|
|
<i data-lucide="rotate-ccw" class="w-4 h-4"></i> {% trans "Reopen Observation" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4">
|
|
<form method="post" action="{% url 'observations:observation_reopen' observation.id %}">
|
|
{% csrf_token %}
|
|
<div class="mb-3">
|
|
<label class="block text-xs font-semibold text-slate uppercase tracking-wide mb-2">{% trans "Reopen Note" %}</label>
|
|
<textarea name="note" rows="2" class="w-full border border-slate-200 rounded-xl p-3 text-sm focus:ring-2 focus:ring-navy/20 outline-none" placeholder="{% trans 'Reason for reopening...' %}"></textarea>
|
|
</div>
|
|
<button type="submit" class="w-full bg-amber-500 text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-amber-600 transition flex items-center justify-center gap-2">
|
|
<i data-lucide="rotate-ccw" class="w-4 h-4"></i> {% trans "Reopen" %}
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% if can_send_to_department and not observation.department_responded_at %}
|
|
<!-- Send to Department -->
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="p-4">
|
|
<button onclick="showSendModal('{{ observation.id }}', 'observation')" class="w-full bg-indigo-500 text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-indigo-600 transition flex items-center justify-center gap-2">
|
|
<i data-lucide="send" class="w-4 h-4"></i> {% trans "Send to Department" %}
|
|
</button>
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
{% if can_convert %}
|
|
<!-- Assign/Reassign -->
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 bg-blue text-white">
|
|
<h3 class="font-bold flex items-center gap-2">
|
|
<i data-lucide="user-plus" class="w-4 h-4"></i>
|
|
{% if observation.assigned_to %}{% trans "Reassign" %}{% else %}{% trans "Assign" %}{% endif %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4">
|
|
<form method="post" action="{% url 'observations:observation_assign' observation.id %}">
|
|
{% csrf_token %}
|
|
<div class="mb-3">
|
|
<label class="block text-xs font-semibold text-slate uppercase tracking-wide mb-2">{% trans "Assign To" %}</label>
|
|
<select name="user_id" class="w-full border border-slate-200 rounded-xl p-3 text-sm focus:ring-2 focus:ring-navy/20 outline-none" required>
|
|
<option value="">{% trans "Select User" %}</option>
|
|
{% for u in assignable_users %}
|
|
<option value="{{ u.id }}" {% if observation.assigned_to and observation.assigned_to.id == u.id %}selected{% endif %}>{{ u.get_full_name }}</option>
|
|
{% endfor %}
|
|
</select>
|
|
</div>
|
|
<button type="submit" class="w-full bg-navy text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-blue transition flex items-center justify-center gap-2">
|
|
<i data-lucide="user-plus" class="w-4 h-4"></i>
|
|
{% if observation.assigned_to %}{% trans "Reassign" %}{% else %}{% trans "Assign" %}{% endif %}
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
{% if can_delete %}
|
|
<section class="bg-white rounded-2xl shadow-sm border border-slate-100 overflow-hidden">
|
|
<div class="px-4 py-3 bg-red-500 text-white">
|
|
<h3 class="font-bold flex items-center gap-2">
|
|
<i data-lucide="trash-2" class="w-4 h-4"></i> {% trans "Delete" %}
|
|
</h3>
|
|
</div>
|
|
<div class="p-4">
|
|
<form method="post" action="{% url 'observations:observation_soft_delete' observation.id %}" id="obs-delete-form">
|
|
{% csrf_token %}
|
|
<button type="button" onclick="if(confirm('{% trans "Are you sure you want to delete this observation? It can be restored from the trash." %}'))document.getElementById('obs-delete-form').submit()"
|
|
class="w-full bg-red-500 text-white px-4 py-2.5 rounded-xl font-semibold hover:bg-red-600 transition flex items-center justify-center gap-2">
|
|
<i data-lucide="trash-2" class="w-4 h-4"></i> {% trans "Delete Observation" %}
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
</div>
|
|
</div>
|
|
|
|
<script>
|
|
function switchObsRecipientType(type) {
|
|
const staffLabel = document.getElementById('obsRecipientLabelStaff');
|
|
const deptEmailLabel = document.getElementById('obsRecipientLabelDeptEmail');
|
|
const hint = document.getElementById('obsDeptEmailHint');
|
|
|
|
if (type === 'staff') {
|
|
staffLabel.classList.add('border-indigo-500', 'bg-indigo-50', 'text-indigo-700', 'font-semibold');
|
|
staffLabel.classList.remove('border-slate-200', 'text-slate-500');
|
|
deptEmailLabel.classList.remove('border-indigo-500', 'bg-indigo-50', 'text-indigo-700', 'font-semibold');
|
|
deptEmailLabel.classList.add('border-slate-200', 'text-slate-500');
|
|
if (hint) hint.style.display = 'none';
|
|
} else {
|
|
deptEmailLabel.classList.add('border-indigo-500', 'bg-indigo-50', 'text-indigo-700', 'font-semibold');
|
|
deptEmailLabel.classList.remove('border-slate-200', 'text-slate-500');
|
|
staffLabel.classList.remove('border-indigo-500', 'bg-indigo-50', 'text-indigo-700', 'font-semibold');
|
|
staffLabel.classList.add('border-slate-200', 'text-slate-500');
|
|
if (hint) hint.style.display = '';
|
|
}
|
|
}
|
|
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
if (typeof lucide !== 'undefined') lucide.createIcons();
|
|
});
|
|
</script>
|
|
|
|
{% include "components/send_to_modal.html" with users=assignable_users departments=departments %}
|
|
|
|
{% endblock %}
|