This commit is contained in:
HotSwapp
2025-08-09 16:37:57 -05:00
parent 5f74243c8c
commit c2f3c4411d
35 changed files with 9209 additions and 4633 deletions

View File

@@ -1,174 +1,184 @@
{% extends "base.html" %}
{% block title %}Dashboard - {{ super() }}{% endblock %}
{% block title %}Dashboard - Delphi Database{% endblock %}
{% block content %}
<div class="row">
<div class="col-12">
<div class="d-flex justify-content-between align-items-center mb-4">
<h1><i class="bi bi-speedometer2"></i> Dashboard</h1>
<div>
<button class="btn btn-outline-secondary btn-sm" onclick="showShortcuts()">
<i class="bi bi-keyboard"></i> Shortcuts (F1)
</button>
<div class="space-y-6">
<!-- Page Header -->
<div class="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4">
<div class="flex items-center gap-3">
<div class="flex items-center justify-center w-10 h-10 bg-primary-100 dark:bg-primary-900/30 text-primary-600 dark:text-primary-400 rounded-xl">
<i class="fa-solid fa-gauge-high text-lg"></i>
</div>
<h1 class="text-2xl font-bold text-neutral-900 dark:text-neutral-100">Dashboard</h1>
</div>
<div class="flex items-center gap-3">
<button onclick="openShortcutsModal()" class="bg-neutral-100 dark:bg-neutral-700 text-neutral-700 dark:text-neutral-300 hover:bg-neutral-200 dark:hover:bg-neutral-600 px-3 py-1.5 rounded-lg text-xs font-medium transition-colors duration-200 flex items-center gap-2">
<i class="fa-solid fa-keyboard"></i>
<span>Shortcuts (F1)</span>
</button>
</div>
</div>
</div>
<!-- Quick Stats Cards -->
<div class="row mb-4">
<div class="col-md-3">
<div class="card bg-primary text-white">
<div class="card-body">
<div class="d-flex align-items-center">
<div class="me-3">
<i class="bi bi-people fs-1"></i>
</div>
<div>
<h5 class="card-title">Customers</h5>
<h2 class="mb-0" id="customer-count">-</h2>
</div>
<!-- Quick Stats Cards -->
<div class="grid grid-cols-1 md:grid-cols-4 gap-4">
<div class="bg-primary-600 text-white rounded-xl shadow-soft p-4">
<div class="flex items-center gap-3">
<div class="flex items-center justify-center w-10 h-10 bg-primary-700 rounded-lg">
<i class="fa-solid fa-users text-xl"></i>
</div>
<div>
<h5 class="text-sm font-medium mb-1">Customers</h5>
<h2 class="text-2xl font-bold mb-0" id="customer-count">-</h2>
</div>
<a href="/customers" class="text-white-50 small">
View all <i class="bi bi-arrow-right"></i>
</a>
</div>
<a href="/customers" class="text-primary-200 hover:text-white text-xs font-medium flex items-center gap-1 mt-2 transition-colors">
View all
<i class="fa-solid fa-arrow-right"></i>
</a>
</div>
<div class="bg-success-600 text-white rounded-xl shadow-soft p-4">
<div class="flex items-center gap-3">
<div class="flex items-center justify-center w-10 h-10 bg-success-700 rounded-lg">
<i class="fa-solid fa-folder text-xl"></i>
</div>
<div>
<h5 class="text-sm font-medium mb-1">Active Files</h5>
<h2 class="text-2xl font-bold mb-0" id="file-count">-</h2>
</div>
</div>
<a href="/files" class="text-success-200 hover:text-white text-xs font-medium flex items-center gap-1 mt-2 transition-colors">
View all
<i class="fa-solid fa-arrow-right"></i>
</a>
</div>
<div class="bg-info-600 text-white rounded-xl shadow-soft p-4">
<div class="flex items-center gap-3">
<div class="flex items-center justify-center w-10 h-10 bg-info-700 rounded-lg">
<i class="fa-solid fa-receipt text-xl"></i>
</div>
<div>
<h5 class="text-sm font-medium mb-1">Transactions</h5>
<h2 class="text-2xl font-bold mb-0" id="transaction-count">-</h2>
</div>
</div>
<a href="/financial" class="text-info-200 hover:text-white text-xs font-medium flex items-center gap-1 mt-2 transition-colors">
View ledger
<i class="fa-solid fa-arrow-right"></i>
</a>
</div>
<div class="bg-warning-600 text-white rounded-xl shadow-soft p-4">
<div class="flex items-center gap-3">
<div class="flex items-center justify-center w-10 h-10 bg-warning-700 rounded-lg">
<i class="fa-regular fa-file-lines text-xl"></i>
</div>
<div>
<h5 class="text-sm font-medium mb-1">Documents</h5>
<h2 class="text-2xl font-bold mb-0" id="document-count">-</h2>
</div>
</div>
<a href="/documents" class="text-warning-200 hover:text-white text-xs font-medium flex items-center gap-1 mt-2 transition-colors">
View all
<i class="fa-solid fa-arrow-right"></i>
</a>
</div>
</div>
<div class="col-md-3">
<div class="card bg-success text-white">
<div class="card-body">
<div class="d-flex align-items-center">
<div class="me-3">
<i class="bi bi-folder fs-1"></i>
</div>
<div>
<h5 class="card-title">Active Files</h5>
<h2 class="mb-0" id="file-count">-</h2>
</div>
</div>
<a href="/files" class="text-white-50 small">
View all <i class="bi bi-arrow-right"></i>
</a>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card bg-info text-white">
<div class="card-body">
<div class="d-flex align-items-center">
<div class="me-3">
<i class="bi bi-receipt fs-1"></i>
</div>
<div>
<h5 class="card-title">Transactions</h5>
<h2 class="mb-0" id="transaction-count">-</h2>
</div>
</div>
<a href="/financial" class="text-white-50 small">
View ledger <i class="bi bi-arrow-right"></i>
</a>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card bg-warning text-dark">
<div class="card-body">
<div class="d-flex align-items-center">
<div class="me-3">
<i class="bi bi-file-text fs-1"></i>
</div>
<div>
<h5 class="card-title">Documents</h5>
<h2 class="mb-0" id="document-count">-</h2>
</div>
</div>
<a href="/documents" class="text-dark-50 small">
View all <i class="bi bi-arrow-right"></i>
</a>
</div>
</div>
</div>
</div>
<!-- Quick Actions -->
<div class="row mb-4">
<div class="col-md-8">
<div class="card">
<div class="card-header">
<h5><i class="bi bi-lightning"></i> Quick Actions</h5>
<!-- Quick Actions and Recent Activity -->
<div class="grid grid-cols-1 md:grid-cols-3 gap-4">
<div class="md:col-span-2 bg-white dark:bg-neutral-800 border border-neutral-200 dark:border-neutral-700 rounded-xl shadow-soft">
<div class="px-6 py-4 border-b border-neutral-200 dark:border-neutral-700">
<h5 class="text-lg font-semibold text-neutral-900 dark:text-neutral-100 flex items-center gap-2">
<i class="fa-solid fa-bolt"></i>
<span>Quick Actions</span>
</h5>
</div>
<div class="card-body">
<div class="row">
<div class="col-md-6 mb-3">
<div class="d-grid gap-2">
<button class="btn btn-outline-primary btn-lg" onclick="newCustomer()">
<i class="bi bi-person-plus"></i> New Customer
<small class="d-block text-muted">Ctrl+Shift+C</small>
</button>
<button class="btn btn-outline-success btn-lg" onclick="newFile()">
<i class="bi bi-folder-plus"></i> New File
<small class="d-block text-muted">Ctrl+Shift+F</small>
</button>
</div>
<div class="p-6">
<div class="grid grid-cols-1 sm:grid-cols-2 gap-4">
<div class="space-y-3">
<button onclick="newCustomer()" class="w-full flex flex-col items-center justify-center p-4 bg-neutral-50 dark:bg-neutral-900/50 hover:bg-neutral-100 dark:hover:bg-neutral-900 rounded-lg border border-neutral-200 dark:border-neutral-700 transition-colors duration-200">
<i class="fa-solid fa-user-plus text-2xl text-primary-600 mb-1"></i>
<span class="font-medium">New Customer</span>
<kbd class="text-xs text-neutral-500 dark:text-neutral-400 mt-1">Ctrl+Shift+C</kbd>
</button>
<button onclick="newFile()" class="w-full flex flex-col items-center justify-center p-4 bg-neutral-50 dark:bg-neutral-900/50 hover:bg-neutral-100 dark:hover:bg-neutral-900 rounded-lg border border-neutral-200 dark:border-neutral-700 transition-colors duration-200">
<i class="fa-solid fa-folder-plus text-2xl text-success-600 mb-1"></i>
<span class="font-medium">New File</span>
<kbd class="text-xs text-neutral-500 dark:text-neutral-400 mt-1">Ctrl+Shift+F</kbd>
</button>
</div>
<div class="col-md-6 mb-3">
<div class="d-grid gap-2">
<button class="btn btn-outline-info btn-lg" onclick="newTransaction()">
<i class="bi bi-plus-circle"></i> New Transaction
<small class="d-block text-muted">Ctrl+Shift+T</small>
</button>
<button class="btn btn-outline-warning btn-lg" onclick="globalSearch()">
<i class="bi bi-search"></i> Global Search
<small class="d-block text-muted">Ctrl+F</small>
</button>
</div>
<div class="space-y-3">
<button onclick="newTransaction()" class="w-full flex flex-col items-center justify-center p-4 bg-neutral-50 dark:bg-neutral-900/50 hover:bg-neutral-100 dark:hover:bg-neutral-900 rounded-lg border border-neutral-200 dark:border-neutral-700 transition-colors duration-200">
<i class="fa-solid fa-circle-plus text-2xl text-info-600 mb-1"></i>
<span class="font-medium">New Transaction</span>
<kbd class="text-xs text-neutral-500 dark:text-neutral-400 mt-1">Ctrl+Shift+T</kbd>
</button>
<button onclick="globalSearch()" class="w-full flex flex-col items-center justify-center p-4 bg-neutral-50 dark:bg-neutral-900/50 hover:bg-neutral-100 dark:hover:bg-neutral-900 rounded-lg border border-neutral-200 dark:border-neutral-700 transition-colors duration-200">
<i class="fa-solid fa-magnifying-glass text-2xl text-warning-600 mb-1"></i>
<span class="font-medium">Global Search</span>
<kbd class="text-xs text-neutral-500 dark:text-neutral-400 mt-1">Ctrl+F</kbd>
</button>
</div>
</div>
</div>
</div>
</div>
<div class="col-md-4">
<div class="card">
<div class="card-header">
<h5><i class="bi bi-clock-history"></i> Recent Activity</h5>
<div class="bg-white dark:bg-neutral-800 border border-neutral-200 dark:border-neutral-700 rounded-xl shadow-soft">
<div class="px-6 py-4 border-b border-neutral-200 dark:border-neutral-700">
<h5 class="text-lg font-semibold text-neutral-900 dark:text-neutral-100 flex items-center gap-2">
<i class="fa-solid fa-clock-rotate-left"></i>
<span>Recent Activity</span>
</h5>
</div>
<div class="card-body">
<div id="recent-activity">
<p class="text-muted text-center">
<i class="bi bi-hourglass-split"></i><br>
Loading recent activity...
<div class="p-6" id="recent-activity">
<div class="flex flex-col items-center justify-center py-4 text-neutral-500 dark:text-neutral-400">
<i class="fa-solid fa-hourglass-half text-2xl mb-2"></i>
<p>Loading recent activity...</p>
</div>
</div>
</div>
</div>
<!-- System Status -->
<div class="bg-white dark:bg-neutral-800 border border-neutral-200 dark:border-neutral-700 rounded-xl shadow-soft">
<div class="px-6 py-4 border-b border-neutral-200 dark:border-neutral-700">
<h5 class="text-lg font-semibold text-neutral-900 dark:text-neutral-100 flex items-center gap-2">
<i class="fa-solid fa-circle-info"></i>
<span>System Information</span>
</h5>
</div>
<div class="p-6">
<div class="grid grid-cols-1 md:grid-cols-2 gap-6">
<div class="space-y-3 text-sm">
<p class="flex justify-between">
<strong>System:</strong>
<span>Delphi Consulting Group Database System</span>
</p>
<p class="flex justify-between">
<strong>Version:</strong>
<span>1.0.0</span>
</p>
<p class="flex justify-between">
<strong>Database:</strong>
<span>SQLite</span>
</p>
</div>
</div>
</div>
</div>
</div>
<!-- System Status -->
<div class="row">
<div class="col-12">
<div class="card">
<div class="card-header">
<h5><i class="bi bi-info-circle"></i> System Information</h5>
</div>
<div class="card-body">
<div class="row">
<div class="col-md-6">
<p><strong>System:</strong> Delphi Consulting Group Database System</p>
<p><strong>Version:</strong> 1.0.0</p>
<p><strong>Database:</strong> SQLite</p>
</div>
<div class="col-md-6">
<p><strong>Last Backup:</strong> <span id="last-backup">Not available</span></p>
<p><strong>Database Size:</strong> <span id="db-size">-</span></p>
<p><strong>Status:</strong> <span id="system-status" class="badge bg-success">Healthy</span></p>
</div>
<div class="space-y-3 text-sm">
<p class="flex justify-between">
<strong>Last Backup:</strong>
<span id="last-backup">Not available</span>
</p>
<p class="flex justify-between">
<strong>Database Size:</strong>
<span id="db-size">-</span>
</p>
<p class="flex justify-between">
<strong>Status:</strong>
<span id="system-status" class="px-2 py-1 bg-success-600 text-white text-xs font-medium rounded-full">Healthy</span>
</p>
</div>
</div>
</div>
@@ -178,55 +188,49 @@
{% block extra_scripts %}
<script>
// Load dashboard data
async function loadDashboardData() {
try {
// This would typically be authenticated API calls
const response = await fetch('/api/admin/stats', {
headers: {
'Authorization': 'Bearer ' + localStorage.getItem('token')
}
});
if (response.ok) {
const data = await response.json();
document.getElementById('customer-count').textContent = data.total_customers || '0';
document.getElementById('file-count').textContent = data.total_files || '0';
document.getElementById('transaction-count').textContent = data.total_transactions || '0';
document.getElementById('document-count').textContent = data.total_qdros || '0';
document.getElementById('db-size').textContent = data.database_size || '-';
document.getElementById('last-backup').textContent = data.last_backup || 'Not available';
// Load dashboard data
async function loadDashboardData() {
try {
const response = await fetch('/api/admin/stats', {
headers: {
'Authorization': `Bearer ${localStorage.getItem('auth_token')}`
}
} catch (error) {
console.error('Error loading dashboard data:', error);
});
if (response.ok) {
const data = await response.json();
document.getElementById('customer-count').textContent = data.total_customers || '0';
document.getElementById('file-count').textContent = data.total_files || '0';
document.getElementById('transaction-count').textContent = data.total_transactions || '0';
document.getElementById('document-count').textContent = data.total_qdros || '0';
document.getElementById('db-size').textContent = data.database_size || '-';
document.getElementById('last-backup').textContent = data.last_backup || 'Not available';
}
} catch (error) {
console.error('Error loading dashboard data:', error);
}
// Quick action functions
function newCustomer() {
window.location.href = '/customers/new';
}
function newFile() {
window.location.href = '/files/new';
}
function newTransaction() {
window.location.href = '/financial/new';
}
function globalSearch() {
window.location.href = '/search';
}
function showShortcuts() {
const modal = new bootstrap.Modal(document.getElementById('shortcutsModal'));
modal.show();
}
// Load data on page load
document.addEventListener('DOMContentLoaded', function() {
// loadDashboardData(); // Uncomment when authentication is implemented
});
}
// Quick action functions
function newCustomer() {
window.location.href = '/customers';
}
function newFile() {
window.location.href = '/files';
}
function newTransaction() {
window.location.href = '/financial';
}
function globalSearch() {
window.location.href = '/search';
}
// Load data on page load
document.addEventListener('DOMContentLoaded', function() {
loadDashboardData(); // Uncomment when authentication is implemented
});
</script>
{% endblock %}