agc-chatbot/agc-document-search.html

3571 lines
144 KiB
HTML
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>AGC Document Search - Attorney General's Chambers</title>
<meta
name="description"
content="AI-powered document search system for Attorney General's Chambers"
/>
<!-- Favicon -->
<link
rel="icon"
type="image/svg+xml"
href=""
/>
<!-- CDN Libraries -->
<!-- Tailwind CSS -->
<script src="https://cdn.tailwindcss.com"></script>
<script>
tailwind.config = {
theme: {
extend: {
colors: {
"agc-primary": "#1E40AF",
"agc-secondary": "#D97706",
"agc-success": "#059669",
"agc-warning": "#D97706",
"agc-error": "#DC2626",
"agc-gray": {
50: "#F9FAFB",
100: "#F3F4F6",
200: "#E5E7EB",
300: "#D1D5DB",
400: "#9CA3AF",
500: "#6B7280",
600: "#4B5563",
700: "#374151",
800: "#1F2937",
900: "#111827",
},
},
fontFamily: {
inter: ["Inter", "system-ui", "sans-serif"],
mono: ["Fira Code", "monospace"],
},
animation: {
"fade-in": "fadeIn 0.3s ease-in-out",
"slide-up": "slideUp 0.3s ease-out",
"slide-down": "slideDown 0.3s ease-out",
"scale-in": "scaleIn 0.2s ease-out",
"pulse-slow": "pulse 2s cubic-bezier(0.4, 0, 0.6, 1) infinite",
},
keyframes: {
fadeIn: {
"0%": { opacity: "0" },
"100%": { opacity: "1" },
},
slideUp: {
"0%": { transform: "translateY(100%)", opacity: "0" },
"100%": { transform: "translateY(0)", opacity: "1" },
},
slideDown: {
"0%": { transform: "translateY(-100%)", opacity: "0" },
"100%": { transform: "translateY(0)", opacity: "1" },
},
scaleIn: {
"0%": { transform: "scale(0.95)", opacity: "0" },
"100%": { transform: "scale(1)", opacity: "1" },
},
},
},
},
};
</script>
<!-- Alpine.js -->
<script
defer
src="https://cdn.jsdelivr.net/npm/alpinejs@3.x.x/dist/cdn.min.js"
></script>
<!-- Chart.js -->
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<!-- Google Fonts - Inter -->
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
<link
href="https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700&family=Fira+Code:wght@400;500&display=swap"
rel="stylesheet"
/>
<!-- Heroicons -->
<script
src="https://unpkg.com/heroicons@2.0.16/24/outline/index.js"
type="module"
></script>
<!-- Custom Styles -->
<style>
/* Enhanced animations and transitions */
* {
font-family: "Inter", system-ui, sans-serif;
}
.glass-morphism {
background: rgba(255, 255, 255, 0.25);
backdrop-filter: blur(10px);
-webkit-backdrop-filter: blur(10px);
border: 1px solid rgba(255, 255, 255, 0.18);
}
.loading-skeleton {
background: linear-gradient(
90deg,
#f0f0f0 25%,
#e0e0e0 50%,
#f0f0f0 75%
);
background-size: 200% 100%;
animation: loading 1.5s infinite;
}
@keyframes loading {
0% {
background-position: 200% 0;
}
100% {
background-position: -200% 0;
}
}
.hover-lift {
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
}
.hover-lift:hover {
transform: translateY(-4px);
box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.1),
0 10px 10px -5px rgba(0, 0, 0, 0.04);
}
.custom-scrollbar::-webkit-scrollbar {
width: 6px;
}
.custom-scrollbar::-webkit-scrollbar-track {
background: #f1f1f1;
border-radius: 10px;
}
.custom-scrollbar::-webkit-scrollbar-thumb {
background: #c1c1c1;
border-radius: 10px;
}
.custom-scrollbar::-webkit-scrollbar-thumb:hover {
background: #a1a1a1;
}
/* Status indicators */
.status-online {
color: #059669;
}
.status-offline {
color: #dc2626;
}
.status-loading {
color: #d97706;
}
/* Mobile optimizations */
@media (max-width: 768px) {
.mobile-hide {
display: none;
}
.mobile-full {
width: 100% !important;
}
}
/* Enhanced Component Styles for Phase 2-3 */
.btn-primary {
@apply bg-gradient-to-r from-agc-primary to-blue-600 text-white px-4 py-2 rounded-lg font-medium shadow-md hover:shadow-lg transform hover:scale-105 transition-all duration-200 disabled:opacity-50 disabled:cursor-not-allowed disabled:transform-none;
}
.btn-secondary {
@apply bg-agc-gray-100 text-agc-gray-700 px-4 py-2 rounded-lg font-medium hover:bg-agc-gray-200 transition-all duration-200 border border-agc-gray-200;
}
.btn-ghost {
@apply text-agc-gray-500 hover:text-agc-gray-700 hover:bg-agc-gray-100 transition-all duration-200 rounded-lg;
}
.input-field {
@apply w-full px-3 py-2 border border-agc-gray-300 rounded-lg focus:ring-2 focus:ring-agc-primary focus:border-agc-primary transition-all duration-200 bg-white;
}
.search-input {
@apply w-full pl-12 pr-24 py-4 text-lg border-2 border-agc-gray-200 rounded-xl focus:ring-4 focus:ring-agc-primary focus:ring-opacity-20 focus:border-agc-primary transition-all duration-300 bg-white shadow-sm;
}
.agc-card {
@apply bg-white rounded-xl shadow-sm border border-agc-gray-100 transition-all duration-200;
}
.grid-responsive {
@apply grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6;
}
.badge {
@apply inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium;
}
.badge-primary {
@apply bg-agc-primary text-white;
}
.badge-secondary {
@apply bg-agc-gray-100 text-agc-gray-800;
}
.badge-success {
@apply bg-agc-success text-white;
}
.badge-warning {
@apply bg-agc-warning text-white;
}
.badge-error {
@apply bg-agc-error text-white;
}
.doc-type-icon {
@apply w-12 h-12 rounded-lg flex items-center justify-center text-xs font-bold text-white;
}
.doc-type-lkk {
@apply bg-gradient-to-br from-agc-primary to-blue-600;
}
.doc-type-case {
@apply bg-gradient-to-br from-agc-success to-green-600;
}
.doc-type-legal {
@apply bg-gradient-to-br from-agc-warning to-orange-600;
}
.doc-type-default {
@apply bg-gradient-to-br from-agc-gray-500 to-agc-gray-600;
}
.text-gradient {
background: linear-gradient(135deg, #1e40af 0%, #3b82f6 100%);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.status-indicator {
@apply w-3 h-3 rounded-full;
}
.status-indicator.status-online {
@apply bg-agc-success;
animation: pulse 2s infinite;
}
.status-indicator.status-offline {
@apply bg-agc-error;
}
.status-indicator.status-loading {
@apply bg-agc-warning;
animation: pulse 1s infinite;
}
.pill-button {
@apply inline-flex items-center px-3 py-2 bg-agc-gray-100 text-agc-gray-700 rounded-full text-sm hover:bg-agc-primary hover:text-white transition-all duration-200 cursor-pointer;
}
.line-clamp-2 {
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
overflow: hidden;
}
.line-clamp-3 {
display: -webkit-box;
-webkit-line-clamp: 3;
-webkit-box-orient: vertical;
overflow: hidden;
}
/* Search result highlighting */
.search-highlight {
@apply bg-yellow-200 px-1 rounded font-medium;
}
/* Responsive utilities */
@media (max-width: 640px) {
.grid-responsive {
@apply grid-cols-1;
}
}
/* AI Assistant Styles - Phase 4 */
.ai-suggestion-btn {
@apply flex items-center justify-center px-4 py-3 bg-gradient-to-br from-purple-50 to-indigo-50 text-purple-700 rounded-lg border border-purple-200 hover:from-purple-100 hover:to-indigo-100 hover:border-purple-300 transition-all duration-200 text-sm font-medium;
}
.ai-suggestion-btn:hover {
transform: translateY(-1px);
box-shadow: 0 4px 12px rgba(139, 92, 246, 0.15);
}
/* Chat message animations */
.chat-message-enter {
animation: chatMessageSlide 0.3s ease-out;
}
@keyframes chatMessageSlide {
from {
opacity: 0;
transform: translateY(10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
/* Chart container styles */
.chart-container {
position: relative;
height: 300px;
width: 100%;
}
.chart-legend {
@apply flex items-center justify-center space-x-4 text-sm text-agc-gray-600 mt-4;
}
.chart-legend-item {
@apply flex items-center space-x-2;
}
.chart-legend-color {
@apply w-3 h-3 rounded-full;
}
</style>
</head>
<body class="bg-agc-gray-50 font-inter antialiased">
<!-- Main Application Container -->
<div
id="app"
x-data="agcApp()"
x-init="initializeApp()"
class="min-h-screen flex flex-col"
>
<!-- Loading Overlay -->
<div
x-show="appState.isInitializing"
x-transition:enter="transition ease-out duration-300"
x-transition:enter-start="opacity-0"
x-transition:enter-end="opacity-100"
x-transition:leave="transition ease-in duration-200"
x-transition:leave-start="opacity-100"
x-transition:leave-end="opacity-0"
class="fixed inset-0 bg-white z-50 flex items-center justify-center"
>
<div class="text-center">
<div
class="w-16 h-16 border-4 border-agc-primary border-t-transparent rounded-full animate-spin mx-auto mb-4"
></div>
<h2 class="text-xl font-semibold text-agc-gray-800 mb-2">
AGC Document Search
</h2>
<p class="text-agc-gray-600">Initializing system...</p>
</div>
</div>
<!-- Header -->
<header
class="bg-white shadow-lg border-b border-agc-gray-200 sticky top-0 z-40"
style="background: linear-gradient(135deg, #ffffff 0%, #f8fafc 100%)"
>
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
<div class="flex justify-between items-center h-18">
<!-- Enhanced Logo and Branding -->
<div class="flex items-center space-x-4">
<div class="flex-shrink-0">
<div
class="w-12 h-12 bg-gradient-to-br from-agc-primary to-blue-600 rounded-xl flex items-center justify-center shadow-lg hover:shadow-xl transition-all duration-300"
>
<svg
class="w-7 h-7 text-white"
fill="currentColor"
viewBox="0 0 24 24"
>
<path
d="M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5"
/>
</svg>
</div>
</div>
<div class="py-2">
<h1 class="text-xl font-bold text-agc-gray-900 tracking-tight">
<span class="text-gradient">AGC Document Search</span>
</h1>
<p class="text-sm text-agc-gray-600 mobile-hide font-medium">
Attorney General's Chambers Malaysia
</p>
</div>
</div>
<!-- Enhanced Status and Info Panel -->
<div class="flex items-center space-x-6">
<!-- Connection Status with Enhanced Indicator -->
<div class="flex items-center space-x-3">
<div class="flex items-center space-x-2">
<div
class="status-indicator"
:class="{
'status-online': appState.isOnline && appState.apiStatus === 'connected',
'status-loading': appState.apiStatus === 'connecting',
'status-offline': !appState.isOnline || appState.apiStatus === 'disconnected'
}"
></div>
<div class="mobile-hide">
<p
class="text-sm font-medium text-agc-gray-900"
x-text="appState.isOnline ? 'System Online' : 'System Offline'"
></p>
<p
class="text-xs text-agc-gray-500"
x-text="'API: ' + appState.apiStatus"
></p>
</div>
</div>
</div>
<!-- Enhanced Stats Display -->
<div
class="mobile-hide bg-agc-gray-50 rounded-lg px-4 py-2 border border-agc-gray-200"
>
<div class="flex items-center space-x-4 text-sm">
<div class="flex items-center space-x-2">
<div class="w-2 h-2 bg-agc-primary rounded-full"></div>
<span class="font-medium text-agc-gray-700">
<span x-text="appState.stats.totalDocuments"></span>
<span class="text-agc-gray-500">Documents</span>
</span>
</div>
<div class="w-px h-4 bg-agc-gray-300"></div>
<div class="flex items-center space-x-2">
<div class="w-2 h-2 bg-agc-secondary rounded-full"></div>
<span class="font-medium text-agc-gray-700">
<span x-text="appState.stats.searchCount"></span>
<span class="text-agc-gray-500">Searches</span>
</span>
</div>
</div>
</div>
<!-- Settings/Menu Button -->
<button
class="btn-ghost p-2 rounded-lg hover:bg-agc-gray-100 transition-colors"
>
<svg
class="w-5 h-5 text-agc-gray-600"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z"
></path>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"
></path>
</svg>
</button>
</div>
</div>
</div>
<!-- Optional: Quick Stats Bar (shown on mobile) -->
<div
class="lg:hidden bg-agc-gray-50 border-t border-agc-gray-200 px-4 py-2"
>
<div class="flex justify-between items-center text-sm">
<span class="text-agc-gray-600">
<span x-text="appState.stats.totalDocuments"></span> docs
</span>
<span class="text-agc-gray-600">
<span x-text="appState.stats.searchCount"></span> searches
</span>
<span
:class="appState.apiStatus === 'connected' ? 'text-agc-success' : 'text-agc-error'"
>
<span x-text="appState.apiStatus"></span>
</span>
</div>
</div>
</header>
<!-- Main Content Area -->
<main class="flex-1 max-w-7xl mx-auto w-full px-4 sm:px-6 lg:px-8 py-8">
<!-- Enhanced Navigation Tabs -->
<div class="mb-8">
<div
class="border-b border-agc-gray-200 bg-white rounded-t-lg shadow-sm"
>
<nav class="flex space-x-8 px-6" aria-label="Tabs">
<template x-for="tab in appState.tabs" :key="tab.id">
<button
@click="switchTab(tab.id)"
:class="{
'border-agc-primary text-agc-primary bg-blue-50': appState.activeTab === tab.id,
'border-transparent text-agc-gray-500 hover:text-agc-gray-700 hover:border-agc-gray-300': appState.activeTab !== tab.id
}"
class="relative whitespace-nowrap py-4 px-6 border-b-3 font-semibold text-sm transition-all duration-200 rounded-t-lg"
>
<div class="flex items-center space-x-2">
<!-- Tab Icons -->
<div x-show="tab.id === 'search'">
<svg
class="w-5 h-5"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"
/>
</svg>
</div>
<div x-show="tab.id === 'browse'">
<svg
class="w-5 h-5"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M19 11H5m14 0a2 2 0 012 2v6a2 2 0 01-2 2H5a2 2 0 01-2-2v-6a2 2 0 012-2m14 0V9a2 2 0 00-2-2M5 11V9a2 2 0 012-2m0 0V5a2 2 0 012-2h6a2 2 0 012 2v2M7 7h10"
/>
</svg>
</div>
<div x-show="tab.id === 'analytics'">
<svg
class="w-5 h-5"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"
/>
</svg>
</div>
<span x-text="tab.name"></span>
<!-- Enhanced Badge -->
<span
x-show="tab.badge"
x-text="tab.badge"
class="badge badge-primary"
></span>
</div>
<!-- Active Tab Indicator -->
<div
x-show="appState.activeTab === tab.id"
class="absolute inset-x-0 bottom-0 h-1 bg-gradient-to-r from-agc-primary to-blue-500 rounded-full"
></div>
</button>
</template>
</nav>
</div>
</div>
<!-- Tab Content -->
<div class="space-y-6">
<!-- Search Tab -->
<div
x-show="appState.activeTab === 'search'"
x-transition:enter="transition ease-out duration-300"
x-transition:enter-start="opacity-0 transform translate-y-4"
x-transition:enter-end="opacity-100 transform translate-y-0"
>
<!-- Main Search Interface -->
<div class="agc-card p-8">
<div class="text-center mb-8">
<h2 class="text-3xl font-bold text-agc-gray-900 mb-3">
<span class="text-gradient"
>AI-Powered Legal Document Search</span
>
</h2>
<p class="text-lg text-agc-gray-600 max-w-2xl mx-auto">
Intelligent search through Attorney General's Chambers legal
documents using advanced AI technology
</p>
</div>
<!-- Search Input Section -->
<div class="max-w-4xl mx-auto mb-8">
<div class="relative">
<div
class="absolute inset-y-0 left-0 pl-4 flex items-center pointer-events-none"
>
<svg
class="h-6 w-6 text-agc-gray-400"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"
/>
</svg>
</div>
<input
type="text"
placeholder="Ask me anything about legal documents... (e.g., 'Find cases about cybercrime from 2023')"
class="search-input w-full"
x-model="appState.searchQuery"
@keydown.enter="performSearch(appState.searchQuery)"
/>
<div
class="absolute inset-y-0 right-0 flex items-center pr-3"
>
<button
class="btn-primary px-6 py-2 rounded-lg flex items-center space-x-2"
@click="performSearch(appState.searchQuery)"
:disabled="appState.loading.search"
>
<span x-show="!appState.loading.search">Search</span>
<span x-show="appState.loading.search">Searching...</span>
</button>
</div>
</div>
</div>
<!-- Quick Search Suggestions -->
<div class="max-w-4xl mx-auto">
<h3 class="text-lg font-semibold text-agc-gray-900 mb-4">
Quick Searches
</h3>
<div
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4"
>
<template
x-for="suggestion in [
'Recent criminal cases',
'Constitutional law decisions',
'Corporate litigation',
'Human rights cases',
'Tax law judgments',
'Employment disputes'
]"
:key="suggestion"
>
<button
class="agc-card p-4 text-left hover-lift border-l-4 border-agc-primary"
@click="appState.searchQuery = suggestion; performSearch(suggestion)"
>
<div class="flex items-center space-x-3">
<div
class="w-10 h-10 bg-agc-primary bg-opacity-10 rounded-lg flex items-center justify-center"
>
<svg
class="w-5 h-5 text-agc-primary"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"
/>
</svg>
</div>
<div>
<p
class="font-medium text-agc-gray-900"
x-text="suggestion"
></p>
<p class="text-sm text-agc-gray-500">
Click to search
</p>
</div>
</div>
</button>
</template>
</div>
</div>
</div>
<!-- Search Results Section -->
<div
x-show="appState.searchResults && appState.searchResults.documents"
class="space-y-6"
>
<!-- Results Header -->
<div class="agc-card p-6">
<div class="flex items-center justify-between mb-4">
<h3 class="text-xl font-semibold text-agc-gray-900">
Search Results
</h3>
<div class="flex items-center space-x-4">
<span class="text-sm text-agc-gray-500">
<span
x-text="appState.searchResults.documents?.length || 0"
></span>
results found
</span>
<button
class="btn-secondary text-sm"
@click="exportResults()"
>
Export Results
</button>
</div>
</div>
<!-- AI Answer Section -->
<div
x-show="appState.searchResults.answer"
class="bg-gradient-to-r from-blue-50 to-indigo-50 border border-blue-200 rounded-lg p-6 mb-6"
>
<div class="flex items-start space-x-3">
<div
class="w-8 h-8 bg-agc-primary rounded-lg flex items-center justify-center flex-shrink-0"
>
<svg
class="w-5 h-5 text-white"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M9.663 17h4.673M12 3v1m6.364 1.636l-.707.707M21 12h-1M4 12H3m3.343-5.657l-.707-.707m2.828 9.9a5 5 0 117.072 0l-.548.547A3.374 3.374 0 0014 18.469V19a2 2 0 11-4 0v-.531c0-.895-.356-1.754-.988-2.386l-.548-.547z"
/>
</svg>
</div>
<div class="flex-1">
<h4 class="font-semibold text-agc-gray-900 mb-2">
AI Analysis
</h4>
<p
class="text-agc-gray-700 leading-relaxed"
x-text="appState.searchResults.answer"
></p>
<div class="mt-3 flex items-center space-x-4 text-sm">
<button
class="text-agc-primary hover:text-agc-primary font-medium"
>
<svg
class="w-4 h-4 inline mr-1"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M8 16H6a2 2 0 01-2-2V6a2 2 0 012-2h8a2 2 0 012 2v2m-6 12h8a2 2 0 002-2v-8a2 2 0 00-2-2h-8a2 2 0 00-2 2v8a2 2 0 002 2z"
/>
</svg>
Copy Answer
</button>
<button
class="text-agc-primary hover:text-agc-primary font-medium"
>
<svg
class="w-4 h-4 inline mr-1"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M8.684 13.342C8.886 12.938 9 12.482 9 12c0-.482-.114-.938-.316-1.342m0 2.684a3 3 0 110-2.684m0 2.684l6.632 3.316m-6.632-6l6.632-3.316m0 0a3 3 0 105.367-2.684 3 3 0 00-5.367 2.684zm0 9.316a3 3 0 105.367 2.684 3 3 0 00-5.367-2.684z"
/>
</svg>
Share
</button>
</div>
</div>
</div>
</div>
<!-- Enhanced Query Display -->
<div
x-show="appState.searchResults.enhanced_query"
class="bg-agc-gray-50 border border-agc-gray-200 rounded-lg p-4"
>
<div class="flex items-center space-x-2 mb-2">
<svg
class="w-4 h-4 text-agc-primary"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M13 10V3L4 14h7v7l9-11h-7z"
/>
</svg>
<span class="text-sm font-medium text-agc-gray-700"
>Enhanced Query</span
>
</div>
<p
class="text-sm text-agc-gray-600"
x-text="appState.searchResults.enhanced_query"
></p>
</div>
</div>
<!-- Search Results Filter Bar -->
<div class="agc-card p-4">
<div
class="flex flex-col sm:flex-row sm:items-center sm:justify-between space-y-3 sm:space-y-0"
>
<div class="flex items-center space-x-4">
<label class="flex items-center space-x-2 text-sm">
<input
type="checkbox"
class="rounded border-agc-gray-300 text-agc-primary"
/>
<span>Show only high relevance (>70%)</span>
</label>
<select class="input-field py-1 text-sm w-auto">
<option>All document types</option>
<option>LKK only</option>
<option>Cases only</option>
<option>Legal documents only</option>
</select>
</div>
<div
class="flex items-center space-x-2 text-sm text-agc-gray-500"
>
<span>Sort by:</span>
<select class="input-field py-1 text-sm w-auto">
<option>Relevance</option>
<option>Date (newest)</option>
<option>Date (oldest)</option>
<option>Title A-Z</option>
</select>
</div>
</div>
</div>
<!-- Results List -->
<div class="space-y-4">
<template
x-for="(doc, index) in appState.searchResults.documents"
:key="doc.id || index"
>
<div
class="agc-card hover-lift p-6 transition-all duration-300 border-l-4"
:class="{
'border-green-400 bg-green-50': doc.similarity > 0.8,
'border-blue-400 bg-blue-50': doc.similarity > 0.6 && doc.similarity <= 0.8,
'border-yellow-400 bg-yellow-50': doc.similarity > 0.4 && doc.similarity <= 0.6,
'border-gray-400 bg-gray-50': doc.similarity <= 0.4 || !doc.similarity
}"
>
<div class="flex items-start space-x-4">
<!-- Document Type Icon -->
<div class="flex-shrink-0">
<div
class="doc-type-icon"
:class="getDocTypeClass(doc.doc_type)"
>
<span
x-text="formatDocumentType(doc.doc_type)"
></span>
</div>
</div>
<!-- Document Content -->
<div class="flex-1 min-w-0">
<div class="flex items-start justify-between mb-3">
<div class="flex-1">
<h4
class="text-lg font-semibold text-agc-gray-900 mb-2 cursor-pointer hover:text-agc-primary"
@click="viewDocument(doc)"
x-text="doc.title"
></h4>
<!-- Document Metadata -->
<div
class="flex items-center space-x-4 text-sm text-agc-gray-500 mb-3"
>
<span
x-text="'Type: ' + (doc.doc_type || 'Document')"
></span>
<span
x-show="doc.created_at"
x-text="'Date: ' + utils.formatDate(doc.created_at)"
></span>
<span
x-show="doc.source"
x-text="'Source: ' + doc.source"
></span>
</div>
</div>
<!-- Action Buttons and Relevance -->
<div
class="flex items-center space-x-3 ml-4 flex-shrink-0"
>
<span
x-show="doc.similarity"
class="badge badge-primary text-xs"
>
<span
x-text="Math.round(doc.similarity * 100)"
></span
>% match
</span>
<button
class="btn-ghost p-2"
:class="{ 'text-agc-primary': isBookmarked(doc.id) }"
@click="toggleBookmark(doc.id)"
:title="isBookmarked(doc.id) ? 'Remove bookmark' : 'Add bookmark'"
>
<svg
class="w-4 h-4"
:fill="isBookmarked(doc.id) ? 'currentColor' : 'none'"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M5 5a2 2 0 012-2h10a2 2 0 012 2v16l-7-3.5L5 21V5z"
/>
</svg>
</button>
<div class="relative" x-data="{ open: false }">
<button
class="btn-ghost p-2"
@click="open = !open"
>
<svg
class="w-4 h-4"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M12 5v.01M12 12v.01M12 19v.01M12 6a1 1 0 110-2 1 1 0 010 2zm0 7a1 1 0 110-2 1 1 0 010 2zm0 7a1 1 0 110-2 1 1 0 010 2z"
/>
</svg>
</button>
<!-- Dropdown Menu -->
<div
x-show="open"
@click.away="open = false"
x-transition:enter="transition ease-out duration-100"
x-transition:enter-start="transform opacity-0 scale-95"
x-transition:enter-end="transform opacity-100 scale-100"
class="absolute right-0 mt-2 w-48 bg-white rounded-lg shadow-lg border border-agc-gray-200 z-10"
>
<div class="py-1">
<button
class="w-full text-left px-4 py-2 text-sm text-agc-gray-700 hover:bg-agc-gray-50"
@click="viewDocument(doc); open = false"
>
<svg
class="w-4 h-4 inline mr-2"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"
/>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z"
/>
</svg>
View Full Document
</button>
<button
class="w-full text-left px-4 py-2 text-sm text-agc-gray-700 hover:bg-agc-gray-50"
>
<svg
class="w-4 h-4 inline mr-2"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M8.684 13.342C8.886 12.938 9 12.482 9 12c0-.482-.114-.938-.316-1.342m0 2.684a3 3 0 110-2.684m0 2.684l6.632 3.316m-6.632-6l6.632-3.316m0 0a3 3 0 105.367-2.684 3 3 0 00-5.367 2.684zm0 9.316a3 3 0 105.367 2.684 3 3 0 00-5.367-2.684z"
/>
</svg>
Share Document
</button>
<button
class="w-full text-left px-4 py-2 text-sm text-agc-gray-700 hover:bg-agc-gray-50"
>
<svg
class="w-4 h-4 inline mr-2"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"
/>
</svg>
Download PDF
</button>
<button
class="w-full text-left px-4 py-2 text-sm text-agc-gray-700 hover:bg-agc-gray-50"
>
<svg
class="w-4 h-4 inline mr-2"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z"
/>
</svg>
Find Similar
</button>
</div>
</div>
</div>
</div>
</div>
<!-- Content Preview -->
<div class="mb-4">
<p
class="text-agc-gray-600 leading-relaxed"
x-html="highlightSearchTerms(doc.content_preview || utils.truncateText(doc.content, 300), appState.searchResults.query)"
></p>
</div>
<!-- Document Actions Bar -->
<div
class="flex items-center justify-between pt-4 border-t border-agc-gray-100"
>
<div class="flex items-center space-x-2">
<span
class="badge"
:class="{
'badge-primary': doc.doc_type === 'LKK',
'badge-success': doc.doc_type === 'Case',
'badge-warning': doc.doc_type === 'Legal',
'badge-error': !doc.doc_type
}"
x-text="doc.doc_type || 'Unknown'"
>
</span>
<span
x-show="doc.content"
class="text-xs text-agc-gray-500"
>
<span x-text="doc.content.length"></span>
characters
</span>
</div>
<div class="flex items-center space-x-2">
<button
class="btn-secondary text-sm px-3 py-1"
@click="viewDocument(doc)"
>
View Details
</button>
<button
class="btn-primary text-sm px-3 py-1"
@click="openDocument(doc)"
>
Open Document
</button>
</div>
</div>
</div>
</div>
</div>
</template>
</div>
<!-- Load More / Pagination -->
<div class="text-center py-8">
<button
class="btn-secondary"
x-show="appState.searchResults.documents && appState.searchResults.documents.length >= 10"
>
Load More Results
</button>
</div>
</div>
<!-- AI Assistant Panel - Phase 4 Feature -->
<div x-show="appState.searchResults && appState.searchResults.documents" class="grid grid-cols-1 lg:grid-cols-3 gap-6 mb-6">
<!-- AI Chat Interface -->
<div class="lg:col-span-2">
<div class="agc-card p-6">
<div class="flex items-center justify-between mb-6">
<div class="flex items-center space-x-3">
<div class="w-10 h-10 bg-gradient-to-br from-purple-500 to-indigo-600 rounded-lg flex items-center justify-center">
<svg class="w-6 h-6 text-white" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9.663 17h4.673M12 3v1m6.364 1.636l-.707.707M21 12h-1M4 12H3m3.343-5.657l-.707-.707m2.828 9.9a5 5 0 117.072 0l-.548.547A3.374 3.374 0 0014 18.469V19a2 2 0 11-4 0v-.531c0-.895-.356-1.754-.988-2.386l-.548-.547z"/>
</svg>
</div>
<div>
<h3 class="text-lg font-semibold text-agc-gray-900">AI Legal Assistant</h3>
<p class="text-sm text-agc-gray-600">Ask questions about legal concepts, document analysis, or get explanations</p>
</div>
</div>
<div class="flex items-center space-x-2">
<div class="status-indicator status-online"></div>
<span class="text-sm text-agc-gray-600">AI Ready</span>
</div>
</div>
<!-- Chat Messages -->
<div class="mb-6 h-64 overflow-y-auto custom-scrollbar bg-agc-gray-50 rounded-lg p-4" x-ref="chatContainer">
<div class="space-y-4" x-show="appState.aiChat.messages.length === 0">
<div class="text-center py-8">
<div class="w-16 h-16 bg-gradient-to-br from-purple-100 to-indigo-100 rounded-full flex items-center justify-center mx-auto mb-4">
<svg class="w-8 h-8 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 12h.01M12 12h.01M16 12h.01M21 12c0 4.418-4.03 8-9 8a9.863 9.863 0 01-4.255-.949L3 20l1.395-3.72C3.512 15.042 3 13.574 3 12c0-4.418 4.03-8 9-8s9 3.582 9 8z"/>
</svg>
</div>
<h4 class="text-lg font-medium text-agc-gray-900 mb-2">Start a conversation</h4>
<p class="text-agc-gray-600 mb-4">Ask me anything about legal documents, case analysis, or legal concepts.</p>
<!-- Quick Start Options -->
<div class="grid grid-cols-1 sm:grid-cols-2 gap-3 max-w-lg mx-auto">
<button class="ai-suggestion-btn" @click="askAI('Explain the difference between civil and criminal law')">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3.055 11H5a2 2 0 012 2v1a2 2 0 002 2 2 2 0 012 2v2.945M8 3.935V5.5A2.5 2.5 0 0010.5 8h.5a2 2 0 012 2 2 2 0 104 0 2 2 0 012-2h1.064M15 20.488V18a2 2 0 012-2h3.064M21 12a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
Legal Concepts
</button>
<button class="ai-suggestion-btn" @click="askAI('Summarize the key points from my recent search results')">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"/>
</svg>
Document Analysis
</button>
<button class="ai-suggestion-btn" @click="askAI('How do I search for cases related to intellectual property?')">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"/>
</svg>
Search Tips
</button>
<button class="ai-suggestion-btn" @click="askAI('What are the recent trends in Malaysian case law?')">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 7h8m0 0v8m0-8l-8 8-4-4-6 6"/>
</svg>
Legal Trends
</button>
</div>
</div>
</div>
<!-- Chat Messages -->
<div class="space-y-4" x-show="appState.aiChat.messages.length > 0">
<template x-for="(message, index) in appState.aiChat.messages" :key="index">
<div class="flex" :class="message.role === 'user' ? 'justify-end' : 'justify-start'">
<div class="flex items-start space-x-3 max-w-3xl">
<div x-show="message.role === 'assistant'" class="w-8 h-8 bg-gradient-to-br from-purple-500 to-indigo-600 rounded-full flex items-center justify-center flex-shrink-0">
<svg class="w-4 h-4 text-white" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9.663 17h4.673M12 3v1m6.364 1.636l-.707.707M21 12h-1M4 12H3m3.343-5.657l-.707-.707m2.828 9.9a5 5 0 117.072 0l-.548.547A3.374 3.374 0 0014 18.469V19a2 2 0 11-4 0v-.531c0-.895-.356-1.754-.988-2.386l-.548-.547z"/>
</svg>
</div>
<div
class="rounded-lg px-4 py-3 text-sm"
:class="{
'bg-agc-primary text-white': message.role === 'user',
'bg-white border border-agc-gray-200 text-agc-gray-800': message.role === 'assistant'
}"
>
<div x-html="formatAIMessage(message.content)"></div>
<div class="text-xs opacity-75 mt-2" x-text="utils.formatDate(message.timestamp)"></div>
</div>
<div x-show="message.role === 'user'" class="w-8 h-8 bg-agc-primary rounded-full flex items-center justify-center flex-shrink-0">
<svg class="w-4 h-4 text-white" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"/>
</svg>
</div>
</div>
</div>
</template>
</div>
<!-- Typing Indicator -->
<div x-show="appState.aiChat.isTyping" class="flex justify-start">
<div class="flex items-center space-x-3">
<div class="w-8 h-8 bg-gradient-to-br from-purple-500 to-indigo-600 rounded-full flex items-center justify-center">
<svg class="w-4 h-4 text-white" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9.663 17h4.673M12 3v1m6.364 1.636l-.707.707M21 12h-1M4 12H3m3.343-5.657l-.707-.707m2.828 9.9a5 5 0 117.072 0l-.548.547A3.374 3.374 0 0014 18.469V19a2 2 0 11-4 0v-.531c0-.895-.356-1.754-.988-2.386l-.548-.547z"/>
</svg>
</div>
<div class="bg-white border border-agc-gray-200 rounded-lg px-4 py-3">
<div class="flex space-x-1">
<div class="w-2 h-2 bg-agc-gray-400 rounded-full animate-bounce"></div>
<div class="w-2 h-2 bg-agc-gray-400 rounded-full animate-bounce" style="animation-delay: 0.1s"></div>
<div class="w-2 h-2 bg-agc-gray-400 rounded-full animate-bounce" style="animation-delay: 0.2s"></div>
</div>
</div>
</div>
</div>
</div>
<!-- Chat Input -->
<div class="flex items-end space-x-3">
<div class="flex-1">
<textarea
x-model="appState.aiChat.currentMessage"
@keydown.enter.prevent="$event.ctrlKey || $event.metaKey ? sendAIMessage() : null"
placeholder="Ask about legal concepts, document analysis, or search guidance... (Ctrl+Enter to send)"
class="input-field resize-none h-20"
:disabled="appState.aiChat.isTyping"
></textarea>
</div>
<button
class="btn-primary px-4 py-2 h-12"
@click="sendAIMessage()"
:disabled="!appState.aiChat.currentMessage.trim() || appState.aiChat.isTyping"
>
<svg x-show="!appState.aiChat.isTyping" class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 19l9 2-9-18-9 18 9-2zm0 0v-8"/>
</svg>
<div x-show="appState.aiChat.isTyping" class="w-5 h-5 border-2 border-white border-t-transparent rounded-full animate-spin"></div>
</button>
</div>
</div>
</div>
<!-- Legal Concept Explorer -->
<div class="lg:col-span-1">
<div class="agc-card p-6">
<h4 class="text-lg font-semibold text-agc-gray-900 mb-4">Legal Concept Explorer</h4>
<!-- Recent Concepts -->
<div class="mb-6">
<h5 class="text-sm font-medium text-agc-gray-700 mb-3">Recently Explained</h5>
<div class="space-y-2">
<template x-for="concept in appState.aiChat.recentConcepts" :key="concept.id">
<button
class="w-full text-left p-3 bg-agc-gray-50 hover:bg-agc-gray-100 rounded-lg transition-colors"
@click="askAI('Explain ' + concept.name)"
>
<div class="flex items-center justify-between">
<span class="text-sm font-medium text-agc-gray-900" x-text="concept.name"></span>
<svg class="w-4 h-4 text-agc-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
</div>
<p class="text-xs text-agc-gray-600 mt-1" x-text="concept.description"></p>
</button>
</template>
</div>
</div>
<!-- Common Legal Terms -->
<div class="mb-6">
<h5 class="text-sm font-medium text-agc-gray-700 mb-3">Common Legal Terms</h5>
<div class="flex flex-wrap gap-2">
<template x-for="term in ['Tort', 'Injunction', 'Precedent', 'Jurisdiction', 'Plaintiff', 'Defendant']" :key="term">
<button
class="pill-button text-xs"
@click="askAI('Define ' + term + ' in Malaysian law context')"
x-text="term"
></button>
</template>
</div>
</div>
<!-- Quick Actions -->
<div>
<h5 class="text-sm font-medium text-agc-gray-700 mb-3">Quick Actions</h5>
<div class="space-y-2">
<button class="w-full btn-secondary text-sm py-2" @click="generateDocumentSummary()">
<svg class="w-4 h-4 inline mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"/>
</svg>
Summarize Results
</button>
<button class="w-full btn-secondary text-sm py-2" @click="analyzeLegalTrends()">
<svg class="w-4 h-4 inline mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 7h8m0 0v8m0-8l-8 8-4-4-6 6"/>
</svg>
Analyze Trends
</button>
<button class="w-full btn-secondary text-sm py-2" @click="clearAIChat()">
<svg class="w-4 h-4 inline mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" 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>
Clear Chat
</button>
</div>
</div>
</div>
</div>
</div>
<!-- Search History Section -->
<div x-show="appState.searchHistory.length > 0" class="agc-card p-6">
<div class="flex items-center justify-between mb-4">
<h3 class="text-lg font-semibold text-agc-gray-900">Recent Searches</h3>
<button class="text-sm text-agc-primary hover:text-agc-primary font-medium" @click="clearSearchHistory()">
Clear History
</button>
</div>
<div class="flex flex-wrap gap-2">
<template x-for="(search, index) in appState.searchHistory.slice(0, 10)" :key="search.id">
<button class="pill-button" @click="appState.searchQuery = search.query; performSearch(search.query)">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<span x-text="search.query"></span>
<span class="text-xs text-agc-gray-400 ml-2" x-text="utils.formatDate(search.timestamp)"></span>
</button>
</template>
</div>
</div>
</div>
<!-- Browse Tab -->
<div
x-show="appState.activeTab === 'browse'"
x-transition:enter="transition ease-out duration-300"
x-transition:enter-start="opacity-0 transform translate-y-4"
x-transition:enter-end="opacity-100 transform translate-y-0"
>
<!-- Browse Header and Filters -->
<div class="agc-card p-6 mb-6">
<div
class="flex flex-col lg:flex-row lg:items-center lg:justify-between mb-6"
>
<div>
<h2 class="text-2xl font-bold text-agc-gray-900 mb-2">
Document Library
</h2>
<p class="text-agc-gray-600">
Browse and filter through all available legal documents
</p>
</div>
<!-- Bulk Actions and View Toggle -->
<div class="flex items-center space-x-4 mt-4 lg:mt-0">
<!-- Bulk Actions -->
<div
x-show="appState.selectedDocuments.length > 0"
class="flex items-center space-x-2"
>
<span class="text-sm text-agc-gray-600">
<span x-text="appState.selectedDocuments.length"></span>
selected
</span>
<button
class="btn-secondary text-sm px-3 py-1"
@click="bulkExport()"
>
Export Selected
</button>
<button
class="btn-ghost text-sm px-3 py-1"
@click="clearSelection()"
>
Clear
</button>
</div>
<!-- View Toggle -->
<div class="flex items-center space-x-2">
<span class="text-sm text-agc-gray-500">View:</span>
<div class="flex bg-agc-gray-100 rounded-lg p-1">
<button
class="px-3 py-1 rounded-md text-sm font-medium transition-all"
:class="appState.preferences.viewMode === 'grid' ? 'bg-white text-agc-gray-700 shadow-sm' : 'text-agc-gray-500 hover:text-agc-gray-700'"
@click="setViewMode('grid')"
>
<svg
class="w-4 h-4"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M4 6a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2H6a2 2 0 01-2-2V6zM14 6a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2h-2a2 2 0 01-2-2V6zM4 16a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2H6a2 2 0 01-2-2v-2zM14 16a2 2 0 012-2h2a2 2 0 012 2v2a2 2 0 01-2 2h-2a2 2 0 01-2-2v-2z"
/>
</svg>
</button>
<button
class="px-3 py-1 rounded-md text-sm font-medium transition-all"
:class="appState.preferences.viewMode === 'list' ? 'bg-white text-agc-gray-700 shadow-sm' : 'text-agc-gray-500 hover:text-agc-gray-700'"
@click="setViewMode('list')"
>
<svg
class="w-4 h-4"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M4 6h16M4 10h16M4 14h16M4 18h16"
/>
</svg>
</button>
</div>
</div>
</div>
</div>
<!-- Advanced Search and Filters Row -->
<div class="grid grid-cols-1 md:grid-cols-5 gap-4">
<!-- Search Documents -->
<div class="md:col-span-2">
<label
class="block text-sm font-medium text-agc-gray-700 mb-2"
>Search Documents</label
>
<div class="relative">
<input
type="text"
placeholder="Search by title, content, or metadata..."
class="input-field pr-10"
x-model="appState.filters.title_filter"
@input="debouncedFilter()"
/>
<div
class="absolute inset-y-0 right-0 pr-3 flex items-center"
>
<svg
class="w-4 h-4 text-agc-gray-400"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"
/>
</svg>
</div>
</div>
</div>
<!-- Document Type Filter -->
<div>
<label
class="block text-sm font-medium text-agc-gray-700 mb-2"
>Document Type</label
>
<select
class="input-field"
x-model="appState.filters.doc_type"
@change="loadDocuments(appState.filters)"
>
<template
x-for="type in appState.documentTypes"
:key="type"
>
<option :value="type" x-text="type"></option>
</template>
</select>
</div>
<!-- Date Range Filter -->
<div>
<label
class="block text-sm font-medium text-agc-gray-700 mb-2"
>Date Range</label
>
<select
class="input-field"
x-model="appState.filters.dateRange"
@change="loadDocuments(appState.filters)"
>
<option value="">All Dates</option>
<option value="today">Today</option>
<option value="week">Last 7 days</option>
<option value="month">Last 30 days</option>
<option value="year">Last year</option>
</select>
</div>
<!-- Sort Options -->
<div>
<label
class="block text-sm font-medium text-agc-gray-700 mb-2"
>Sort By</label
>
<select
class="input-field"
x-model="appState.filters.sortBy"
@change="loadDocuments(appState.filters)"
>
<option value="created_at">Date Added</option>
<option value="title">Title A-Z</option>
<option value="doc_type">Document Type</option>
<option value="content_length">Content Length</option>
</select>
</div>
</div>
<!-- Active Filters Display -->
<div x-show="hasActiveFilters()" class="mt-4">
<div class="flex items-center space-x-2 flex-wrap">
<span class="text-sm text-agc-gray-500">Active filters:</span>
<template x-if="appState.filters.title_filter">
<span class="badge badge-primary">
Search:
<span x-text="appState.filters.title_filter"></span>
<button
class="ml-1 text-white hover:text-agc-gray-200"
@click="appState.filters.title_filter = ''; loadDocuments(appState.filters)"
>
×
</button>
</span>
</template>
<template x-if="appState.filters.doc_type !== 'All Types'">
<span class="badge badge-primary">
Type: <span x-text="appState.filters.doc_type"></span>
<button
class="ml-1 text-white hover:text-agc-gray-200"
@click="appState.filters.doc_type = 'All Types'; loadDocuments(appState.filters)"
>
×
</button>
</span>
</template>
<template x-if="appState.filters.dateRange">
<span class="badge badge-primary">
Date: <span x-text="appState.filters.dateRange"></span>
<button
class="ml-1 text-white hover:text-agc-gray-200"
@click="appState.filters.dateRange = ''; loadDocuments(appState.filters)"
>
×
</button>
</span>
</template>
<button
class="text-sm text-agc-primary hover:text-agc-primary font-medium"
@click="clearAllFilters()"
>
Clear all filters
</button>
</div>
</div>
<!-- Results Summary -->
<div
class="mt-4 flex items-center justify-between text-sm text-agc-gray-600"
>
<span>
Showing <span x-text="getDisplayedCount()"></span> of
<span x-text="appState.stats.totalDocuments"></span> documents
</span>
<div class="flex items-center space-x-2">
<span>Items per page:</span>
<select
class="input-field py-1 text-sm w-auto"
x-model="appState.preferences.resultsPerPage"
@change="loadDocuments(appState.filters)"
>
<option value="10">10</option>
<option value="20">20</option>
<option value="50">50</option>
<option value="100">100</option>
</select>
</div>
</div>
</div>
<!-- Documents Grid/List Display -->
<div
:class="appState.preferences.viewMode === 'grid' ? 'grid-responsive' : 'space-y-4'"
>
<template
x-for="(doc, index) in appState.documents"
:key="doc.id || index"
>
<div
class="agc-card hover-lift transition-all duration-300 relative"
:class="appState.preferences.viewMode === 'grid' ? 'p-6' : 'p-4'"
>
<!-- Selection Checkbox -->
<div class="absolute top-4 left-4 z-10">
<input
type="checkbox"
class="rounded border-agc-gray-300 text-agc-primary"
:checked="isDocumentSelected(doc.id)"
@change="toggleDocumentSelection(doc.id)"
/>
</div>
<!-- Grid View Layout -->
<div x-show="appState.preferences.viewMode === 'grid'">
<!-- Document Header -->
<div class="flex items-start justify-between mb-4 ml-8">
<div class="doc-type-icon doc-type-lkk">
<span x-text="doc.doc_type || 'DOC'"></span>
</div>
<div class="flex items-center space-x-2">
<button
class="btn-ghost p-2"
:class="{ 'text-agc-primary': isBookmarked(doc.id) }"
@click="toggleBookmark(doc.id)"
:title="isBookmarked(doc.id) ? 'Remove bookmark' : 'Add bookmark'"
>
<svg
class="w-4 h-4"
:fill="isBookmarked(doc.id) ? 'currentColor' : 'none'"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M5 5a2 2 0 012-2h10a2 2 0 012 2v16l-7-3.5L5 21V5z"
/>
</svg>
</button>
<button
class="btn-ghost p-2"
title="More actions"
@click="showDocumentMenu(doc)"
>
<svg
class="w-4 h-4"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M12 5v.01M12 12v.01M12 19v.01M12 6a1 1 0 110-2 1 1 0 010 2zm0 7a1 1 0 110-2 1 1 0 010 2zm0 7a1 1 0 110-2 1 1 0 010 2z"
/>
</svg>
</button>
</div>
</div>
<!-- Document Content -->
<div class="mb-4 ml-8">
<h3
class="text-lg font-semibold text-agc-gray-900 mb-2 line-clamp-2 cursor-pointer hover:text-agc-primary"
x-text="doc.title"
@click="viewDocument(doc)"
></h3>
<p
class="text-agc-gray-600 text-sm mb-3 line-clamp-3"
x-text="utils.truncateText(doc.content, 150)"
></p>
<!-- Document Metadata -->
<div class="space-y-2">
<div
class="flex items-center justify-between text-xs text-agc-gray-500"
>
<span
x-text="'Type: ' + (doc.doc_type || 'Document')"
></span>
<span
x-show="doc.created_at"
x-text="utils.formatDate(doc.created_at)"
></span>
</div>
<!-- Document Stats -->
<div
class="flex items-center space-x-4 text-xs text-agc-gray-500"
>
<span
x-text="doc.content ? (doc.content.length + ' chars') : 'No content'"
></span>
<span
x-show="doc.source"
x-text="'Source: ' + doc.source"
></span>
</div>
</div>
</div>
<!-- Action Buttons -->
<div
class="flex items-center justify-between pt-4 border-t border-agc-gray-100 ml-8"
>
<div class="flex items-center space-x-2">
<span
class="badge"
:class="{
'badge-primary': doc.doc_type === 'LKK',
'badge-success': doc.doc_type === 'Case',
'badge-warning': doc.doc_type === 'Legal',
'badge-error': !doc.doc_type
}"
x-text="doc.doc_type || 'Unknown'"
>
</span>
</div>
<div class="flex items-center space-x-2">
<button
class="btn-secondary text-xs px-3 py-1"
@click="viewDocument(doc)"
>
View Details
</button>
<button
class="btn-primary text-xs px-3 py-1"
@click="openDocument(doc)"
>
Open
</button>
</div>
</div>
</div>
<!-- List View Layout -->
<div
x-show="appState.preferences.viewMode === 'list'"
class="flex items-center space-x-4 ml-8"
>
<div class="doc-type-icon doc-type-lkk flex-shrink-0">
<span x-text="doc.doc_type || 'DOC'"></span>
</div>
<div class="flex-1 min-w-0">
<h3
class="text-base font-semibold text-agc-gray-900 cursor-pointer hover:text-agc-primary truncate"
x-text="doc.title"
@click="viewDocument(doc)"
></h3>
<p
class="text-sm text-agc-gray-600 truncate"
x-text="utils.truncateText(doc.content, 100)"
></p>
</div>
<div
class="flex items-center space-x-4 text-sm text-agc-gray-500 flex-shrink-0"
>
<span x-text="doc.doc_type || 'Unknown'"></span>
<span
x-show="doc.created_at"
x-text="utils.formatDate(doc.created_at)"
></span>
<div class="flex items-center space-x-2">
<button
class="btn-ghost p-1"
:class="{ 'text-agc-primary': isBookmarked(doc.id) }"
@click="toggleBookmark(doc.id)"
>
<svg
class="w-4 h-4"
:fill="isBookmarked(doc.id) ? 'currentColor' : 'none'"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M5 5a2 2 0 012-2h10a2 2 0 012 2v16l-7-3.5L5 21V5z"
/>
</svg>
</button>
<button
class="btn-secondary text-xs px-2 py-1"
@click="viewDocument(doc)"
>
View
</button>
<button
class="btn-primary text-xs px-2 py-1"
@click="openDocument(doc)"
>
Open
</button>
</div>
</div>
</div>
</div>
</template>
</div>
<!-- Pagination -->
<div class="mt-8 flex items-center justify-between">
<div class="text-sm text-agc-gray-600">
Showing
<span
x-text="((appState.pagination.currentPage - 1) * appState.preferences.resultsPerPage) + 1"
></span>
to
<span
x-text="Math.min(appState.pagination.currentPage * appState.preferences.resultsPerPage, appState.stats.totalDocuments)"
></span>
of <span x-text="appState.stats.totalDocuments"></span> results
</div>
<div class="flex items-center space-x-2">
<button
class="btn-ghost px-3 py-1"
:disabled="appState.pagination.currentPage <= 1"
@click="goToPage(appState.pagination.currentPage - 1)"
>
Previous
</button>
<template x-for="page in getPaginationPages()" :key="page">
<button
class="px-3 py-1 rounded"
:class="page === appState.pagination.currentPage ? 'bg-agc-primary text-white' : 'text-agc-gray-600 hover:bg-agc-gray-100'"
@click="goToPage(page)"
x-text="page"
></button>
</template>
<button
class="btn-ghost px-3 py-1"
:disabled="appState.pagination.currentPage >= appState.pagination.totalPages"
@click="goToPage(appState.pagination.currentPage + 1)"
>
Next
</button>
</div>
</div>
</div>
<!-- Analytics Tab -->
<div
x-show="appState.activeTab === 'analytics'"
x-transition:enter="transition ease-out duration-300"
x-transition:enter-start="opacity-0 transform translate-y-4"
x-transition:enter-end="opacity-100 transform translate-y-0"
>
<!-- Analytics Header -->
<div class="agc-card p-6 mb-6">
<div
class="flex flex-col lg:flex-row lg:items-center lg:justify-between"
>
<div>
<h2 class="text-2xl font-bold text-agc-gray-900 mb-2">
Document Analytics
</h2>
<p class="text-agc-gray-600">
Insights and statistics about your document collection
</p>
</div>
<button class="btn-secondary text-sm mt-4 lg:mt-0">
Export Report
</button>
</div>
</div>
<!-- Key Metrics Cards -->
<div
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6 mb-8"
>
<div
class="agc-card p-6 bg-gradient-to-br from-blue-50 to-blue-100"
>
<div class="flex items-center">
<div
class="w-12 h-12 bg-agc-primary rounded-lg flex items-center justify-center mr-4"
>
<svg
class="w-6 h-6 text-white"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"
/>
</svg>
</div>
<div>
<p class="text-sm text-agc-gray-600">Total Documents</p>
<p
class="text-2xl font-bold"
x-text="appState.stats.totalDocuments"
></p>
</div>
</div>
</div>
<div
class="agc-card p-6 bg-gradient-to-br from-green-50 to-green-100"
>
<div class="flex items-center">
<div
class="w-12 h-12 bg-agc-success rounded-lg flex items-center justify-center mr-4"
>
<svg
class="w-6 h-6 text-white"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"
/>
</svg>
</div>
<div>
<p class="text-sm text-agc-gray-600">Total Searches</p>
<p
class="text-2xl font-bold"
x-text="appState.stats.searchCount"
></p>
</div>
</div>
</div>
<div
class="agc-card p-6 bg-gradient-to-br from-amber-50 to-amber-100"
>
<div class="flex items-center">
<div
class="w-12 h-12 bg-agc-secondary rounded-lg flex items-center justify-center mr-4"
>
<svg
class="w-6 h-6 text-white"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M19 11H5m14 0a2 2 0 012 2v6a2 2 0 01-2 2H5a2 2 0 01-2-2v-6a2 2 0 012-2m14 0V9a2 2 0 00-2-2M5 11V9a2 2 0 012-2m0 0V5a2 2 0 012-2h6a2 2 0 012 2v2M7 7h10"
/>
</svg>
</div>
<div>
<p class="text-sm text-agc-gray-600">Document Types</p>
<p
class="text-2xl font-bold"
x-text="appState.documentTypes.length - 1"
></p>
</div>
</div>
</div>
<div
class="agc-card p-6 bg-gradient-to-br from-purple-50 to-purple-100"
>
<div class="flex items-center">
<div
class="w-12 h-12 bg-purple-600 rounded-lg flex items-center justify-center mr-4"
>
<div
class="status-indicator"
:class="appState.apiStatus === 'connected' ? 'status-online' : 'status-offline'"
></div>
</div>
<div>
<p class="text-sm text-agc-gray-600">System Status</p>
<p
class="text-lg font-bold"
:class="appState.apiStatus === 'connected' ? 'text-agc-success' : 'text-agc-error'"
x-text="appState.apiStatus === 'connected' ? 'Online' : 'Offline'"
></p>
</div>
</div>
</div>
</div>
<!-- Charts Section -->
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
<div class="agc-card p-6">
<h3 class="text-lg font-semibold text-agc-gray-900 mb-4">
Document Distribution
</h3>
<div class="chart-container">
<canvas id="documentDistributionChart" x-ref="documentDistributionChart"></canvas>
</div>
<div class="chart-legend">
<div class="chart-legend-item">
<div class="chart-legend-color bg-agc-primary"></div>
<span>LKK Documents</span>
</div>
<div class="chart-legend-item">
<div class="chart-legend-color bg-agc-success"></div>
<span>Legal Cases</span>
</div>
<div class="chart-legend-item">
<div class="chart-legend-color bg-agc-warning"></div>
<span>Criminal Cases</span>
</div>
</div>
</div>
<div class="agc-card p-6">
<h3 class="text-lg font-semibold text-agc-gray-900 mb-4">
Search Activity Trends
</h3>
<div class="chart-container">
<canvas id="searchActivityChart" x-ref="searchActivityChart"></canvas>
</div>
<div class="chart-legend">
<div class="chart-legend-item">
<div class="chart-legend-color bg-blue-500"></div>
<span>Daily Searches</span>
</div>
<div class="chart-legend-item">
<div class="chart-legend-color bg-green-500"></div>
<span>AI Interactions</span>
</div>
</div>
</div>
</div>
<!-- Advanced Analytics -->
<div class="grid grid-cols-1 lg:grid-cols-3 gap-6 mt-8">
<!-- Search Performance -->
<div class="agc-card p-6">
<h4 class="text-lg font-semibold text-agc-gray-900 mb-4">Search Performance</h4>
<div class="space-y-4">
<div class="flex items-center justify-between">
<span class="text-sm text-agc-gray-600">Average Response Time</span>
<span class="font-semibold text-agc-gray-900">245ms</span>
</div>
<div class="flex items-center justify-between">
<span class="text-sm text-agc-gray-600">Success Rate</span>
<span class="font-semibold text-agc-success">98.5%</span>
</div>
<div class="flex items-center justify-between">
<span class="text-sm text-agc-gray-600">AI Accuracy</span>
<span class="font-semibold text-agc-primary">94.2%</span>
</div>
<div class="w-full bg-agc-gray-200 rounded-full h-2">
<div class="bg-gradient-to-r from-agc-primary to-blue-500 h-2 rounded-full" style="width: 94.2%"></div>
</div>
</div>
</div>
<!-- Top Search Terms -->
<div class="agc-card p-6">
<h4 class="text-lg font-semibold text-agc-gray-900 mb-4">Popular Search Terms</h4>
<div class="space-y-3">
<template x-for="(term, index) in ['cybercrime', 'intellectual property', 'contract law', 'constitutional rights', 'criminal procedure']" :key="term">
<div class="flex items-center justify-between">
<span class="text-sm text-agc-gray-700" x-text="term"></span>
<div class="flex items-center space-x-2">
<div class="w-16 bg-agc-gray-200 rounded-full h-1.5">
<div class="bg-agc-primary h-1.5 rounded-full" :style="'width: ' + (100 - index * 15) + '%'"></div>
</div>
<span class="text-xs text-agc-gray-500" x-text="(100 - index * 15) + '%'"></span>
</div>
</div>
</template>
</div>
</div>
<!-- User Activity -->
<div class="agc-card p-6">
<h4 class="text-lg font-semibold text-agc-gray-900 mb-4">Recent Activity</h4>
<div class="space-y-3">
<div class="flex items-start space-x-3">
<div class="w-2 h-2 bg-agc-primary rounded-full mt-2"></div>
<div>
<p class="text-sm text-agc-gray-900">New search performed</p>
<p class="text-xs text-agc-gray-500">2 minutes ago</p>
</div>
</div>
<div class="flex items-start space-x-3">
<div class="w-2 h-2 bg-agc-success rounded-full mt-2"></div>
<div>
<p class="text-sm text-agc-gray-900">Document bookmarked</p>
<p class="text-xs text-agc-gray-500">15 minutes ago</p>
</div>
</div>
<div class="flex items-start space-x-3">
<div class="w-2 h-2 bg-purple-500 rounded-full mt-2"></div>
<div>
<p class="text-sm text-agc-gray-900">AI assistant consulted</p>
<p class="text-xs text-agc-gray-500">1 hour ago</p>
</div>
</div>
<div class="flex items-start space-x-3">
<div class="w-2 h-2 bg-agc-warning rounded-full mt-2"></div>
<div>
<p class="text-sm text-agc-gray-900">Collection created</p>
<p class="text-xs text-agc-gray-500">3 hours ago</p>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- AI Assistant Panel -->
<div
x-show="appState.aiChat.isOpen"
class="fixed bottom-0 right-0 w-96 bg-white shadow-lg rounded-t-xl border border-agc-gray-200 overflow-hidden"
>
<!-- Chat Header -->
<div class="p-4 border-b border-agc-gray-200 flex justify-between items-center bg-gradient-to-r from-agc-primary to-blue-600">
<h3 class="text-lg font-semibold text-white">AI Assistant</h3>
<button
@click="toggleAIChat()"
class="text-white hover:text-agc-gray-200 transition-colors"
>
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12"></path>
</svg>
</button>
</div>
<!-- Chat Messages -->
<div class="h-96 overflow-y-auto p-4 space-y-4 bg-agc-gray-50">
<template x-for="message in appState.aiChat.messages" :key="message.timestamp">
<div
:class="message.role === 'user' ? 'ml-auto bg-agc-primary text-white' : 'mr-auto bg-white'"
class="max-w-[80%] rounded-lg p-3 shadow-sm"
>
<p x-text="message.content" class="text-sm"></p>
<span class="text-xs opacity-75 mt-1 block" x-text="new Date(message.timestamp).toLocaleTimeString()"></span>
</div>
</template>
<!-- Typing Indicator -->
<div
x-show="appState.aiChat.isTyping"
class="flex space-x-2 items-center text-agc-gray-500"
>
<span class="w-2 h-2 bg-agc-gray-400 rounded-full animate-pulse"></span>
<span class="w-2 h-2 bg-agc-gray-400 rounded-full animate-pulse delay-100"></span>
<span class="w-2 h-2 bg-agc-gray-400 rounded-full animate-pulse delay-200"></span>
</div>
</div>
<!-- Error Message -->
<div
x-show="appState.aiChat.error"
class="p-2 bg-red-100 text-red-700 text-sm"
x-text="appState.aiChat.error"
></div>
<!-- Chat Input -->
<div class="p-4 border-t border-agc-gray-200">
<form @submit.prevent="sendMessage(appState.aiChat.currentMessage)">
<div class="flex space-x-2">
<input
type="text"
x-model="appState.aiChat.currentMessage"
@input="updateCurrentMessage($event.target.value)"
placeholder="Type your message..."
class="flex-1 px-3 py-2 border border-agc-gray-300 rounded-lg focus:ring-2 focus:ring-agc-primary focus:border-agc-primary"
:disabled="appState.aiChat.isTyping"
>
<button
type="submit"
class="px-4 py-2 bg-agc-primary text-white rounded-lg hover:bg-blue-700 transition-colors disabled:opacity-50"
:disabled="!appState.aiChat.currentMessage.trim() || appState.aiChat.isTyping"
>
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 19l9 2-9-18-9 18 9-2zm0 0v-8"></path>
</svg>
</button>
</div>
</form>
</div>
</div>
<!-- AI Chat Toggle Button -->
<button
@click="toggleAIChat()"
class="fixed bottom-4 right-4 p-3 bg-agc-primary text-white rounded-full shadow-lg hover:bg-blue-700 transition-colors"
>
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 10h.01M12 10h.01M16 10h.01M9 16H5a2 2 0 01-2-2V6a2 2 0 012-2h14a2 2 0 012 2v8a2 2 0 01-2 2h-5l-5 5v-5z"></path>
</svg>
</button>
</div>
<!-- JavaScript Application Logic -->
<script>
// Phase 1: Architecture & Foundation Implementation
// Global Configuration
const CONFIG = {
API_BASE_URL: "http://localhost:8000",
API_TIMEOUT: 10000,
STORAGE_PREFIX: "agc_",
VERSION: "1.0.0",
RETRY_ATTEMPTS: 3,
RETRY_DELAY: 1000,
};
// Utility Functions
const Utils = {
// Debounce function for search inputs
debounce(func, wait) {
let timeout;
return function executedFunction(...args) {
const later = () => {
clearTimeout(timeout);
func(...args);
};
clearTimeout(timeout);
timeout = setTimeout(later, wait);
};
},
// Format date for display
formatDate(dateString) {
if (!dateString) return "N/A";
return new Date(dateString).toLocaleDateString("en-MY", {
year: "numeric",
month: "short",
day: "numeric",
});
},
// Generate unique ID
generateId() {
return Date.now().toString(36) + Math.random().toString(36).substr(2);
},
// Truncate text with ellipsis
truncateText(text, maxLength = 100) {
if (!text || text.length <= maxLength) return text;
return text.substr(0, maxLength) + "...";
},
// Check if device is mobile
isMobile() {
return window.innerWidth <= 768;
},
};
// API Service Class
class AGCApiService {
constructor(baseUrl = CONFIG.API_BASE_URL) {
this.baseUrl = baseUrl;
this.isOnline = navigator.onLine;
this.cache = new Map();
this.cacheExpiry = new Map();
this.pendingRequests = new Map();
this.setupNetworkListeners();
}
setupNetworkListeners() {
window.addEventListener("online", () => {
this.isOnline = true;
console.log("AGC API: Network connection restored");
this.retryFailedRequests();
});
window.addEventListener("offline", () => {
this.isOnline = false;
console.log("AGC API: Network connection lost");
});
}
async retryFailedRequests() {
// Retry any pending requests when connection is restored
if (this._retryQueue && this._retryQueue.length > 0) {
console.log(`AGC API: Retrying ${this._retryQueue.length} failed requests`);
const requests = [...this._retryQueue];
this._retryQueue = [];
for (const req of requests) {
try {
const result = await this.request(req.endpoint, req.options);
req.resolve(result);
} catch (error) {
req.reject(error);
}
}
}
}
async request(endpoint, options = {}) {
const {
method = "GET",
data = null,
headers = {},
timeout = CONFIG.API_TIMEOUT,
retries = CONFIG.RETRY_ATTEMPTS,
cacheDuration = 60000, // Default cache of 1 minute
bypassCache = false,
} = options;
const url = `${this.baseUrl}${endpoint}`;
const cacheKey = this.getCacheKey(url, method, data);
// Check if there's already a pending request for this exact resource
if (this.pendingRequests.has(cacheKey)) {
console.log(`AGC API: Reusing pending request for ${method} ${url}`);
return this.pendingRequests.get(cacheKey);
}
// Return cached response if available and valid
if (
!bypassCache &&
method === "GET" &&
this.cache.has(cacheKey) &&
this.cacheExpiry.has(cacheKey) &&
this.cacheExpiry.get(cacheKey) > Date.now()
) {
console.log(`AGC API: Cache hit for ${method} ${url}`);
return JSON.parse(JSON.stringify(this.cache.get(cacheKey))); // Deep clone to prevent mutation
}
const config = {
method,
headers: {
"Content-Type": "application/json",
...headers,
},
};
// Only set AbortSignal when a timeout is provided
if (timeout) {
config.signal = AbortSignal.timeout(timeout);
}
if (data && method !== "GET") {
config.body = JSON.stringify(data);
}
// Create a promise that will be used for duplicate request coalescing
const requestPromise = new Promise(async (resolve, reject) => {
let lastError;
for (let attempt = 0; attempt <= retries; attempt++) {
try {
console.log(`AGC API: ${method} ${url} (attempt ${attempt + 1})`);
const response = await fetch(url, config);
if (!response.ok) {
const errorData = await response.json().catch(() => ({}));
throw new Error(
errorData.detail ||
`HTTP ${response.status}: ${response.statusText}`
);
}
const result = await response.json();
console.log(`AGC API: Success for ${method} ${url}`);
// Cache successful GET responses
if (method === "GET") {
this.cache.set(cacheKey, result);
this.cacheExpiry.set(cacheKey, Date.now() + cacheDuration);
}
resolve(result);
return;
} catch (error) {
lastError = error;
console.warn(
`AGC API: Attempt ${attempt + 1} failed for ${method} ${url}:`,
error.message
);
if (attempt < retries) {
const delay = CONFIG.RETRY_DELAY * Math.pow(2, attempt); // Exponential backoff
console.log(`AGC API: Waiting ${delay}ms before retry`);
await new Promise((r) => setTimeout(r, delay));
}
}
}
// If we got here, all attempts failed
console.error(
`AGC API: All ${retries + 1} attempts failed for ${method} ${url}:`,
lastError
);
// Queue for retry when online if currently offline
if (!this.isOnline) {
if (!this._retryQueue) this._retryQueue = [];
this._retryQueue.push({
endpoint,
options,
resolve,
reject
});
console.log(`AGC API: Queued request for retry when online: ${method} ${url}`);
} else {
reject(lastError);
}
});
// Store the pending request and clean it up when done
this.pendingRequests.set(cacheKey, requestPromise);
try {
const result = await requestPromise;
return result;
} finally {
this.pendingRequests.delete(cacheKey);
}
}
getCacheKey(url, method, data) {
if (method === "GET") {
return url;
}
return `${method}:${url}:${JSON.stringify(data)}`;
}
invalidateCache(endpoint) {
const prefixToInvalidate = `${this.baseUrl}${endpoint}`;
// Find and remove all matching cache entries
for (const key of this.cache.keys()) {
if (key.startsWith(prefixToInvalidate)) {
this.cache.delete(key);
this.cacheExpiry.delete(key);
}
}
}
// API Endpoints
async getRoot() {
return this.request("/");
}
async getDocuments(filters = {}, bypassCache = false) {
const params = new URLSearchParams();
if (filters.doc_type && filters.doc_type !== "All Types") {
params.append("doc_type", filters.doc_type);
}
if (filters.title_filter) {
params.append("title_filter", filters.title_filter);
}
const queryString = params.toString() ? `?${params.toString()}` : "";
return this.request(`/documents${queryString}`, {
bypassCache,
cacheDuration: 300000 // 5 minutes cache for documents list
});
}
async getDocument(id) {
return this.request(`/documents/${id}`, {
cacheDuration: 600000 // 10 minutes cache for document details
});
}
async searchDocuments(query, profileSearch = false) {
return this.request("/search", {
method: "POST",
data: { query, profile_search: profileSearch },
bypassCache: true // Always bypass cache for searches
});
}
async getDocumentTypes() {
return this.request("/document-types", {
cacheDuration: 3600000 // 1 hour cache for document types
});
}
async checkHealth() {
try {
await this.request("/", {
timeout: 5000, // Short timeout for health check
bypassCache: true
});
return "connected";
} catch (error) {
return "disconnected";
}
}
}
// State Management Class
class AGCStateManager {
constructor() {
this.version = "1.0"; // State version for compatibility checks
this.state = this.getInitialState();
this.listeners = [];
this.setupStorageListener();
this.setupThemeDetection();
}
getInitialState() {
const savedState = this.loadFromStorage();
// Deep merge of saved state with default state
return {
// App State
isInitializing: true,
activeTab: savedState.activeTab || "search",
isOnline: navigator.onLine,
apiStatus: "connecting",
// AI Chat State
aiChat: {
isTyping: false,
currentMessage: "",
messages: savedState.aiChat?.messages || [],
recentConcepts: savedState.aiChat?.recentConcepts || [],
suggestions: [],
isOpen: false,
error: null
},
// Search State
searchQuery: "",
searchResults: null,
searchHistory: savedState.searchHistory || [],
selectedDocuments: [],
documents: [],
documentTypes: ["All Types"],
selectedDocument: null,
bookmarks: savedState.bookmarks || [],
// Loading States
loading: {
search: false,
documents: false,
export: false,
},
// UI State
ui: {
viewMode: savedState.ui?.viewMode || "grid",
resultsPerPage: savedState.ui?.resultsPerPage || 20,
sidebar: {
collapsed: savedState.ui?.sidebar?.collapsed || false,
width: savedState.ui?.sidebar?.width || 280,
},
...savedState.ui,
},
// Data State
stats: {
totalDocuments: 0,
searchCount: savedState.stats?.searchCount || 0,
lastSearchTime: savedState.stats?.lastSearchTime || 0,
lastUpdated: null,
},
// Error Handling
error: {
show: false,
message: "",
type: "error",
},
// Tabs Configuration
tabs: [
{ id: "search", name: "Search", badge: null },
{ id: "browse", name: "Browse", badge: null },
{ id: "analytics", name: "Analytics", badge: null },
],
// User Preferences
preferences: {
theme: this.detectPreferredTheme(savedState),
fontSize: savedState.preferences?.fontSize || "medium",
language: savedState.preferences?.language || "en",
saveSearchHistory: savedState.preferences?.saveSearchHistory !== false, // Default to true
enableAIFeatures: savedState.preferences?.enableAIFeatures !== false, // Default to true
...savedState.preferences,
},
// Pagination
pagination: {
currentPage: 1,
totalPages: 1,
pageSize: savedState.pagination?.pageSize || 20,
},
// Filters
filters: {
doc_type: savedState.filters?.doc_type || "All Types",
title_filter: savedState.filters?.title_filter || "",
dateRange: savedState.filters?.dateRange || "",
sortBy: savedState.filters?.sortBy || "created_at",
sortDirection: savedState.filters?.sortDirection || "desc",
},
};
}
detectPreferredTheme(savedState) {
// First check saved preference
if (savedState.preferences?.theme) {
return savedState.preferences.theme;
}
// Then check system preference
if (window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches) {
return 'dark';
}
// Default to light
return 'light';
}
setupThemeDetection() {
// Listen for system theme changes
if (window.matchMedia) {
window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', e => {
// Only update if user hasn't explicitly set a preference
if (!this.state.preferences.hasExplicitTheme) {
this.setState({
preferences: {
...this.state.preferences,
theme: e.matches ? 'dark' : 'light'
}
});
}
});
}
}
setState(updates) {
const oldState = JSON.parse(JSON.stringify(this.state)); // Deep copy for comparison
// Deep merge updates with current state
this.state = this.deepMerge(this.state, updates);
// Save critical state to localStorage
this.saveToStorage();
// Notify listeners
this.listeners.forEach((listener) => {
try {
listener(this.state, oldState);
} catch (error) {
console.error("State listener error:", error);
}
});
}
// Deep merge objects helper
deepMerge(target, source) {
const output = Object.assign({}, target);
if (this.isObject(target) && this.isObject(source)) {
Object.keys(source).forEach(key => {
if (this.isObject(source[key])) {
if (!(key in target)) {
Object.assign(output, { [key]: source[key] });
} else {
output[key] = this.deepMerge(target[key], source[key]);
}
} else {
Object.assign(output, { [key]: source[key] });
}
});
}
return output;
}
isObject(item) {
return (item && typeof item === 'object' && !Array.isArray(item));
}
subscribe(listener) {
this.listeners.push(listener);
return () => {
this.listeners = this.listeners.filter((l) => l !== listener);
};
}
saveToStorage() {
try {
const persistentState = {
version: this.version,
timestamp: Date.now(),
activeTab: this.state.activeTab,
searchHistory: this.state.preferences.saveSearchHistory ? this.state.searchHistory.slice(-20) : [], // Save last 20 searches if enabled
aiChat: {
messages: this.state.aiChat.messages.slice(-100), // Save last 100 messages
recentConcepts: this.state.aiChat.recentConcepts
},
bookmarks: this.state.bookmarks,
stats: {
searchCount: this.state.stats.searchCount,
lastSearchTime: this.state.stats.lastSearchTime
},
preferences: this.state.preferences,
ui: {
viewMode: this.state.ui.viewMode,
resultsPerPage: this.state.ui.resultsPerPage,
sidebar: {
collapsed: this.state.ui.sidebar?.collapsed,
width: this.state.ui.sidebar?.width
}
},
pagination: {
pageSize: this.state.pagination.pageSize
},
filters: this.state.filters
};
// Compress the state to reduce storage size
const stateStr = JSON.stringify(persistentState);
localStorage.setItem(
CONFIG.STORAGE_PREFIX + "state",
stateStr
);
// Also keep a backup in session storage in case local storage fails
sessionStorage.setItem(
CONFIG.STORAGE_PREFIX + "state_backup",
stateStr
);
return true;
} catch (error) {
console.warn("Failed to save state to localStorage:", error);
return false;
}
}
loadFromStorage() {
try {
// Try getting from localStorage first
let saved = localStorage.getItem(CONFIG.STORAGE_PREFIX + "state");
// If not available, try from sessionStorage
if (!saved) {
saved = sessionStorage.getItem(CONFIG.STORAGE_PREFIX + "state_backup");
if (saved) {
console.log("Restored state from session storage backup");
}
}
if (!saved) {
return {};
}
const parsed = JSON.parse(saved);
// Check version compatibility for potential migrations
if (parsed.version !== this.version) {
console.log(`State version mismatch: saved=${parsed.version}, current=${this.version}`);
// Future: Migration logic could go here
}
return parsed;
} catch (error) {
console.warn("Failed to load state from storage:", error);
return {};
}
}
setupStorageListener() {
window.addEventListener("storage", (e) => {
if (e.key === CONFIG.STORAGE_PREFIX + "state") {
// Another tab updated the state
try {
const savedState = JSON.parse(e.newValue);
// Don't apply changes originating from this instance
if (savedState.timestamp !== this.state.timestamp) {
console.log("Syncing state from another tab");
this.setState(savedState);
}
} catch (error) {
console.warn("Failed to parse state from storage event:", error);
}
}
});
}
// Convenience methods for common state updates
setLoading(key, value) {
this.setState({
loading: { ...this.state.loading, [key]: value },
});
}
setError(message, type = "error") {
this.setState({
error: { show: true, message, type },
});
// Auto-clear error after 5 seconds
setTimeout(() => {
// Only clear if it's the same error (prevent clearing new errors)
if (this.state.error.message === message) {
this.setState({
error: { show: false, message: "", type: "error" },
});
}
}, 5000);
}
clearError() {
this.setState({
error: { show: false, message: "", type: "error" },
});
}
// Reset application state for logged out users or "forget me" feature
resetState(preservePreferences = true) {
const preferences = preservePreferences ? this.state.preferences : {};
localStorage.removeItem(CONFIG.STORAGE_PREFIX + "state");
sessionStorage.removeItem(CONFIG.STORAGE_PREFIX + "state_backup");
this.state = this.getInitialState();
if (preservePreferences) {
this.setState({ preferences });
}
return true;
}
}
// Main Alpine.js Application
function agcApp() {
return {
appState: null,
stateManager: null,
apiService: null,
// Initialize the application
async initializeApp() {
// Initialize services
this.apiService = new AGCApiService();
this.stateManager = new AGCStateManager();
this.appState = this.stateManager.state;
// Check API connection
const apiStatus = await this.apiService.checkHealth();
this.stateManager.setState({ apiStatus });
// Load initial data
await this.loadInitialData();
// Initialize charts
this.initializeCharts();
// Mark initialization as complete
this.stateManager.setState({ isInitializing: false });
},
// Load initial application data
async loadInitialData() {
try {
// Load document types (simulated for now)
const documentTypes = ["All Types", "LKK", "Case", "Legal"];
this.stateManager.setState({ documentTypes });
// Load initial documents
await this.loadDocuments(this.appState.filters);
// Set initial stats
this.stateManager.setState({
stats: {
...this.appState.stats,
totalDocuments: 156, // Simulated count until API is connected
lastUpdated: new Date().toISOString()
}
});
} catch (error) {
console.error("Failed to load initial data:", error);
this.stateManager.setState({
error: {
show: true,
message: "Failed to load initial data. Please refresh the page.",
type: "error"
}
});
}
},
// Load documents based on filters
async loadDocuments(filters = {}) {
this.stateManager.setLoading("documents", true);
try {
// For demo purposes, generating mock documents
// In a real implementation, this would fetch from the API
const mockDocuments = Array.from({ length: 20 }, (_, i) => ({
id: `doc-${i}`,
title: `Sample Document ${i + 1}`,
content: `This is a sample document with some content for demonstration purposes. It contains information relevant to the Attorney General's Chambers.`,
doc_type: ["LKK", "Case", "Legal"][Math.floor(Math.random() * 3)],
created_at: new Date(Date.now() - Math.random() * 10000000000).toISOString(),
source: "AGC Database"
}));
this.stateManager.setState({ documents: mockDocuments });
// Set pagination
this.stateManager.setState({
pagination: {
...this.appState.pagination,
currentPage: 1,
totalPages: 5 // Simulated for demo
}
});
return mockDocuments;
} catch (error) {
console.error("Failed to load documents:", error);
this.stateManager.setError("Failed to load documents. Please try again.");
return [];
} finally {
this.stateManager.setLoading("documents", false);
}
},
// Perform search with query
async performSearch(query) {
if (!query.trim()) return;
this.stateManager.setLoading("search", true);
try {
// For demo purposes, generating mock search results
// In a real implementation, this would fetch from the API
const mockDocuments = Array.from({ length: 5 }, (_, i) => ({
id: `result-${i}`,
title: `Result for "${query}" ${i + 1}`,
content: `This document contains information related to "${query}" and is highly relevant to your search query.`,
content_preview: `...information related to <span class="search-highlight">${query}</span> and is highly relevant...`,
doc_type: ["LKK", "Case", "Legal"][Math.floor(Math.random() * 3)],
created_at: new Date(Date.now() - Math.random() * 10000000000).toISOString(),
similarity: Math.random() * 0.5 + 0.5, // Random score between 0.5 and 1.0
source: "AGC Database"
}));
// Update search results
this.stateManager.setState({
searchResults: {
query: query,
enhanced_query: `Advanced search for documents containing "${query}" with relevant legal context`,
documents: mockDocuments,
answer: `Based on the available documents, "${query}" appears in several legal contexts. The most relevant documents are from recent cases where this topic was discussed in detail.`
}
});
// Add to search history if enabled
if (this.appState.preferences.saveSearchHistory) {
const history = [
{
id: Utils.generateId(),
query: query,
timestamp: new Date().toISOString(),
count: mockDocuments.length
},
...this.appState.searchHistory
].slice(0, 20); // Keep last 20 searches
this.stateManager.setState({
searchHistory: history,
stats: {
...this.appState.stats,
searchCount: this.appState.stats.searchCount + 1,
lastSearchTime: Date.now()
}
});
}
return mockDocuments;
} catch (error) {
console.error("Search failed:", error);
this.stateManager.setError("Search failed. Please try again.");
return [];
} finally {
this.stateManager.setLoading("search", false);
}
},
// View a document's details
viewDocument(doc) {
if (!doc) return;
this.stateManager.setState({ selectedDocument: doc });
// In a real implementation, this would open a modal or navigate to detail page
console.log("Viewing document:", doc.title);
alert(`Viewing document: ${doc.title}`);
},
// Open the full document
openDocument(doc) {
if (!doc) return;
// In a real implementation, this would open the document in a viewer
console.log("Opening document:", doc.title);
alert(`Opening document: ${doc.title} (ID: ${doc.id})`);
},
// Toggle document bookmark status
toggleBookmark(docId) {
if (!docId) return;
const bookmarks = [...this.appState.bookmarks];
const index = bookmarks.indexOf(docId);
if (index === -1) {
bookmarks.push(docId);
} else {
bookmarks.splice(index, 1);
}
this.stateManager.setState({ bookmarks });
},
// Check if document is bookmarked
isBookmarked(docId) {
return this.appState.bookmarks.includes(docId);
},
// Toggle document selection for bulk actions
toggleDocumentSelection(docId) {
if (!docId) return;
const selectedDocuments = [...this.appState.selectedDocuments];
const index = selectedDocuments.indexOf(docId);
if (index === -1) {
selectedDocuments.push(docId);
} else {
selectedDocuments.splice(index, 1);
}
this.stateManager.setState({ selectedDocuments });
},
// Check if document is selected
isDocumentSelected(docId) {
return this.appState.selectedDocuments.includes(docId);
},
// Clear all selected documents
clearSelection() {
this.stateManager.setState({ selectedDocuments: [] });
},
// Export selected documents or search results
exportResults() {
alert("Export functionality would be implemented here");
},
// Get document type CSS class
getDocTypeClass(type) {
switch(type) {
case 'LKK': return 'doc-type-lkk';
case 'Case': return 'doc-type-case';
case 'Legal': return 'doc-type-legal';
default: return 'doc-type-default';
}
},
// Format document type for display
formatDocumentType(type) {
return type || "DOC";
},
// Highlight search terms in text
highlightSearchTerms(text, query) {
if (!text || !query) return text;
// Simple implementation - in a real app, use a more robust approach
const escapedQuery = query.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
const regex = new RegExp(`(${escapedQuery})`, 'gi');
return text.replace(regex, '<span class="search-highlight">$1</span>');
},
// Switch between tabs
switchTab(tabId) {
if (this.appState.activeTab === tabId) return;
this.stateManager.setState({ activeTab: tabId });
},
// Set view mode (grid or list)
setViewMode(mode) {
if (mode !== 'grid' && mode !== 'list') return;
this.stateManager.setState({
preferences: {
...this.appState.preferences,
viewMode: mode
}
});
},
// Check if any filters are active
hasActiveFilters() {
return (
this.appState.filters.title_filter ||
this.appState.filters.doc_type !== 'All Types' ||
this.appState.filters.dateRange
);
},
// Clear all filters
clearAllFilters() {
this.stateManager.setState({
filters: {
doc_type: 'All Types',
title_filter: '',
dateRange: '',
sortBy: 'created_at',
sortDirection: 'desc'
}
});
this.loadDocuments(this.appState.filters);
},
// Get number of displayed documents
getDisplayedCount() {
return this.appState.documents.length;
},
// Get pagination pages for display
getPaginationPages() {
const totalPages = this.appState.pagination.totalPages;
const currentPage = this.appState.pagination.currentPage;
if (totalPages <= 5) {
return Array.from({ length: totalPages }, (_, i) => i + 1);
}
// Show current page and pages around it
if (currentPage <= 3) {
return [1, 2, 3, 4, 5];
} else if (currentPage >= totalPages - 2) {
return [totalPages - 4, totalPages - 3, totalPages - 2, totalPages - 1, totalPages];
} else {
return [currentPage - 2, currentPage - 1, currentPage, currentPage + 1, currentPage + 2];
}
},
// Go to specific page
goToPage(page) {
if (page < 1 || page > this.appState.pagination.totalPages) return;
this.stateManager.setState({
pagination: {
...this.appState.pagination,
currentPage: page
}
});
this.loadDocuments(this.appState.filters);
},
// Clear search history
clearSearchHistory() {
this.stateManager.setState({ searchHistory: [] });
},
// Debounced filter function for search input
debouncedFilter: Utils.debounce(function() {
this.loadDocuments(this.appState.filters);
}, 300),
// Ask AI a specific question
askAI(question) {
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
currentMessage: question,
isOpen: true
}
});
this.sendAIMessage();
},
// Send message to AI
sendAIMessage() {
this.sendMessage(this.appState.aiChat.currentMessage);
},
// Format AI message with markdown or HTML
formatAIMessage(message) {
// Simple implementation - in a real app, use a proper markdown parser
return message;
},
// Generate document summary
generateDocumentSummary() {
this.askAI("Summarize the documents in my current search results");
},
// Analyze legal trends
analyzeLegalTrends() {
this.askAI("What are the trends in the legal documents I'm viewing?");
},
// Clear AI chat
clearAIChat() {
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
messages: [],
error: null
}
});
},
// Initialize charts for analytics
initializeCharts() {
// Only initialize when the DOM elements exist and charts are needed
this.$watch('appState.activeTab', (tab) => {
if (tab === 'analytics') {
this.$nextTick(() => {
this.renderDocumentDistributionChart();
this.renderSearchActivityChart();
});
}
});
},
// Render document distribution chart
renderDocumentDistributionChart() {
const chartElement = this.$refs.documentDistributionChart;
if (!chartElement) return;
// Cleanup existing chart if it exists
if (this.documentDistributionChart) {
this.documentDistributionChart.destroy();
}
// Prepare data
const docTypes = ['LKK', 'Case', 'Legal', 'Other'];
const data = [42, 65, 29, 20]; // Mock data
// Create chart
this.documentDistributionChart = new Chart(chartElement, {
type: 'doughnut',
data: {
labels: docTypes,
datasets: [{
data: data,
backgroundColor: [
'#1E40AF', // Primary
'#059669', // Success
'#D97706', // Warning
'#6B7280' // Gray
],
borderColor: '#FFFFFF',
borderWidth: 2
}]
},
options: {
responsive: true,
maintainAspectRatio: false,
plugins: {
legend: {
display: false
},
tooltip: {
backgroundColor: 'rgba(255, 255, 255, 0.8)',
titleColor: '#1F2937',
bodyColor: '#1F2937',
borderColor: '#E5E7EB',
borderWidth: 1,
cornerRadius: 8,
boxPadding: 6,
usePointStyle: true,
callbacks: {
label: function(context) {
const label = context.label || '';
const value = context.raw || 0;
const total = context.dataset.data.reduce((acc, val) => acc + val, 0);
const percentage = Math.round((value / total) * 100);
return `${label}: ${value} (${percentage}%)`;
}
}
}
},
cutout: '60%',
animation: {
animateScale: true,
animateRotate: true
}
}
});
},
// Render search activity chart
renderSearchActivityChart() {
const chartElement = this.$refs.searchActivityChart;
if (!chartElement) return;
// Cleanup existing chart if it exists
if (this.searchActivityChart) {
this.searchActivityChart.destroy();
}
// Generate mock data for the last 7 days
const labels = Array.from({ length: 7 }, (_, i) => {
const date = new Date();
date.setDate(date.getDate() - (6 - i));
return date.toLocaleDateString('en-MY', { month: 'short', day: 'numeric' });
});
const searchData = [12, 19, 15, 23, 28, 25, 32]; // Mock data
const aiData = [5, 8, 10, 14, 16, 19, 23]; // Mock data
// Create chart
this.searchActivityChart = new Chart(chartElement, {
type: 'line',
data: {
labels: labels,
datasets: [
{
label: 'Daily Searches',
data: searchData,
borderColor: '#3B82F6',
backgroundColor: 'rgba(59, 130, 246, 0.1)',
borderWidth: 3,
tension: 0.3,
fill: true
},
{
label: 'AI Interactions',
data: aiData,
borderColor: '#10B981',
backgroundColor: 'rgba(16, 185, 129, 0.1)',
borderWidth: 3,
tension: 0.3,
fill: true
}
]
},
options: {
responsive: true,
maintainAspectRatio: false,
interaction: {
mode: 'index',
intersect: false
},
plugins: {
legend: {
display: false
},
tooltip: {
backgroundColor: 'rgba(255, 255, 255, 0.8)',
titleColor: '#1F2937',
bodyColor: '#1F2937',
borderColor: '#E5E7EB',
borderWidth: 1,
cornerRadius: 8,
boxPadding: 6
}
},
scales: {
y: {
beginAtZero: true,
grid: {
color: 'rgba(226, 232, 240, 0.5)'
}
},
x: {
grid: {
display: false
}
}
}
}
});
},
// AI Chat Methods
toggleAIChat() {
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
isOpen: !this.appState.aiChat.isOpen
}
});
},
async sendMessage(message) {
if (!message.trim()) return;
// Add user message
const messages = [...this.appState.aiChat.messages];
messages.push({
role: 'user',
content: message,
timestamp: new Date().toISOString()
});
// Update state with user message and typing status
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
messages,
currentMessage: '',
isTyping: true,
error: null
}
});
try {
// Simulate AI response (replace with actual API call)
await new Promise(resolve => setTimeout(resolve, 1000));
// Add AI response
const updatedMessages = [...messages];
updatedMessages.push({
role: 'assistant',
content: 'This is a simulated response. Replace with actual API integration.',
timestamp: new Date().toISOString()
});
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
messages: updatedMessages,
isTyping: false
}
});
} catch (error) {
console.error('AI Chat Error:', error);
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
isTyping: false,
error: 'Failed to get response from AI'
}
});
}
},
updateCurrentMessage(message) {
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
currentMessage: message
}
});
},
clearChat() {
this.stateManager.setState({
aiChat: {
...this.appState.aiChat,
messages: [],
error: null
}
});
},
// ... existing code ...
};
}
// Global error handler
window.addEventListener("error", (event) => {
console.error("Global error:", event.error);
});
window.addEventListener("unhandledrejection", (event) => {
console.error("Unhandled promise rejection:", event.reason);
});
// Performance monitoring
window.addEventListener("load", () => {
console.log("AGC Document Search: Page loaded");
if (window.performance) {
const loadTime =
window.performance.timing.loadEventEnd -
window.performance.timing.navigationStart;
console.log(`AGC Document Search: Load time: ${loadTime}ms`);
}
});
console.log(
"AGC Document Search: Scripts loaded, waiting for Alpine.js initialization..."
);
</script>
</body>
</html>