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:
@@ -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>
|
||||
Reference in New Issue
Block a user