Feature: User-Feedback, Regen-Uhrzeit im Wetter-Chip, Admin-Karten klickbar (SW by-v833)
- Feedback-Modal im Settings (Kategorie + Text → E-Mail an support@banyaro.app) - Wetter-Chip (Karte + Gassi-Score): zeigt nächste Regenstunde ab ≥20% Wahrscheinlichkeit - Gassi-Score-Chip: zweizeilige Wetter-Info, linksbündig, volle Chipbreite - Admin-Übersicht: Stat-Karten anklickbar → navigiert direkt zum jeweiligen Tab - ui.js: visualViewport-Listener hebt Modal über Tastatur (alle Modals) - api.js: Pydantic v2 Array-Detail korrekt als Fehlermeldung extrahiert - map.js: Wetter-Fallback über watchPosition wenn getCurrentPosition scheitert - Update-Loop-Fix: index.html ?v= synchron mit APP_VER halten (alle 4 Stellen)
This commit is contained in:
parent
d18c592ef0
commit
70af387147
12 changed files with 211 additions and 42 deletions
|
|
@ -269,6 +269,12 @@ window.Page_settings = (() => {
|
|||
<span>Hilfe & FAQ</span>
|
||||
<span style="margin-left:auto;color:var(--c-text-secondary)">›</span>
|
||||
</div>
|
||||
<div class="sidebar-item" id="settings-feedback-btn"
|
||||
style="padding:var(--space-4);border-radius:0;border-bottom:1px solid var(--c-border);cursor:pointer">
|
||||
<svg class="ph-icon" aria-hidden="true"><use href="/icons/phosphor.svg#chat-dots"></use></svg>
|
||||
<span>Feedback geben</span>
|
||||
<span style="margin-left:auto;color:var(--c-text-secondary)">›</span>
|
||||
</div>
|
||||
${!_appState.user?.subscription_tier || _appState.user.subscription_tier === 'standard' || _appState.user.subscription_tier === 'standard_test' ? `
|
||||
<div style="margin:var(--space-3) 0;padding:var(--space-3) var(--space-4);
|
||||
background:rgba(196,132,58,0.1);border-radius:var(--radius-md);
|
||||
|
|
@ -790,6 +796,52 @@ window.Page_settings = (() => {
|
|||
App.navigate('hilfe');
|
||||
});
|
||||
|
||||
document.getElementById('settings-feedback-btn')?.addEventListener('click', () => {
|
||||
const sel = (id) => document.getElementById(id);
|
||||
const inputStyle = 'width:100%;padding:10px 12px;border:1.5px solid var(--c-border);border-radius:var(--radius-md);background:var(--c-bg-card);color:var(--c-text);font-size:var(--text-sm);box-sizing:border-box';
|
||||
UI.modal.open({
|
||||
title: 'Feedback geben',
|
||||
body: `
|
||||
<form id="feedback-form" style="display:flex;flex-direction:column;gap:var(--space-4)">
|
||||
<div>
|
||||
<label style="display:block;font-size:var(--text-sm);font-weight:600;margin-bottom:var(--space-1)">Kategorie</label>
|
||||
<select id="feedback-kat" name="kategorie" style="${inputStyle}">
|
||||
<option value="idee">💡 Idee / Wunsch</option>
|
||||
<option value="bug">🐛 Bug / Fehler</option>
|
||||
<option value="lob">🎉 Lob</option>
|
||||
<option value="sonstiges">💬 Sonstiges</option>
|
||||
</select>
|
||||
</div>
|
||||
<div>
|
||||
<label style="display:block;font-size:var(--text-sm);font-weight:600;margin-bottom:var(--space-1)">Deine Nachricht</label>
|
||||
<textarea id="feedback-text" name="text" rows="5" maxlength="2000"
|
||||
placeholder="Was möchtest du uns mitteilen?"
|
||||
style="${inputStyle};resize:vertical"></textarea>
|
||||
</div>
|
||||
</form>
|
||||
`,
|
||||
footer: `
|
||||
<div class="w3-btn-stack">
|
||||
<button type="submit" form="feedback-form" id="feedback-submit-btn" class="btn btn-primary" style="width:100%">Absenden</button>
|
||||
<button type="button" class="btn btn-secondary" data-modal-close>Abbrechen</button>
|
||||
</div>
|
||||
`,
|
||||
});
|
||||
|
||||
sel('feedback-form')?.addEventListener('submit', async e => {
|
||||
e.preventDefault();
|
||||
const btn = sel('feedback-submit-btn');
|
||||
const kat = sel('feedback-kat')?.value;
|
||||
const text = sel('feedback-text')?.value?.trim();
|
||||
if (!text) { UI.toast.error('Bitte schreib etwas.'); return; }
|
||||
await UI.asyncButton(btn, async () => {
|
||||
await API.post('/feedback', { kategorie: kat, text });
|
||||
UI.modal.close?.();
|
||||
UI.toast.success('Vielen Dank für dein Feedback!');
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
document.getElementById('settings-logout-btn')?.addEventListener('click', async () => {
|
||||
const ok = await UI.modal.confirm({
|
||||
title : 'Abmelden?',
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue