mirror of
https://github.com/ruvnet/RuView
synced 2026-06-09 10:13:17 +00:00
49fb2ca9f4
* feat(ui): add keyboard shortcuts, perf monitor, toast system, theme toggle, and WCAG accessibility - Keyboard shortcuts overlay (press ? for help, 1-8 for tabs, T for theme, P for perf) - Real-time performance monitor with FPS, memory, latency sparklines (draggable) - Enhanced toast notification system with stacking, auto-dismiss, progress bars - Dark/light theme toggle with localStorage persistence and system preference detection - WCAG accessibility: skip-to-content link, ARIA roles/attributes on tabs and panels, arrow key navigation in tab bar, focus-visible outlines - ESLint config for UI directory with security and quality rules * feat(ui): add command palette, activity log, data export, fullscreen mode, connection status - Command palette (Ctrl+K / Cmd+K) with fuzzy search across tabs and actions - Activity log panel (L key) with real-time console interception, filters, resizable - Data export utility (E key) for sensor data as JSON/CSV with dialog - Fullscreen mode (F key / F11) for visualization tabs with exit button - Connection status widget in header showing WebSocket state and reconnect * feat(ui): add mobile hamburger nav, PWA support, and 40 unit tests - Mobile hamburger navigation: slide-out drawer replacing tab bar on <768px, swipe-to-close, animated hamburger icon, auto-sync with tab manager - PWA manifest + service worker: installable dashboard, offline shell caching (cache-first for static, network-first for API), auto-cleanup of old caches - 40 unit tests for ToastManager, ThemeToggle, KeyboardShortcuts, PerfMonitor, TabManager - browser-based test runner at ui/tests/unit-tests.html - PWA meta tags: theme-color, apple-mobile-web-app-capable, manifest link - Icon generator page for creating PWA icons (ui/icons/generate.html) * feat(ui): add URL routing, onboarding tour, idle detection, notification center - Hash router: tabs are bookmarkable/shareable via URL (#demo, #sensing, etc.), syncs with TabManager, supports browser back/forward navigation - Onboarding tour: interactive 6-step first-run walkthrough with spotlight highlighting, step indicators, skip/back/next controls, localStorage persistence - Idle detection: pauses health polling and reduces CSS animations after 3 min of inactivity, resumes on user interaction, integrates with Page Visibility API - Notification center: bell icon in header with unread badge, event history panel with mark-read/clear, persists across page views via sessionStorage * feat(ui): add i18n (EN/PL), screenshot tool, settings panel, reduced motion, uptime clock - i18n: English/Polish translations with auto-detection, language selector in header, data-i18n attributes on dashboard elements, localStorage persistence - Screenshot tool (S key): captures active tab to clipboard or downloads PNG, flash effect, canvas rendering with watermark, fallback for tainted canvases - Quick settings panel (gear icon): reduced motion toggle, high contrast mode, compact layout mode, health polling toggle, clear data, reset onboarding - Uptime clock: current time + session duration in header - prefers-reduced-motion: system-level and manual toggle, disables all animations and transitions for vestibular accessibility - High contrast mode: WCAG AAA compliant colors for both light and dark themes - Compact mode: condensed layout for dense information display
192 lines
7.0 KiB
JavaScript
192 lines
7.0 KiB
JavaScript
// Quick Settings Panel - Centralized configuration for all UI features
|
|
// Accessible via gear icon in header
|
|
|
|
export class QuickSettings {
|
|
constructor(app) {
|
|
this.app = app;
|
|
this.button = null;
|
|
this.panel = null;
|
|
this.isOpen = false;
|
|
}
|
|
|
|
init() {
|
|
this.createButton();
|
|
this.createPanel();
|
|
}
|
|
|
|
createButton() {
|
|
this.button = document.createElement('button');
|
|
this.button.className = 'settings-gear';
|
|
this.button.setAttribute('aria-label', 'Settings');
|
|
this.button.setAttribute('title', 'Quick settings');
|
|
this.button.innerHTML = `<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="12" cy="12" r="3"/><path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06A1.65 1.65 0 0 0 4.68 15a1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06A1.65 1.65 0 0 0 9 4.68a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06A1.65 1.65 0 0 0 19.4 9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"/></svg>`;
|
|
|
|
this.button.addEventListener('click', () => this.toggle());
|
|
|
|
const headerInfo = document.querySelector('.header-info');
|
|
if (headerInfo) headerInfo.appendChild(this.button);
|
|
}
|
|
|
|
createPanel() {
|
|
this.panel = document.createElement('div');
|
|
this.panel.className = 'quick-settings-panel';
|
|
this.panel.setAttribute('role', 'dialog');
|
|
this.panel.setAttribute('aria-label', 'Quick settings');
|
|
|
|
this.panel.innerHTML = `
|
|
<div class="qs-header">
|
|
<h3>Settings</h3>
|
|
<button class="qs-close" aria-label="Close">×</button>
|
|
</div>
|
|
<div class="qs-body">
|
|
<div class="qs-section">
|
|
<div class="qs-section-title">Display</div>
|
|
<label class="qs-toggle">
|
|
<span>Reduced motion</span>
|
|
<input type="checkbox" id="qs-reduced-motion" ${this.prefersReducedMotion() ? 'checked' : ''}>
|
|
<span class="qs-switch"></span>
|
|
</label>
|
|
<label class="qs-toggle">
|
|
<span>High contrast</span>
|
|
<input type="checkbox" id="qs-high-contrast">
|
|
<span class="qs-switch"></span>
|
|
</label>
|
|
<label class="qs-toggle">
|
|
<span>Compact mode</span>
|
|
<input type="checkbox" id="qs-compact" ${this.getSetting('compact') ? 'checked' : ''}>
|
|
<span class="qs-switch"></span>
|
|
</label>
|
|
</div>
|
|
<div class="qs-section">
|
|
<div class="qs-section-title">Monitoring</div>
|
|
<label class="qs-toggle">
|
|
<span>Health polling</span>
|
|
<input type="checkbox" id="qs-health-polling" checked>
|
|
<span class="qs-switch"></span>
|
|
</label>
|
|
<label class="qs-toggle">
|
|
<span>Auto-reconnect</span>
|
|
<input type="checkbox" id="qs-auto-reconnect" checked>
|
|
<span class="qs-switch"></span>
|
|
</label>
|
|
</div>
|
|
<div class="qs-section">
|
|
<div class="qs-section-title">Data</div>
|
|
<div class="qs-row">
|
|
<span>Clear local data</span>
|
|
<button class="qs-btn-danger" id="qs-clear-data">Clear</button>
|
|
</div>
|
|
<div class="qs-row">
|
|
<span>Reset onboarding</span>
|
|
<button class="qs-btn" id="qs-reset-tour">Reset</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
`;
|
|
|
|
// Bind events
|
|
this.panel.querySelector('.qs-close').addEventListener('click', () => this.close());
|
|
|
|
this.panel.querySelector('#qs-reduced-motion').addEventListener('change', (e) => {
|
|
document.body.classList.toggle('reduced-motion', e.target.checked);
|
|
this.saveSetting('reduced-motion', e.target.checked);
|
|
});
|
|
|
|
this.panel.querySelector('#qs-high-contrast').addEventListener('change', (e) => {
|
|
document.body.classList.toggle('high-contrast', e.target.checked);
|
|
this.saveSetting('high-contrast', e.target.checked);
|
|
});
|
|
|
|
this.panel.querySelector('#qs-compact').addEventListener('change', (e) => {
|
|
document.body.classList.toggle('compact-mode', e.target.checked);
|
|
this.saveSetting('compact', e.target.checked);
|
|
});
|
|
|
|
this.panel.querySelector('#qs-health-polling').addEventListener('change', (e) => {
|
|
const healthService = this.app?.components?.dashboard?.healthSubscription;
|
|
if (e.target.checked) {
|
|
// Resume would need import - just dispatch event
|
|
document.dispatchEvent(new CustomEvent('health-polling-toggle', { detail: true }));
|
|
} else {
|
|
document.dispatchEvent(new CustomEvent('health-polling-toggle', { detail: false }));
|
|
}
|
|
});
|
|
|
|
this.panel.querySelector('#qs-clear-data').addEventListener('click', () => {
|
|
try {
|
|
localStorage.clear();
|
|
sessionStorage.clear();
|
|
} catch { /* noop */ }
|
|
this.close();
|
|
window.location.reload();
|
|
});
|
|
|
|
this.panel.querySelector('#qs-reset-tour').addEventListener('click', () => {
|
|
try { localStorage.removeItem('ruview-onboarding-done'); } catch { /* noop */ }
|
|
this.close();
|
|
document.dispatchEvent(new CustomEvent('start-onboarding'));
|
|
});
|
|
|
|
document.body.appendChild(this.panel);
|
|
|
|
// Close on outside click
|
|
document.addEventListener('click', (e) => {
|
|
if (this.isOpen && !this.panel.contains(e.target) && !this.button.contains(e.target)) {
|
|
this.close();
|
|
}
|
|
});
|
|
|
|
// Apply saved settings on init
|
|
this.applySavedSettings();
|
|
}
|
|
|
|
applySavedSettings() {
|
|
if (this.getSetting('reduced-motion') || this.prefersReducedMotion()) {
|
|
document.body.classList.add('reduced-motion');
|
|
const cb = this.panel.querySelector('#qs-reduced-motion');
|
|
if (cb) cb.checked = true;
|
|
}
|
|
if (this.getSetting('high-contrast')) {
|
|
document.body.classList.add('high-contrast');
|
|
const cb = this.panel.querySelector('#qs-high-contrast');
|
|
if (cb) cb.checked = true;
|
|
}
|
|
if (this.getSetting('compact')) {
|
|
document.body.classList.add('compact-mode');
|
|
}
|
|
}
|
|
|
|
prefersReducedMotion() {
|
|
return window.matchMedia('(prefers-reduced-motion: reduce)').matches;
|
|
}
|
|
|
|
toggle() {
|
|
this.isOpen ? this.close() : this.open();
|
|
}
|
|
|
|
open() {
|
|
this.isOpen = true;
|
|
this.panel.classList.add('open');
|
|
}
|
|
|
|
close() {
|
|
this.isOpen = false;
|
|
this.panel.classList.remove('open');
|
|
}
|
|
|
|
getSetting(key) {
|
|
try { return JSON.parse(localStorage.getItem(`ruview-setting-${key}`)); }
|
|
catch { return null; }
|
|
}
|
|
|
|
saveSetting(key, value) {
|
|
try { localStorage.setItem(`ruview-setting-${key}`, JSON.stringify(value)); }
|
|
catch { /* noop */ }
|
|
}
|
|
|
|
dispose() {
|
|
this.button?.remove();
|
|
this.panel?.remove();
|
|
}
|
|
}
|