🎉 v1.0.0 — Vela Platform launch
Self-hosted P&L tracking app with component-level pricing. Offers: Atlas/Atlas+/Rif/Rif+ with granular cost breakdown. API + MCP + multi-user auth.
This commit is contained in:
91
backend/templates/base.html
Normal file
91
backend/templates/base.html
Normal file
@@ -0,0 +1,91 @@
|
||||
<!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>
|
||||
Reference in New Issue
Block a user