Add Homepage dashboard on Proxmox with Palantir theme and Admin UI.

Deploy gethomepage on pve CT 120, categorized services from Homarr, RSS feeds,
custom styling, and a browser-based admin UI on the NAS for adding sites.

Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
mo
2026-05-17 18:45:55 +02:00
parent 9f431ff97b
commit 43c4ed7a6d
27 changed files with 2851 additions and 0 deletions
+260
View File
@@ -0,0 +1,260 @@
<!DOCTYPE html>
<html lang="nl">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>Homepage Admin — EL-KADI OPS</title>
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link href="https://fonts.googleapis.com/css2?family=DM+Sans:wght@400;500;600;700&family=JetBrains+Mono:wght@400;500&display=swap" rel="stylesheet" />
<style>
:root {
--void: #02060c;
--surface: #0a1424;
--border: rgba(56, 132, 220, 0.2);
--blue: #3b82f6;
--cyan: #22d3ee;
--orange: #fb923c;
--text: #e8eef7;
--muted: #64748b;
--mono: 'JetBrains Mono', monospace;
--sans: 'DM Sans', system-ui, sans-serif;
}
* { box-sizing: border-box; margin: 0; padding: 0; }
body {
font-family: var(--sans);
background: var(--void);
color: var(--text);
min-height: 100vh;
padding: 1.5rem;
}
body::before {
content: '';
position: fixed; inset: 0;
background:
radial-gradient(ellipse 80% 50% at 10% -10%, rgba(59,130,246,.2), transparent 50%),
radial-gradient(ellipse 50% 40% at 90% 80%, rgba(251,146,60,.1), transparent 45%);
pointer-events: none; z-index: 0;
}
.wrap { max-width: 920px; margin: 0 auto; position: relative; z-index: 1; }
header {
display: flex; flex-wrap: wrap; align-items: center; justify-content: space-between;
gap: 1rem; margin-bottom: 1.5rem; padding-bottom: 1rem; border-bottom: 1px solid var(--border);
}
h1 { font-size: 1.1rem; font-weight: 600; letter-spacing: 0.12em; text-transform: uppercase; }
h1 span { color: var(--orange); font-family: var(--mono); font-size: 0.7rem; display: block; margin-top: 0.25rem; letter-spacing: 0.2em; }
.links a {
color: var(--cyan); text-decoration: none; font-size: 0.85rem; margin-left: 1rem;
}
.links a:hover { color: var(--orange); }
.card {
background: rgba(10, 22, 40, 0.92);
border: 1px solid var(--border);
border-radius: 12px;
padding: 1.25rem;
margin-bottom: 1.25rem;
backdrop-filter: blur(12px);
}
.card h2 {
font-family: var(--mono); font-size: 0.65rem; letter-spacing: 0.2em;
text-transform: uppercase; color: var(--cyan); margin-bottom: 1rem;
}
label { display: block; font-size: 0.75rem; color: var(--muted); margin-bottom: 0.35rem; }
input, select {
width: 100%; padding: 0.65rem 0.75rem; margin-bottom: 0.85rem;
background: rgba(5, 12, 24, 0.8); border: 1px solid var(--border); border-radius: 8px;
color: var(--text); font-family: var(--sans); font-size: 0.95rem;
}
input:focus, select:focus { outline: none; border-color: var(--orange); }
.grid { display: grid; grid-template-columns: 1fr 2fr 2fr auto; gap: 0.75rem; align-items: end; }
@media (max-width: 700px) { .grid { grid-template-columns: 1fr; } }
.btn {
padding: 0.65rem 1.1rem; border: none; border-radius: 8px; cursor: pointer;
font-family: var(--mono); font-size: 0.72rem; letter-spacing: 0.08em;
text-transform: uppercase; transition: opacity 0.2s, transform 0.15s;
}
.btn:hover { transform: translateY(-1px); }
.btn-primary { background: linear-gradient(135deg, var(--blue), #2563eb); color: #fff; }
.btn-accent { background: linear-gradient(135deg, var(--orange), #ea580c); color: #fff; width: 100%; margin-top: 0.5rem; }
.btn-ghost { background: transparent; border: 1px solid var(--border); color: var(--muted); padding: 0.4rem 0.6rem; }
.btn-ghost:hover { border-color: #f87171; color: #f87171; }
.btn:disabled { opacity: 0.5; cursor: not-allowed; transform: none; }
table { width: 100%; border-collapse: collapse; font-size: 0.88rem; }
th {
text-align: left; font-family: var(--mono); font-size: 0.58rem;
letter-spacing: 0.15em; text-transform: uppercase; color: var(--muted);
padding: 0.5rem 0.4rem; border-bottom: 1px solid var(--border);
}
td { padding: 0.65rem 0.4rem; border-bottom: 1px solid rgba(255,255,255,0.04); vertical-align: middle; }
td a { color: var(--cyan); text-decoration: none; word-break: break-all; }
td a:hover { color: var(--orange); }
.tag {
display: inline-block; font-family: var(--mono); font-size: 0.58rem;
padding: 0.15rem 0.45rem; border-radius: 4px;
background: rgba(59,130,246,0.15); color: var(--cyan); margin-right: 0.35rem;
}
.empty { color: var(--muted); font-style: italic; padding: 1rem 0; }
#log {
font-family: var(--mono); font-size: 0.7rem; line-height: 1.5;
background: rgba(0,0,0,0.35); border-radius: 8px; padding: 0.75rem;
max-height: 200px; overflow: auto; color: var(--muted); white-space: pre-wrap;
margin-top: 0.75rem; display: none;
}
#log.show { display: block; }
#log.ok { color: #4ade80; }
#log.err { color: #f87171; }
.toast {
position: fixed; bottom: 1.5rem; right: 1.5rem; padding: 0.75rem 1rem;
background: var(--surface); border: 1px solid var(--border); border-radius: 8px;
font-size: 0.85rem; opacity: 0; transition: opacity 0.3s; z-index: 10;
}
.toast.show { opacity: 1; }
</style>
</head>
<body>
<div class="wrap">
<header>
<h1>Homepage Admin<span>Websites toevoegen</span></h1>
<div class="links">
<a href="http://192.168.1.192:3000" target="_blank" rel="noreferrer">→ Open dashboard</a>
</div>
</header>
<section class="card">
<h2>Nieuwe website</h2>
<form id="add-form">
<div class="grid">
<div>
<label for="group">Groep / tab</label>
<select id="group" required></select>
</div>
<div>
<label for="name">Naam</label>
<input id="name" type="text" placeholder="bijv. Jellyfin" required />
</div>
<div>
<label for="url">URL</label>
<input id="url" type="url" placeholder="http://192.168.1.10:8096" required />
</div>
<div>
<button type="submit" class="btn btn-primary">Toevoegen</button>
</div>
</div>
</form>
</section>
<section class="card">
<h2>Jouw extra sites</h2>
<div id="list-wrap">
<p class="empty">Laden…</p>
</div>
</section>
<section class="card">
<h2>Dashboard bijwerken</h2>
<p style="font-size:0.85rem;color:var(--muted);margin-bottom:0.75rem;">
Na toevoegen of verwijderen: klik hieronder. Dit genereert de config en deployt naar Proxmox (±1 min).
</p>
<button type="button" id="apply-btn" class="btn btn-accent">Toepassen op Homepage</button>
<pre id="log"></pre>
</section>
</div>
<div id="toast" class="toast"></div>
<script>
const API = '';
const $ = (s) => document.querySelector(s);
function toast(msg) {
const t = $('#toast');
t.textContent = msg;
t.classList.add('show');
setTimeout(() => t.classList.remove('show'), 2800);
}
async function loadGroups() {
const groups = await fetch(API + '/api/groups').then((r) => r.json());
$('#group').innerHTML = groups
.map((g) => `<option value="${g.name}">${g.name} (tab ${g.tab})</option>`)
.join('');
}
async function loadSites() {
const sites = await fetch(API + '/api/sites').then((r) => r.json());
const wrap = $('#list-wrap');
if (!sites.length) {
wrap.innerHTML = '<p class="empty">Nog geen extra sites — voeg hierboven toe.</p>';
return;
}
wrap.innerHTML = `<table>
<thead><tr><th>Groep</th><th>Naam</th><th>URL</th><th></th></tr></thead>
<tbody>${sites.map((s) => `
<tr>
<td><span class="tag">${s.tab}</span>${esc(s.group)}</td>
<td>${esc(s.name)}</td>
<td><a href="${esc(s.url)}" target="_blank" rel="noreferrer">${esc(s.url)}</a></td>
<td><button type="button" class="btn btn-ghost" data-id="${s.id}">Verwijder</button></td>
</tr>`).join('')}</tbody>
</table>`;
wrap.querySelectorAll('[data-id]').forEach((btn) => {
btn.onclick = async () => {
if (!confirm('Verwijderen?')) return;
await fetch(API + '/api/sites/' + btn.dataset.id, { method: 'DELETE' });
toast('Verwijderd — klik nog op Toepassen');
loadSites();
};
});
}
function esc(s) {
const d = document.createElement('div');
d.textContent = s || '';
return d.innerHTML;
}
$('#add-form').onsubmit = async (e) => {
e.preventDefault();
const body = {
group: $('#group').value,
name: $('#name').value.trim(),
url: $('#url').value.trim(),
};
const r = await fetch(API + '/api/sites', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(body),
});
if (!r.ok) {
const err = await r.json().catch(() => ({}));
toast(err.detail || 'Fout bij toevoegen');
return;
}
$('#name').value = '';
$('#url').value = '';
toast('Toegevoegd — klik op Toepassen');
loadSites();
};
$('#apply-btn').onclick = async () => {
const btn = $('#apply-btn');
const log = $('#log');
btn.disabled = true;
log.className = 'show';
log.textContent = 'Bezig…\n';
await fetch(API + '/api/apply', { method: 'POST' });
const poll = setInterval(async () => {
const st = await fetch(API + '/api/apply/status').then((r) => r.json());
log.textContent = st.log || '…';
if (!st.running) {
clearInterval(poll);
btn.disabled = false;
log.classList.add(st.ok ? 'ok' : 'err');
toast(st.ok ? 'Homepage bijgewerkt!' : 'Fout — zie log');
}
}, 1500);
};
loadGroups().then(loadSites);
</script>
</body>
</html>