43c4ed7a6d
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>
261 lines
9.8 KiB
HTML
261 lines
9.8 KiB
HTML
<!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>
|