banyaro/backend/static/js/worlds.js
rene 1fdba57365 Feature: UX-Fixes — Zahnrad weg, POI-Kombi-Typen, exp-fab-Position, Welten-Config in DB (SW by-v653)
- worlds-settings Zahnrad komplett entfernt (war auf Mobile sichtbar, auf Desktop schon hidden)
- exp-fab: bottom jetzt calc(--nav-bottom-height + --safe-bottom + --space-2) — kein Overlap mit worlds-back auf iPhone
- Karte POI: neue Typen bank, bank_kotbeutel, bank_kotbeutel_abfall, kotbeutel_abfall (Backend + Frontend)
- Welten-Chip-Config: GET/PUT /profile/world-config, Spalte users.world_config TEXT (Migration), Sync bei Init + Speichern
2026-05-03 19:50:04 +02:00

1093 lines
51 KiB
JavaScript

/* ============================================================
BAN YARO — Drei Welten Navigation
JETZT | HUND | WELT — horizontales Swipe-System
============================================================ */
window.Worlds = (() => {
let _state = null;
let _cur = 1; // 0=JETZT 1=HUND 2=WELT
let _visible = false;
let _map = null;
let _weltInited = false;
let _lastUserId = undefined;
let _dogs = []; // gecachte Hundesliste
let _dogIdx = 0; // aktuell angezeigter Hund
// Touch-Tracking
const _t = { x:0, y:0, active:false, vert:null, moved:0 };
// ── OFFLINE-CACHE (localStorage) ────────────────────────────
// Letzten bekannten Zustand speichern → sofort zeigen, dann updaten
function _wSave(key, data) {
try { localStorage.setItem('w3_' + key, JSON.stringify({ ts: Date.now(), data })); } catch {}
}
function _wLoad(key) {
try {
const p = JSON.parse(localStorage.getItem('w3_' + key) || 'null');
return p ? { data: p.data, ageMin: Math.round((Date.now() - p.ts) / 60000) } : null;
} catch { return null; }
}
// API-Aufruf mit Cache-Fallback: sofort Cache, dann Netz
async function _cachedGet(cacheKey, path) {
const cached = _wLoad(cacheKey);
let fresh = null;
try {
fresh = await API.get(path);
_wSave(cacheKey, fresh);
} catch {}
return { data: fresh ?? cached?.data ?? null, fromCache: !fresh, ageMin: fresh ? 0 : (cached?.ageMin ?? null) };
}
// ── PUBLIC ──────────────────────────────────────────────────
async function init(appState) {
_state = appState;
_cur = 1; // immer HUND als Start
_setupSwipe();
_setupButtons();
_goTo(_cur, false);
show();
// Config aus DB laden (async, dann neu rendern wenn nötig)
await _loadConfigFromServer();
// Welten parallel rendern
_renderJetzt();
_renderHund();
}
function show(worldIdx) {
const ov = document.getElementById('worlds-overlay');
if (!ov) return;
ov.style.display = 'block';
requestAnimationFrame(() => ov.classList.add('worlds-visible'));
_visible = true;
document.getElementById('app-header')?.classList.add('worlds-hidden');
document.getElementById('bottom-nav')?.classList.add('worlds-hidden');
document.getElementById('worlds-back')?.classList.remove('worlds-back-visible');
if (worldIdx != null) _goTo(worldIdx, false);
if (_cur === 2 && !_weltInited) { _weltInited = true; _renderWelt(); }
// Nach Login/Logout neu rendern
const currentUserId = _state?.user?.id ?? null;
if (currentUserId !== _lastUserId) {
_lastUserId = currentUserId;
_renderJetzt();
_renderHund();
}
}
function hide() {
const ov = document.getElementById('worlds-overlay');
if (!ov) return;
ov.classList.remove('worlds-visible');
ov.style.display = 'none';
_visible = false;
document.getElementById('worlds-back')?.classList.add('worlds-back-visible');
}
function navigateTo(pageId) {
hide();
if (window.App?.navigate) window.App.navigate(pageId);
}
// ── SWIPE ────────────────────────────────────────────────────
function _setupSwipe() {
const track = document.getElementById('worlds-track');
if (!track) return;
track.addEventListener('touchstart', e => {
_t.x = e.touches[0].clientX;
_t.y = e.touches[0].clientY;
_t.active = true; _t.vert = null; _t.moved = 0;
track.style.transition = 'none';
}, { passive: true });
track.addEventListener('touchmove', e => {
if (!_t.active) return;
const dx = e.touches[0].clientX - _t.x;
const dy = e.touches[0].clientY - _t.y;
if (_t.vert === null) _t.vert = Math.abs(dy) > Math.abs(dx) + 4;
if (_t.vert) return;
e.preventDefault();
// Nicht über erste/letzte Seite hinausziehen
const cdx = _cur === 0 ? Math.min(0, dx) : _cur === 2 ? Math.max(0, dx) : dx;
_t.moved = cdx;
const base = -_cur * (100 / 3);
track.style.transform = `translateX(calc(${base}% + ${cdx}px))`;
}, { passive: false });
track.addEventListener('touchend', () => {
if (!_t.active || _t.vert) { _t.active = false; return; }
_t.active = false;
let next = _cur;
if (_t.moved < -55 && _cur < 2) next = _cur + 1;
else if (_t.moved > 55 && _cur > 0) next = _cur - 1;
_goTo(next, true);
if (next === 2 && !_weltInited) { _weltInited = true; _renderWelt(); }
});
}
function _goTo(idx, animated) {
_cur = Math.max(0, Math.min(2, idx));
const track = document.getElementById('worlds-track');
if (!track) return;
track.style.transition = animated
? 'transform 0.32s cubic-bezier(0.25, 0.46, 0.45, 0.94)'
: 'none';
track.style.transform = `translateX(${-_cur * (100 / 3)}%)`;
_updateDots();
_updateFab();
// Karte neu rendern nachdem Transition abgeschlossen
if (_cur === 2 && _map) {
setTimeout(() => _map.invalidateSize(), animated ? 380 : 50);
}
}
function _updateDots() {
document.querySelectorAll('.wdot').forEach((d, i) => d.classList.toggle('active', i === _cur));
document.querySelectorAll('.wlabel').forEach((l, i) => l.classList.toggle('active', i === _cur));
}
function _updateFab() {
const fab = document.getElementById('worlds-fab');
if (!fab) return;
const icons = ['note-pencil', 'paw-print', 'warning'];
const titles = ['Schnelleintrag', 'Hund-Eintrag', 'Alarm melden'];
fab.querySelector('use')?.setAttribute('href', `/icons/phosphor.svg#${icons[_cur]}`);
fab.title = titles[_cur];
}
function _setupButtons() {
document.getElementById('worlds-fab')?.addEventListener('click', _openFab);
document.getElementById('worlds-back')?.addEventListener('click', () => show());
document.querySelectorAll('.wdot').forEach((dot, i) => {
dot.style.pointerEvents = 'auto';
dot.addEventListener('click', () => {
_goTo(i, true);
if (i === 2 && !_weltInited) { _weltInited = true; _renderWelt(); }
});
});
document.querySelectorAll('.wlabel').forEach((lbl, i) => {
lbl.style.pointerEvents = 'auto';
lbl.style.cursor = 'pointer';
lbl.addEventListener('click', () => {
_goTo(i, true);
if (i === 2 && !_weltInited) { _weltInited = true; _renderWelt(); }
});
});
}
function _openFab() {
const isWelt = _cur === 2;
const dogName = _state?.user ? null : null; // falls mehrere Hunde: erweiterbar
const options = isWelt ? [
{ icon:'skull', color:'#EF4444', label:'Giftköder melden', sub:'Warnung für andere Hundebesitzer', page:'poison' },
{ icon:'dog', color:'#3B82F6', label:'Verlorenen Hund melden', sub:'Hilf beim Wiederfinden', page:'lost' },
{ icon:'map-pin', color:'#10B981', label:'Ort vorschlagen', sub:'Neuen POI auf der Karte', page:'map' },
] : [
{ icon:'book-open', color:'#8B5CF6', label:'Tagebucheintrag', sub:'Erlebnis, Foto oder Notiz', page:'diary', action:'openNew' },
{ icon:'target', color:'#F59E0B', label:'Training aufzeichnen',sub:'Übung absolviert', page:'uebungen' },
{ icon:'heartbeat', color:'#EF4444', label:'Tierarztbesuch', sub:'Befund oder Impfung eintragen', page:'health' },
{ icon:'wave-sine', color:'#06B6D4', label:'Gewicht messen', sub:'Aktuelles Gewicht eintragen', page:'health' },
];
// Overlay erstellen
const ov = document.createElement('div');
ov.id = 'fab-overlay';
ov.style.cssText = 'position:fixed;inset:0;z-index:300;display:flex;flex-direction:column;justify-content:flex-end';
ov.innerHTML = `
<div id="fab-backdrop" style="position:absolute;inset:0;background:rgba(0,0,0,0.55);backdrop-filter:blur(2px)"></div>
<div style="position:relative;z-index:1;background:var(--c-bg);border-radius:24px 24px 0 0;
padding:20px 16px calc(env(safe-area-inset-bottom,16px) + 16px);
box-shadow:0 -8px 32px rgba(0,0,0,0.2)">
<div style="display:flex;align-items:center;justify-content:space-between;margin-bottom:16px">
<div style="font-size:var(--text-base);font-weight:700">
${isWelt ? 'Was möchtest du melden?' : 'Was möchtest du eintragen?'}
</div>
<button id="fab-close" style="background:var(--c-border);border:none;border-radius:50%;
width:28px;height:28px;cursor:pointer;display:flex;align-items:center;justify-content:center">
<svg class="ph-icon" style="width:14px;height:14px"><use href="/icons/phosphor.svg#x"></use></svg>
</button>
</div>
<div style="display:flex;flex-direction:column;gap:10px">
${options.map(o => `
<button class="fab-option" data-page="${o.page}" data-action="${o.action || ''}"
style="display:flex;align-items:center;gap:14px;width:100%;
background:var(--c-bg-card);border:1px solid var(--c-border);
border-radius:14px;padding:14px 16px;cursor:pointer;text-align:left;
transition:background .12s">
<div style="width:40px;height:40px;border-radius:12px;background:${o.color}18;
display:flex;align-items:center;justify-content:center;flex-shrink:0">
<svg class="ph-icon" style="width:1.2rem;height:1.2rem;color:${o.color}">
<use href="/icons/phosphor.svg#${o.icon}"></use>
</svg>
</div>
<div>
<div style="font-size:var(--text-sm);font-weight:700">${o.label}</div>
<div style="font-size:var(--text-xs);color:var(--c-text-secondary);margin-top:2px">${o.sub}</div>
</div>
</button>
`).join('')}
</div>
</div>
`;
document.body.appendChild(ov);
const _close = () => ov.remove();
ov.querySelector('#fab-backdrop').addEventListener('click', _close);
ov.querySelector('#fab-close').addEventListener('click', _close);
ov.querySelectorAll('.fab-option').forEach(btn => {
btn.addEventListener('click', () => {
_close();
const page = btn.dataset.page;
const action = btn.dataset.action;
navigateTo(page);
if (action === 'openNew') {
setTimeout(() => window.App?.callModule?.(page, 'openNew'), 400);
}
});
});
}
// ── CHIP-KONFIGURATION ──────────────────────────────────────
// Alle verfügbaren Chips mit Metadaten
const _ALL_CHIPS = [
{ icon:'note-pencil', label:'Notizblock', page:'notes' },
{ icon:'currency-eur', label:'Ausgaben', page:'expenses' },
{ icon:'first-aid', label:'Erste Hilfe', page:'erste-hilfe' },
{ icon:'handshake', label:'Playdate', page:'playdate' },
{ icon:'chat-circle-dots', label:'Nachrichten', page:'chat' },
{ icon:'sun', label:'Wetter', page:'wetter' },
{ icon:'book-open', label:'Tagebuch', page:'diary' },
{ icon:'heartbeat', label:'Gesundheit', page:'health' },
{ icon:'target', label:'Übungen', page:'uebungen' },
{ icon:'list-checks', label:'Trainings-\npläne',page:'trainingsplaene'},
{ icon:'heart', label:'Adoption', page:'adoption' },
{ icon:'house-line', label:'Sitting', page:'sitting' },
{ icon:'books', label:'Wiki', page:'wiki' },
{ icon:'scales', label:'Wurfbörse', page:'wurfboerse' },
{ icon:'map-trifold', label:'Karte', page:'map' },
{ icon:'push-pin', label:'Forum', page:'forum' },
{ icon:'users', label:'Freunde', page:'friends' },
{ icon:'paw-print', label:'Gassi', page:'walks' },
{ icon:'skull', label:'Giftköder', page:'poison' },
{ icon:'warning-circle', label:'Rückrufe', page:'recalls' },
{ icon:'dog', label:'Verlorene', page:'lost' },
{ icon:'path', label:'Routen', page:'routes' },
{ icon:'calendar-dots', label:'Events', page:'events' },
{ icon:'sparkle', label:'Jobs', page:'jobs' },
{ icon:'book-open', label:'Knigge', page:'knigge' },
{ icon:'film-slate', label:'Filme', page:'movies' },
{ icon:'tree-structure', label:'Zucht-\nkartei', page:'zuchthunde', role:'breeder' },
{ icon:'notebook', label:'Wurfverw.', page:'litters', role:'breeder' },
{ icon:'sparkle', label:'Social', page:'social', role:'social' },
{ icon:'shield-check', label:'Moderation', page:'moderation', role:'mod' },
{ icon:'gear', label:'Admin', page:'admin', role:'admin' },
];
const _DEFAULT_CONFIG = {
jetzt: ['notes','expenses','erste-hilfe','playdate','chat','wetter','settings'],
hund: ['diary','health','uebungen','trainingsplaene','adoption','sitting','wiki','wurfboerse'],
welt: ['map','forum','friends','walks','poison','recalls','lost','routes','events','jobs','knigge','movies'],
};
// _cfgCache: wird beim Init aus DB geladen, Fallback localStorage → Default
let _cfgCache = null;
async function _loadConfigFromServer() {
try {
const res = await API.get('/profile/world-config');
if (res?.config) {
_cfgCache = res.config;
try { localStorage.setItem('world_chips', JSON.stringify(_cfgCache)); } catch {}
return;
}
} catch {}
// Fallback: localStorage
try { _cfgCache = JSON.parse(localStorage.getItem('world_chips') || 'null') || _DEFAULT_CONFIG; }
catch { _cfgCache = _DEFAULT_CONFIG; }
}
function _getConfig() {
return _cfgCache || _DEFAULT_CONFIG;
}
function _saveConfig(cfg) {
_cfgCache = cfg;
try { localStorage.setItem('world_chips', JSON.stringify(cfg)); } catch {}
if (_state?.user) {
API.put('/profile/world-config', { config: cfg }).catch(() => {});
}
}
function _chipMeta(page) {
return _ALL_CHIPS.find(c => c.page === page) || null;
}
function _chipAllowed(chip) {
const u = _state?.user;
if (!chip?.role) return true;
if (chip.role === 'breeder') return u?.rolle === 'breeder' || u?.rolle === 'admin';
if (chip.role === 'social') return u?.is_social_media || u?.rolle === 'admin';
if (chip.role === 'mod') return u?.rolle === 'admin' || u?.rolle === 'moderator' || u?.is_moderator;
if (chip.role === 'admin') return u?.rolle === 'admin';
return true;
}
function _chipsForWorld(world) {
const pages = _getConfig()[world] || _DEFAULT_CONFIG[world];
return pages.map(_chipMeta).filter(c => c && _chipAllowed(c));
}
// ── KONFIGURATIONS-MODAL ─────────────────────────────────────
function _openConfigModal() {
let cfg = JSON.parse(JSON.stringify(_getConfig())); // deep copy
let _drag = null; // { page, fromWorld, ghost }
const worldColors = { jetzt:'rgba(196,132,58,0.6)', hund:'rgba(196,132,58,0.8)', welt:'rgba(99,130,220,0.6)' };
const worldLabels = { jetzt:'JETZT', hund:'HUND', welt:'WELT', pool:'Nicht verwendet' };
const allAssigned = () => new Set([...cfg.jetzt, ...cfg.hund, ...cfg.welt]);
const poolChips = () => _ALL_CHIPS.filter(c => _chipAllowed(c) && !allAssigned().has(c.page) && !c.pinned);
const bottomNav = document.getElementById('bottom-nav');
if (bottomNav) bottomNav.style.display = 'none';
const ov = document.createElement('div');
ov.id = 'wc-overlay';
ov.style.cssText = 'position:fixed;inset:0;z-index:500;display:flex;flex-direction:column;justify-content:flex-end';
document.body.appendChild(ov);
const _removeDragListeners = () => {
document.removeEventListener('touchmove', _onDragMove);
document.removeEventListener('touchend', _onDragEnd);
document.removeEventListener('touchcancel', _onDragEnd);
};
const _cancelDrag = () => {
if (!_drag) return;
_removeDragListeners();
_drag.ghost?.remove();
if (_drag.chipEl) _drag.chipEl.style.opacity = '';
ov.querySelectorAll('.wc-zone').forEach(z => z.style.borderColor = 'transparent');
_drag = null;
};
const _closeModal = () => {
_cancelDrag(); // laufenden Drag abbrechen
ov.remove();
if (bottomNav) bottomNav.style.removeProperty('display');
};
function _render() {
ov.innerHTML = `
<div id="wc-bg" style="position:absolute;inset:0;background:rgba(0,0,0,0.7);backdrop-filter:blur(4px)"></div>
<div id="wc-sheet" style="position:relative;z-index:1;background:rgba(18,22,32,0.97);border-radius:24px 24px 0 0;
max-height:92vh;overflow-y:auto;-webkit-overflow-scrolling:touch;
padding:0 0 calc(env(safe-area-inset-bottom,16px)+16px)">
<!-- Header -->
<div style="display:flex;align-items:center;justify-content:space-between;
padding:20px 20px 16px;position:sticky;top:0;background:rgba(18,22,32,0.97);
border-bottom:1px solid rgba(255,255,255,0.1);z-index:2">
<button id="wc-cancel" style="background:none;border:none;color:rgba(255,255,255,0.5);
cursor:pointer;font-size:var(--text-sm);padding:8px">Abbrechen</button>
<div style="font-size:var(--text-base);font-weight:700;color:white">Welten einrichten</div>
<button id="wc-save" style="background:var(--c-primary);color:white;border:none;
border-radius:999px;padding:7px 16px;cursor:pointer;font-weight:700;
font-size:var(--text-sm)">Fertig</button>
</div>
<!-- Hinweis + Reset -->
<div style="padding:10px 20px 6px;display:flex;align-items:center;justify-content:space-between;gap:12px">
<div style="font-size:var(--text-xs);color:rgba(255,255,255,0.4);flex:1">
Lang drücken &amp; ziehen zum Verschieben. ✕ zum Entfernen.
</div>
<button id="wc-reset" style="background:none;border:1px solid rgba(255,255,255,0.2);
color:rgba(255,255,255,0.5);border-radius:999px;padding:5px 12px;
cursor:pointer;font-size:11px;white-space:nowrap;flex-shrink:0">
Zurücksetzen
</button>
</div>
${['jetzt','hund','welt','pool'].map(w => {
const chips = w === 'pool' ? poolChips() : (cfg[w] || []).map(_chipMeta).filter(c => c && _chipAllowed(c));
const col = worldColors[w] || 'var(--c-border)';
return `
<div style="padding:12px 14px 6px">
<div style="font-size:10px;font-weight:800;letter-spacing:.1em;color:rgba(255,255,255,0.5);
margin-bottom:8px;display:flex;align-items:center;gap:8px">
<span style="display:inline-block;width:12px;height:12px;border-radius:50%;
background:${col}"></span>
${worldLabels[w]}
${w !== 'pool' ? `<span style="opacity:.5">(${chips.length})</span>` : ''}
</div>
<div class="wc-zone" data-zone="${w}"
style="display:grid;grid-template-columns:repeat(4,1fr);grid-auto-rows:80px;gap:8px;
min-height:${w==='pool'&&chips.length===0?'40px':'auto'};
border:2px dashed transparent;border-radius:16px;padding:4px;
transition:border-color .2s">
${chips.map(c => `
<div class="wc-chip" data-page="${c.page}" data-zone="${w}"
style="background:rgba(38,46,62,0.95);border:1.5px solid ${col};
border-radius:16px;padding:10px 4px 8px;height:80px;box-sizing:border-box;
display:flex;flex-direction:column;align-items:center;justify-content:center;gap:5px;
cursor:grab;position:relative;min-width:0;overflow:hidden;
user-select:none;-webkit-tap-highlight-color:transparent;touch-action:none">
${!c.pinned ? `
<button class="wc-remove" data-page="${c.page}" data-zone="${w}"
style="position:absolute;top:-6px;right:-6px;width:18px;height:18px;
border-radius:50%;background:#EF4444;border:none;cursor:pointer;
display:flex;align-items:center;justify-content:center;z-index:2">
<svg class="ph-icon" style="width:9px;height:9px;color:white">
<use href="/icons/phosphor.svg#x"></use>
</svg>
</button>` : `
<div style="position:absolute;top:-4px;right:-4px;width:14px;height:14px;
border-radius:50%;background:rgba(196,132,58,0.9);
display:flex;align-items:center;justify-content:center"
title="Immer sichtbar">
<svg class="ph-icon" style="width:8px;height:8px;color:white">
<use href="/icons/phosphor.svg#lock-simple"></use>
</svg>
</div>`}
<svg class="ph-icon" style="width:1.4rem;height:1.4rem;color:white;flex-shrink:0">
<use href="/icons/phosphor.svg#${c.icon}"></use>
</svg>
<span style="font-size:9px;font-weight:600;color:rgba(255,255,255,0.9);
line-height:1.2;text-align:center;width:100%;
overflow:hidden;display:-webkit-box;
-webkit-line-clamp:2;-webkit-box-orient:vertical;
padding:0 2px">${c.label.replace('\n','<br>')}</span>
</div>
`).join('')}
${chips.length === 0 ? `<div style="grid-column:1/-1;text-align:center;
font-size:var(--text-xs);color:rgba(255,255,255,0.35);padding:12px">
${w==='pool'?'Alle Chips sind zugeordnet':'Leer — ziehe Chips hierher'}
</div>` : ''}
</div>
</div>
`;
}).join('')}
</div>
`;
_bindEvents();
}
function _bindEvents() {
ov.querySelector('#wc-bg')?.addEventListener('click', _closeModal);
ov.querySelector('#wc-cancel')?.addEventListener('click', _closeModal);
ov.querySelector('#wc-reset')?.addEventListener('click', () => {
cfg = JSON.parse(JSON.stringify(_DEFAULT_CONFIG));
_render();
});
ov.querySelector('#wc-save')?.addEventListener('click', () => {
_saveConfig(cfg);
_closeModal();
_renderJetzt(); _renderHund(); _renderWelt();
});
// Remove-Buttons
ov.querySelectorAll('.wc-remove').forEach(btn => {
btn.addEventListener('click', e => {
e.stopPropagation();
const page = btn.dataset.page, zone = btn.dataset.zone;
const meta = _chipMeta(page);
if (meta?.pinned) return; // gepinnte Chips können nicht entfernt werden
if (zone !== 'pool') cfg[zone] = cfg[zone].filter(p => p !== page);
_render();
});
});
// Touch-Drag
ov.querySelectorAll('.wc-chip').forEach(chip => {
chip.addEventListener('touchstart', e => _onDragStart(e, chip), { passive: true });
});
document.addEventListener('touchmove', _onDragMove, { passive: false });
document.addEventListener('touchend', _onDragEnd);
}
function _onDragStart(e, chipEl) {
if (_drag) _cancelDrag();
const touch = e.touches[0];
// Drag erst nach Bewegungs-Threshold aktivieren (verhindert Scroll-Konflikte)
_drag = {
page: chipEl.dataset.page, zone: chipEl.dataset.zone,
chipEl, ghost: null, dropZone: null, active: false,
startX: touch.clientX, startY: touch.clientY, ox: 0, oy: 0,
};
document.addEventListener('touchmove', _onDragMove, { passive: false });
document.addEventListener('touchend', _onDragEnd);
document.addEventListener('touchcancel', _onDragEnd);
}
function _activateDrag(touch) {
const rect = _drag.chipEl.getBoundingClientRect();
_drag.ox = _drag.startX - rect.left;
_drag.oy = _drag.startY - rect.top;
_drag.active = true;
const ghost = _drag.chipEl.cloneNode(true);
ghost.querySelectorAll('button').forEach(b => b.style.display = 'none');
ghost.style.position = 'fixed';
ghost.style.zIndex = '9999';
ghost.style.opacity = '0.9';
ghost.style.pointerEvents= 'none';
ghost.style.transform = 'scale(1.08) rotate(-2deg)';
ghost.style.width = rect.width + 'px';
ghost.style.height = rect.height + 'px';
ghost.style.left = (touch.clientX - _drag.ox) + 'px';
ghost.style.top = (touch.clientY - _drag.oy) + 'px';
ghost.style.transition = 'none';
ghost.style.boxShadow = '0 8px 24px rgba(0,0,0,0.5)';
document.body.appendChild(ghost);
_drag.ghost = ghost;
_drag.chipEl.style.opacity = '0.2';
}
function _onDragMove(e) {
if (!_drag) return;
const touch = e.touches[0];
if (!_drag.active) {
const dx = Math.abs(touch.clientX - _drag.startX);
const dy = Math.abs(touch.clientY - _drag.startY);
if (dx < 8 && dy < 8) return; // Threshold noch nicht erreicht
_activateDrag(touch);
}
e.preventDefault(); // Scroll erst NACH Threshold blockieren
_drag.ghost.style.left = (touch.clientX - _drag.ox) + 'px';
_drag.ghost.style.top = (touch.clientY - _drag.oy) + 'px';
let foundZone = null;
ov.querySelectorAll('.wc-zone').forEach(z => {
const r = z.getBoundingClientRect();
const over = touch.clientX >= r.left && touch.clientX <= r.right
&& touch.clientY >= r.top && touch.clientY <= r.bottom;
z.style.borderColor = over ? (worldColors[z.dataset.zone] || 'rgba(196,132,58,0.8)') : 'transparent';
if (over) foundZone = z.dataset.zone;
});
_drag.dropZone = foundZone;
}
function _onDragEnd() {
if (!_drag) return;
_removeDragListeners();
const { ghost, chipEl, dropZone, page, zone: fromZone, active } = _drag;
_drag = null;
ov.querySelectorAll('.wc-zone').forEach(z => z.style.borderColor = 'transparent');
ghost?.remove();
if (chipEl) chipEl.style.opacity = '';
if (!active) return; // nur Tap, kein Drag — nichts verschieben
const meta = _chipMeta(page);
if (dropZone && dropZone !== fromZone && !(meta?.pinned && dropZone === 'pool')) {
if (fromZone !== 'pool') cfg[fromZone] = cfg[fromZone].filter(p => p !== page);
if (dropZone !== 'pool' && !cfg[dropZone].includes(page)) cfg[dropZone].push(page);
_render();
}
}
_render();
}
// ── PANORAMA HINTERGRUNDBILD ─────────────────────────────────
async function _loadDailyImage(dog) {
if (!dog) return null;
const todayKey = 'bg_' + new Date().toISOString().slice(0, 10);
const cached = _wLoad(todayKey);
if (cached?.data) return cached.data;
try {
const r = await _cachedGet(`diary_${dog.id}`, `/dogs/${dog.id}/diary?limit=30`);
const entries = r.data?.entries || r.data || [];
const withPhotos = entries.filter(e => (e.foto_urls?.length || e.foto_url));
if (!withPhotos.length) { const u = dog.foto_url || null; if(u) _wSave(todayKey, u); return u; }
const day = Math.floor(Date.now() / 86400000);
const entry = withPhotos[day % withPhotos.length];
const url = (entry.foto_urls?.[0] || entry.foto_url);
_wSave(todayKey, url);
return url;
} catch { return dog.foto_url || null; }
}
function _applyBgImage(url) {
const track = document.getElementById('worlds-track');
if (!track) return;
if (url) {
const img = new Image();
img.onload = () => { track.style.backgroundImage = `url('${url}')`; track.style.backgroundSize = '100% auto'; track.style.backgroundPosition = '0 40%'; track.style.backgroundRepeat = 'no-repeat'; };
img.onerror = () => _applyBgImage(null);
img.src = url;
} else {
track.style.backgroundImage = 'linear-gradient(160deg,#1a1f35 0%,#16213e 33%,#1a2535 67%,#0f1921 100%)';
track.style.backgroundSize = '100% 100%';
}
}
// ── CHIP-HELPER ──────────────────────────────────────────────
function _chip(icon, label, page) {
return `
<div class="world-chip" data-wnav="${page}">
<svg class="ph-icon" style="width:1.4rem;height:1.4rem">
<use href="/icons/phosphor.svg#${icon}"></use>
</svg>
<span class="world-chip-label">${label}</span>
</div>`;
}
// ── JETZT WORLD ──────────────────────────────────────────────
async function _renderJetzt() {
const el = document.getElementById('wj-content');
if (!el) return;
const user = _state?.user;
el.innerHTML = _skeleton(3);
const [weatherRes, dogsRes, alertsRes] = await Promise.allSettled([
_getCachedWeather(),
user ? _cachedGet('dogs', '/dogs') : Promise.resolve({ data: [], fromCache: false, ageMin: 0 }),
user ? _getNearbyAlerts() : Promise.resolve([]),
]);
const weatherObj = weatherRes.value || { data: null, fromCache: false, ageMin: 0 };
const dogsObj = dogsRes.value || { data: [], fromCache: false, ageMin: 0 };
const w = weatherObj.data;
const dogList = dogsObj.data || [];
const dog = dogList[0] || null;
const alertList = alertsRes.value || [];
const isOffline = weatherObj.fromCache && dogsObj.fromCache;
const staleMin = Math.max(weatherObj.ageMin || 0, dogsObj.ageMin || 0);
// Panorama-Bild setzen (nur wenn noch kein Bild vorhanden)
const track = document.getElementById('worlds-track');
if (dog && !track?.style.backgroundImage?.startsWith('url')) {
_loadDailyImage(dog).then(_applyBgImage);
} else if (!dog) { _applyBgImage(null); }
const hour = new Date().getHours();
const greet = hour < 5 ? 'Gute Nacht' : hour < 12 ? 'Guten Morgen' : hour < 18 ? 'Hallo' : 'Guten Abend';
const firstName = user?.name?.split(' ')[0] || '';
const dayStr = new Date().toLocaleDateString('de-DE', { weekday:'long', day:'numeric', month:'long' });
const stale = isOffline && staleMin > 5
? `<span style="font-size:9px;opacity:0.5;margin-left:6px">· Offline</span>` : '';
const weatherLine = w
? `${Math.round(w.temp_c)}° ${_esc(w.desc?.split(' ')[0] || '')} · ${Math.round(w.wind_kmh || 0)} km/h · ${w.precip_prob || 0}% Regen`
: '';
// Streak für 3er-Chip-Zeile
let streakVal = '—', streakCol = 'rgba(255,255,255,0.4)';
if (user && dog) {
try {
const sr = await _cachedGet(`streak_${dog.id}`, `/streak/${dog.id}`);
const s = sr.data;
const streak = s?.current_streak || 0;
const trainedToday = s?.last_training_date === new Date().toISOString().slice(0,10);
streakCol = trainedToday ? '#10B981' : (streak > 0 ? '#F59E0B' : 'rgba(255,255,255,0.4)');
streakVal = streak > 0
? (trainedToday ? `${streak} Tage` : `🔥 ${streak} Tage`)
: (trainedToday ? '✓ Heute' : 'Heute starten');
} catch {}
}
// Alert-Reminder
const alertHtml = alertList.slice(0,1).map(a => `
<div class="world-reminder" data-wnav="${a.page}" style="border-color:rgba(239,68,68,0.5)">
<svg class="ph-icon" style="width:1.1rem;height:1.1rem;color:#EF4444">
<use href="/icons/phosphor.svg#${a.icon}"></use>
</svg>
<span style="font-size:var(--text-xs);font-weight:700;color:#FCA5A5">${_esc(a.title)}</span>
</div>`).join('');
// Feature Chips aus Config
const features = user
? _chipsForWorld('jetzt')
: [
{ icon:'sun', label:'Wetter', page:'wetter' },
{ icon:'first-aid',label:'Erste Hilfe', page:'erste-hilfe' },
{ icon:'gear', label:'Anmelden', page:'settings' },
];
// User-Avatar für JETZT Info-Card
const uFoto = _state?.user?.avatar_url;
const uInitial = firstName ? firstName.charAt(0).toUpperCase() : '?';
const userAvatarHtml = `
<div data-wnav="settings" style="flex-shrink:0;cursor:pointer"
title="Mein Profil">
${uFoto
? `<img src="${_esc(uFoto)}" style="width:40px;height:40px;border-radius:50%;
object-fit:cover;border:2px solid rgba(255,255,255,0.35);display:block">`
: `<div style="width:40px;height:40px;border-radius:50%;background:var(--c-primary);
display:flex;align-items:center;justify-content:center;
font-size:var(--text-base);font-weight:800;color:white;
border:2px solid rgba(255,255,255,0.35)">${uInitial}</div>`}
</div>`;
el.innerHTML = `
<div class="world-top">
<div class="world-info-card">
<div style="display:flex;align-items:center;gap:12px">
<div style="flex:1;min-width:0">
<div class="world-info-title">
${_esc(greet)}${firstName ? `, <span style="color:var(--c-primary)">${_esc(firstName)}</span>` : ''}${stale}
</div>
<div class="world-info-sub">${_esc(dayStr)}${weatherLine ? ' · ' + weatherLine : ''}</div>
</div>
${user ? userAvatarHtml : ''}
</div>
</div>
${alertHtml}
${user && dog ? `
<div class="wj-chip-row">
<div class="wj-chip" data-wnav="uebungen">
<svg class="ph-icon" style="width:1.2rem;height:1.2rem;color:${streakCol}">
<use href="/icons/phosphor.svg#target"></use></svg>
<span class="wj-chip-label">Streak</span>
<span class="wj-chip-val">${streakVal}</span>
</div>
<div class="wj-chip" data-wnav="routes">
<svg class="ph-icon" style="width:1.2rem;height:1.2rem;color:var(--c-primary)">
<use href="/icons/phosphor.svg#path"></use></svg>
<span class="wj-chip-label">Gassirunde</span>
<span class="wj-chip-val" id="wj-route-val">…</span>
</div>
<div class="wj-chip" data-wnav="uebungen">
<svg class="ph-icon" style="width:1.2rem;height:1.2rem;color:var(--c-primary)">
<use href="/icons/phosphor.svg#barbell"></use></svg>
<span class="wj-chip-label">Übung</span>
<span class="wj-chip-val" id="wj-exercise-val">…</span>
</div>
</div>` : ''}
</div>
<div class="world-bottom">
<div class="world-section-label">Deine Bereiche</div>
<div class="world-chips-grid">
${features.map(f => _chip(f.icon, f.label, f.page)).join('')}
</div>
<div class="world-footer-links">
<span data-wnav="impressum">Impressum</span>
</div>
</div>
`;
el.querySelectorAll('[data-wnav]').forEach(e => e.addEventListener('click', () => navigateTo(e.dataset.wnav)));
if (user && dog) {
_loadJetztExercise(dog);
_loadJetztRoute();
}
}
async function _loadJetztExercise(dog) {
const valEl = document.getElementById('wj-exercise-val');
if (!valEl) return;
try {
const res = await _cachedGet(`dash_${dog.id}`, `/dogs/${dog.id}/welcome-dashboard`);
const ex = res.data?.daily_exercise;
valEl.textContent = ex?.name || '—';
} catch { valEl.textContent = '—'; }
}
async function _loadJetztRoute() {
const valEl = document.getElementById('wj-route-val');
if (!valEl) return;
// Tages-Cache (gleicher Key wie welcome.js)
const today = new Date().toISOString().slice(0, 10);
const cacheKey = 'by_daily_route_' + today;
const cached = localStorage.getItem(cacheKey);
if (cached) {
try { _applyJetztRoute(valEl, JSON.parse(cached)); return; } catch {}
}
let loc;
try { loc = await API.getLocation({ timeout: 5000, maximumAge: 300000 }); }
catch { valEl.textContent = 'Standort nötig'; return; }
const dayIdx = Math.floor(Date.now() / 86400000);
const km = [2, 4, 6][dayIdx % 3];
const seed = dayIdx % 5;
try {
const result = await API.post('/routes/suggest', { lat: loc.lat, lon: loc.lon, distance_km: km, seed });
if (!result?.gps_track?.length) { valEl.textContent = 'Keine Route gefunden'; return; }
localStorage.setItem(cacheKey, JSON.stringify(result));
// alte Einträge aufräumen
Object.keys(localStorage)
.filter(k => k.startsWith('by_daily_route_') && k !== cacheKey)
.forEach(k => localStorage.removeItem(k));
_applyJetztRoute(valEl, result);
} catch { valEl.textContent = 'Route nicht verfügbar'; }
}
function _applyJetztRoute(valEl, result) {
const durStr = result.dauer_min < 60
? `${result.dauer_min} min`
: `${Math.floor(result.dauer_min / 60)}h ${result.dauer_min % 60 > 0 ? ' ' + (result.dauer_min % 60) + 'min' : ''}`;
valEl.textContent = `${result.distanz_km} km · ${durStr}`;
}
// ── HUND WORLD ───────────────────────────────────────────────
async function _renderHund() {
const el = document.getElementById('wh-content');
if (!el) return;
const user = _state?.user;
if (!user) {
el.innerHTML = `
<div class="world-info-card" style="text-align:center">
<div style="font-size:4rem;margin-bottom:12px">🐾</div>
<div class="world-info-title">Dein Hund wartet</div>
<div class="world-info-sub" style="margin-bottom:20px">Melde dich an um loszulegen</div>
<button class="btn btn-primary" onclick="Worlds.navigateTo('settings')">Anmelden</button>
</div>`;
return;
}
el.innerHTML = _skeleton(4);
const dogsRes = await _cachedGet('dogs', '/dogs');
const dogs = dogsRes.data || [];
if (!dogs.length) {
el.innerHTML = `
<div class="world-info-card" style="text-align:center">
<div style="font-size:4rem;margin-bottom:12px">🐶</div>
<div class="world-info-title">Noch kein Hund angelegt</div>
<div class="world-info-sub" style="margin-bottom:20px">Erstelle das Profil deines Hundes</div>
<button class="btn btn-primary" onclick="Worlds.navigateTo('dog-profile')">Hund anlegen</button>
</div>`;
return;
}
// Hunde global cachen für schnelles Cycling
_dogs = dogs;
if (_dogIdx >= _dogs.length) _dogIdx = 0;
const dog = _dogs[_dogIdx];
const [streakRes, diaryRes] = await Promise.allSettled([
_cachedGet(`streak_${dog.id}`, `/streak/${dog.id}`),
_cachedGet(`diary_${dog.id}`, `/dogs/${dog.id}/diary?limit=1`),
]);
const streak = streakRes.value?.data ?? streakRes.value;
const diaryData = diaryRes.value?.data ?? diaryRes.value;
const lastEntry = diaryData?.entries?.[0] || diaryData?.[0] || null;
const ageStr = dog.alter_jahre ? _fmtAlter(dog.alter_jahre) : '';
const stats = [dog.rasse, ageStr, dog.gewicht_kg ? dog.gewicht_kg + ' kg' : null].filter(Boolean).join(' · ');
const otherDogs = _dogs.filter((_, i) => i !== _dogIdx);
const chips = _chipsForWorld('hund');
// Avatar des aktuellen Hundes (links, → Hundeprofil)
const dogAvatarHtml = dog.foto_url
? `<img src="${_esc(dog.foto_url)}" style="width:44px;height:44px;border-radius:50%;
object-fit:cover;border:2px solid rgba(255,255,255,0.35);
cursor:pointer;flex-shrink:0" id="wh-avatar">`
: `<div id="wh-avatar" style="width:44px;height:44px;border-radius:50%;background:var(--c-primary);
display:flex;align-items:center;justify-content:center;
font-size:1.3rem;flex-shrink:0;cursor:pointer;
border:2px solid rgba(255,255,255,0.25)">🐶</div>`;
// Andere Hunde-Avatare (rechts, überlagernd, → Hund wechseln)
const otherAvatarsHtml = otherDogs.length > 0 ? `
<div style="display:flex;align-items:center;flex-shrink:0">
${otherDogs.slice(0, 4).map((d, i) => {
const targetIdx = _dogs.indexOf(d);
return `
<div class="wh-other-dog" data-idx="${targetIdx}"
style="width:30px;height:30px;border-radius:50%;overflow:hidden;
border:2px solid rgba(255,255,255,0.4);
margin-left:${i > 0 ? '-10px' : '0'};
cursor:pointer;position:relative;z-index:${10 - i}">
${d.foto_url
? `<img src="${_esc(d.foto_url)}" style="width:100%;height:100%;object-fit:cover">`
: `<div style="width:100%;height:100%;background:rgba(100,70,40,0.85);
display:flex;align-items:center;justify-content:center;
font-size:10px;font-weight:700;color:white">
${_esc(d.name?.charAt(0)?.toUpperCase() || '?')}
</div>`}
</div>`;
}).join('')}
</div>` : '';
el.innerHTML = `
<div class="world-top">
<div class="world-info-card">
<div style="display:grid;grid-template-columns:1fr auto 1fr;align-items:center">
<div style="justify-self:start">${dogAvatarHtml}</div>
<div id="wh-cycle-btn" style="text-align:center;
cursor:${_dogs.length > 1 ? 'pointer' : 'default'};padding:0 8px">
<div class="world-info-title">${_esc(dog.name)}</div>
${stats ? `<div class="world-info-sub">${_esc(stats)}</div>` : ''}
</div>
<div style="justify-self:end;display:flex;align-items:center">${otherAvatarsHtml}</div>
</div>
</div>
</div>
<div class="world-bottom">
<div class="world-section-label">Alles über ${_esc(dog.name)}</div>
<div class="world-chips-grid">
${chips.map(c => _chip(c.icon, c.label, c.page)).join('')}
</div>
<div class="world-footer-links">
<span data-wnav="gruender">Die 100 Gründer</span>
</div>
</div>
`;
// Avatar → Hundeprofil
el.querySelector('#wh-avatar')?.addEventListener('click', () => navigateTo('dog-profile'));
// Chips
el.querySelectorAll('[data-wnav]').forEach(e => e.addEventListener('click', () => navigateTo(e.dataset.wnav)));
// Name → nächster Hund
if (_dogs.length > 1) {
el.querySelector('#wh-cycle-btn')?.addEventListener('click', () => {
_dogIdx = (_dogIdx + 1) % _dogs.length;
_renderHund();
});
}
// Andere Hund-Avatare → zu diesem Hund wechseln
el.querySelectorAll('.wh-other-dog').forEach(btn => {
btn.addEventListener('click', () => {
const idx = parseInt(btn.dataset.idx);
if (!isNaN(idx) && idx !== _dogIdx) { _dogIdx = idx; _renderHund(); }
});
});
}
// ── WELT WORLD ───────────────────────────────────────────────
const _QUOTES = [
{ t:'Ein Hund ist die einzige Kreatur, die dich mehr liebt als sich selbst.', a:'Josh Billings' },
{ t:'Hunde haben Besitzer. Katzen haben Personal.', a:'Unbekannt' },
{ t:'Bis man einen Hund geliebt hat, ist ein Teil der Seele unerwacht.', a:'Anatole France' },
{ t:'Hunde sind nicht unser ganzes Leben — aber sie machen unser Leben ganz.', a:'Roger Caras' },
{ t:'Der Hund hat nur einen Fehler: Er vertraut dem Menschen zu sehr.', a:'Unbekannt' },
{ t:'Ein treuer Hund ist besser als ein unzuverlässiger Mensch.', a:'Deutsches Sprichwort' },
{ t:'Wer einen Hund hat, gibt nie allein zu Tisch.', a:'Unbekannt' },
{ t:'Die reinste Form der Freude ist die Freude eines Hundes.', a:'Milan Kundera' },
{ t:'Hunde lachen nicht mit dem Mund, sondern mit dem Schwanz.', a:'Max Eastman' },
{ t:'Ein Haus ohne Hund ist ein leeres Haus.', a:'Unbekannt' },
{ t:'Wenn du verstehen willst, was Liebe ist, beobachte einen Hund.', a:'Unbekannt' },
{ t:'Der Hund fragt nicht warum. Er fragt nur: wann gehen wir?', a:'Unbekannt' },
{ t:'Hunde riechen nicht schlecht. Sie riechen nur anders als wir denken.', a:'Unbekannt' },
{ t:'In einer Welt voller Unsicherheiten ist der Hund das Verlässlichste.', a:'Unbekannt' },
{ t:'Ein Spaziergang mit einem Hund ist nie wirklich ein Umweg.', a:'Unbekannt' },
{ t:'Hunde bringen das Beste im Menschen hervor.', a:'Unbekannt' },
{ t:'Wer mit Hunden liegt, steht mit Flöhen auf — und mit einem Lächeln.', a:'Unbekannt' },
{ t:'Ein Hund weiß, wann du traurig bist. Er weiß nicht warum. Aber er ist da.', a:'Unbekannt' },
{ t:'Das Geheimnis eines Hundes: Er beurteilt dich nie.', a:'Unbekannt' },
{ t:'Der Hund ist Gattung: Mensch.', a:'Charles M. Schulz' },
{ t:'Hunde haben viel zu sagen. Wir haben verlernt zuzuhören.', a:'Unbekannt' },
{ t:'Ein guter Hund macht aus einem schlechten Tag einen erträglichen.', a:'Unbekannt' },
{ t:'Der Hund ist der philosophischste aller Haustiere.', a:'George Graham Vest' },
{ t:'Wer einen Hund hat, braucht keinen Therapeuten.', a:'Unbekannt' },
{ t:'Hunde lieben bedingungslos. Das ist ihr größtes Geschenk.', a:'Unbekannt' },
{ t:'Der schönste Empfang ist der eines Hundes an der Tür.', a:'Unbekannt' },
{ t:'Ein Hund ändert dein Leben. Meistens zum Besseren.', a:'Unbekannt' },
{ t:'Hunde altern in Würde. Menschen könnten von ihnen lernen.', a:'Unbekannt' },
{ t:'Kein Psychiater der Welt kann so gut zuhören wie ein Hund.', a:'Unbekannt' },
{ t:'Wo Hunde sind, da ist das Zuhause.', a:'Unbekannt' },
{ t:'Der Hund hat keinen Begriff von Vergangenheit oder Zukunft. Er lebt.', a:'Milan Kundera' },
];
function _renderWelt() {
const el = document.getElementById('ww-content');
if (!el) return;
const user = _state?.user;
const isMod = user?.rolle === 'admin' || user?.rolle === 'moderator' || user?.is_moderator;
const isAdmin = user?.rolle === 'admin';
const isSocial = user?.is_social_media || isAdmin;
// Tagesbasierter Spruch
const day = Math.floor(Date.now() / 86400000);
const quote = _QUOTES[day % _QUOTES.length];
const chips = _chipsForWorld('welt');
el.innerHTML = `
<div class="world-top">
<div class="world-info-card">
<div style="font-size:11px;font-weight:700;letter-spacing:.08em;text-transform:uppercase;
color:rgba(255,255,255,0.45);margin-bottom:10px">Gedanke des Tages</div>
<div style="font-size:var(--text-sm);color:white;line-height:1.55;font-style:italic">
»${_esc(quote.t)}«
</div>
<div style="font-size:10px;color:rgba(255,255,255,0.45);margin-top:8px;text-align:right">
${_esc(quote.a)}
</div>
</div>
</div>
<div class="world-bottom">
<div class="world-section-label">Die Welt da draußen</div>
<div class="world-chips-grid">
${chips.map(c => _chip(c.icon, c.label, c.page)).join('')}
</div>
<div class="world-footer-links">
<span data-wnav="datenschutz">Datenschutz</span>
</div>
</div>
`;
el.querySelectorAll('[data-wnav]').forEach(e => e.addEventListener('click', () => navigateTo(e.dataset.wnav)));
}
// ── HELPERS ──────────────────────────────────────────────────
async function _getCachedWeather() {
const cached = _wLoad('weather');
let fresh = null, pos = null;
try {
pos = await API.getLocation({ timeout: 5000, maximumAge: 600_000 });
fresh = await API.weather.get(pos.lat, pos.lon);
_wSave('weather', fresh);
} catch {}
const data = fresh ?? cached?.data ?? null;
return { data, fromCache: !fresh, ageMin: fresh ? 0 : (cached?.ageMin ?? null) };
}
async function _getWeather() {
const r = await _getCachedWeather();
return r.data;
}
async function _getNearbyAlerts() {
const out = [];
try {
const pos = await API.getLocation({ timeout: 4000, maximumAge: 600_000 });
const [p, l] = await Promise.allSettled([
API.get(`/poison/nearby?lat=${pos.lat}&lon=${pos.lon}&radius=5`).catch(() => []),
API.get(`/lost/nearby?lat=${pos.lat}&lon=${pos.lon}&radius=5`).catch(() => []),
]);
if (p.value?.length) out.push({ icon:'skull', color:'#EF4444', title:'Giftköder in der Nähe', sub:`${p.value.length} Meldung${p.value.length>1?'en':''}`, page:'poison' });
if (l.value?.length) out.push({ icon:'dog', color:'#3B82F6', title:'Verlorener Hund', sub:`${l.value.length} Meldung${l.value.length>1?'en':''}`, page:'lost' });
} catch {}
return out;
}
function _skeleton(n) {
return Array.from({length: n}, (_, i) => `
<div style="height:${i===0?80:60}px;background:var(--c-border);border-radius:12px;margin:0 16px 10px;opacity:${0.7 - i*0.1};animation:pulse 1.5s ease-in-out infinite"></div>
`).join('');
}
function _fmtDate(d) {
if (!d) return '';
try { return new Date(d).toLocaleDateString('de-DE', { day:'numeric', month:'short' }); }
catch { return d; }
}
function _fmtAlter(j) {
if (!j) return '';
if (j < 1) return `${Math.round(j * 12)} Monate`;
return j < 2 ? '1 Jahr' : `${Math.round(j)} Jahre`;
}
function _esc(s) {
if (s == null) return '';
return String(s).replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;').replace(/"/g,'&quot;');
}
return { init, show, hide, navigateTo, openConfig: _openConfigModal, get _visible() { return _visible; } };
})();