feat: PWA install button, route persistence, and active-user visual indicators
- Landing page shows an "add to home screen" button when the browser install prompt is available (Android/desktop) or on iOS Safari with a share-menu hint; hidden when already running in standalone mode - Last visited user route (/ray or /cer) is saved to Dexie meta and restored on next open so the app reopens where you left off - Active user in the topbar brand renders as a filled accent-color pill - A 1px accent-color frame wraps the entire viewport while a user is active
This commit is contained in:
parent
584357851d
commit
48deaa3219
3 changed files with 117 additions and 0 deletions
|
|
@ -514,6 +514,62 @@ body[data-user="cer"] .brand__user[data-brand="cer"] { color: var(--accent-cer);
|
|||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
/* ---- Install button ---- */
|
||||
|
||||
.install-wrap {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: var(--gap-2);
|
||||
}
|
||||
|
||||
.install-btn {
|
||||
background: transparent;
|
||||
border: var(--hairline) solid var(--line);
|
||||
color: var(--fg-mute);
|
||||
font-family: var(--font-mono);
|
||||
font-size: 11px;
|
||||
letter-spacing: 0.14em;
|
||||
text-transform: uppercase;
|
||||
padding: 8px 18px;
|
||||
cursor: pointer;
|
||||
border-radius: var(--radius);
|
||||
}
|
||||
.install-btn:hover { border-color: var(--fg-mute); color: var(--fg); }
|
||||
|
||||
.install-hint {
|
||||
font-family: var(--font-mono);
|
||||
font-size: 10px;
|
||||
color: var(--fg-mute);
|
||||
letter-spacing: 0.10em;
|
||||
text-transform: uppercase;
|
||||
margin: 0;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
/* ---- Active user pill in topbar ---- */
|
||||
|
||||
body[data-user="ray"] .brand--sm .brand__user[data-brand="ray"],
|
||||
body[data-user="cer"] .brand--sm .brand__user[data-brand="cer"] {
|
||||
background: var(--accent);
|
||||
color: var(--bg);
|
||||
padding: 2px 6px;
|
||||
border-radius: var(--radius);
|
||||
border-bottom-color: transparent;
|
||||
}
|
||||
|
||||
/* ---- Per-user site frame ---- */
|
||||
|
||||
body[data-user="ray"]::after,
|
||||
body[data-user="cer"]::after {
|
||||
content: '';
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
border: 1px solid var(--accent);
|
||||
pointer-events: none;
|
||||
z-index: 9999;
|
||||
}
|
||||
|
||||
/* ---- Misc ---- */
|
||||
|
||||
.sr-only {
|
||||
|
|
|
|||
|
|
@ -29,6 +29,10 @@
|
|||
<a href="/cer" class="brand__user brand__user--cer" data-link>cer</a>
|
||||
</div>
|
||||
<p class="landing__sub">choose to continue</p>
|
||||
<div class="install-wrap" data-install-wrap hidden>
|
||||
<button class="install-btn" data-install-btn>add to home screen</button>
|
||||
<p class="install-hint" data-install-hint hidden>tap share → "add to home screen"</p>
|
||||
</div>
|
||||
</main>
|
||||
</template>
|
||||
|
||||
|
|
|
|||
|
|
@ -4,6 +4,8 @@ import {
|
|||
startSync,
|
||||
setCompletion,
|
||||
onChange,
|
||||
getMeta,
|
||||
setMeta,
|
||||
} from './db.js';
|
||||
import { initTheme, cycleTheme, currentTheme } from './theme.js';
|
||||
import { renderCalendar } from './calendar.js';
|
||||
|
|
@ -19,6 +21,27 @@ import {
|
|||
const VALID_USERS = new Set(['ray', 'cer']);
|
||||
const app = document.getElementById('app');
|
||||
|
||||
/* ---------- PWA install prompt ---------- */
|
||||
|
||||
let deferredInstallPrompt = null;
|
||||
const isIOS = /iphone|ipad|ipod/i.test(navigator.userAgent);
|
||||
const isStandalone =
|
||||
window.matchMedia('(display-mode: standalone)').matches ||
|
||||
('standalone' in navigator && navigator.standalone === true);
|
||||
|
||||
window.addEventListener('beforeinstallprompt', (e) => {
|
||||
e.preventDefault();
|
||||
deferredInstallPrompt = e;
|
||||
const btn = document.querySelector('[data-install-wrap]');
|
||||
if (btn) btn.hidden = false;
|
||||
});
|
||||
|
||||
window.addEventListener('appinstalled', () => {
|
||||
deferredInstallPrompt = null;
|
||||
const wrap = document.querySelector('[data-install-wrap]');
|
||||
if (wrap) wrap.hidden = true;
|
||||
});
|
||||
|
||||
const state = {
|
||||
route: '/',
|
||||
user: '',
|
||||
|
|
@ -48,6 +71,7 @@ function handleRoute() {
|
|||
state.user = user;
|
||||
document.body.dataset.route = route;
|
||||
document.body.dataset.user = user || '';
|
||||
if (user) setMeta('last_route', route).catch(() => {});
|
||||
render();
|
||||
}
|
||||
|
||||
|
|
@ -85,6 +109,29 @@ function tpl(id) {
|
|||
function renderLanding() {
|
||||
app.innerHTML = '';
|
||||
app.appendChild(tpl('tpl-landing'));
|
||||
|
||||
if (isStandalone) return;
|
||||
|
||||
const wrap = app.querySelector('[data-install-wrap]');
|
||||
const btn = app.querySelector('[data-install-btn]');
|
||||
const hint = app.querySelector('[data-install-hint]');
|
||||
|
||||
if (deferredInstallPrompt) {
|
||||
wrap.hidden = false;
|
||||
btn.addEventListener('click', async () => {
|
||||
deferredInstallPrompt.prompt();
|
||||
const { outcome } = await deferredInstallPrompt.userChoice;
|
||||
if (outcome === 'accepted') {
|
||||
deferredInstallPrompt = null;
|
||||
wrap.hidden = true;
|
||||
}
|
||||
});
|
||||
} else if (isIOS) {
|
||||
wrap.hidden = false;
|
||||
btn.addEventListener('click', () => {
|
||||
hint.hidden = !hint.hidden;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
/* ---------- Render: route page ---------- */
|
||||
|
|
@ -343,6 +390,16 @@ function render() {
|
|||
async function boot() {
|
||||
await initTheme();
|
||||
await loadFromDexie();
|
||||
|
||||
// Restore last visited user route if opening at root
|
||||
const curPath = location.pathname;
|
||||
if (curPath === '/' || curPath === '') {
|
||||
const saved = await getMeta('last_route', null);
|
||||
if (saved && VALID_USERS.has(saved.slice(1))) {
|
||||
history.replaceState({}, '', saved);
|
||||
}
|
||||
}
|
||||
|
||||
handleRoute();
|
||||
|
||||
// Hydrate from server (non-blocking) and start sync
|
||||
|
|
|
|||
Loading…
Reference in a new issue