Self-hosted P&L tracking app with component-level pricing. Offers: Atlas/Atlas+/Rif/Rif+ with granular cost breakdown. API + MCP + multi-user auth.
92 lines
3.9 KiB
HTML
92 lines
3.9 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="en" data-bs-theme="dark">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>{% block title %}Vela Platform{% endblock %}</title>
|
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" rel="stylesheet">
|
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.3/font/bootstrap-icons.css" rel="stylesheet">
|
|
<script src="https://cdn.jsdelivr.net/npm/chart.js@4.4.0/dist/chart.umd.min.js"></script>
|
|
<style>
|
|
body { padding-top: 70px; }
|
|
.navbar-brand { font-weight: 700; letter-spacing: -0.5px; }
|
|
.card-stat { border-left: 4px solid var(--bs-primary); }
|
|
.card-stat.green { border-left-color: var(--bs-success); }
|
|
.card-stat.red { border-left-color: var(--bs-danger); }
|
|
.card-stat.yellow { border-left-color: var(--bs-warning); }
|
|
.component-row td { vertical-align: middle; }
|
|
.component-row input { min-width: 60px; }
|
|
.total-label { font-weight: 700; font-size: 1.05rem; }
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<nav class="navbar navbar-expand-lg bg-body-tertiary fixed-top border-bottom">
|
|
<div class="container">
|
|
<a class="navbar-brand" href="/">
|
|
🏔️ Vela Platform
|
|
</a>
|
|
{% if current_user %}
|
|
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav">
|
|
<span class="navbar-toggler-icon"></span>
|
|
</button>
|
|
<div class="collapse navbar-collapse" id="navbarNav">
|
|
<ul class="navbar-nav me-auto">
|
|
<li class="nav-item">
|
|
<a class="nav-link" href="/"><i class="bi bi-speedometer2"></i> Dashboard</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" href="/transactions"><i class="bi bi-list-ul"></i> Transactions</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" href="/services"><i class="bi bi-gear"></i> Offers</a>
|
|
</li>
|
|
</ul>
|
|
<span class="navbar-text me-3">
|
|
<i class="bi bi-person-circle"></i> {{ current_user.display_name }}
|
|
<span class="badge bg-secondary ms-1">{{ current_user.role }}</span>
|
|
</span>
|
|
<a href="/logout" class="btn btn-outline-secondary btn-sm">Logout</a>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
</nav>
|
|
|
|
<main class="container">
|
|
{% block content %}{% endblock %}
|
|
</main>
|
|
|
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/js/bootstrap.bundle.min.js"></script>
|
|
<script>
|
|
const TOKEN_KEY = 'vela_token';
|
|
function setToken(token) { localStorage.setItem(TOKEN_KEY, token); }
|
|
function getToken() { return localStorage.getItem(TOKEN_KEY); }
|
|
function clearToken() { localStorage.removeItem(TOKEN_KEY); }
|
|
|
|
async function apiFetch(url, options = {}) {
|
|
const token = getToken();
|
|
const headers = { ...options.headers };
|
|
if (token && !(options.body instanceof FormData)) {
|
|
headers['Authorization'] = `Bearer ${token}`;
|
|
headers['Content-Type'] = 'application/json';
|
|
} else if (token) {
|
|
headers['Authorization'] = `Bearer ${token}`;
|
|
}
|
|
const resp = await fetch(url, { ...options, headers });
|
|
if (resp.status === 401) {
|
|
clearToken();
|
|
window.location.href = '/login';
|
|
throw new Error('Unauthorized');
|
|
}
|
|
return resp;
|
|
}
|
|
|
|
function escapeHtml(str) {
|
|
const div = document.createElement('div');
|
|
div.textContent = str || '';
|
|
return div.innerHTML;
|
|
}
|
|
</script>
|
|
{% block scripts %}{% endblock %}
|
|
</body>
|
|
</html>
|