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
169 lines
4.9 KiB
JavaScript
169 lines
4.9 KiB
JavaScript
// Keyboard Shortcuts System
|
|
// Press '?' to show help overlay, number keys to switch tabs, etc.
|
|
|
|
export class KeyboardShortcuts {
|
|
constructor(app) {
|
|
this.app = app;
|
|
this.shortcuts = new Map();
|
|
this.helpVisible = false;
|
|
this.enabled = true;
|
|
this.overlay = null;
|
|
this.registerDefaults();
|
|
}
|
|
|
|
registerDefaults() {
|
|
this.register('?', 'Show keyboard shortcuts', () => this.toggleHelp());
|
|
this.register('Escape', 'Close overlay / dialog', () => this.closeAll());
|
|
this.register('1', 'Switch to Dashboard tab', () => this.switchTab('dashboard'));
|
|
this.register('2', 'Switch to Hardware tab', () => this.switchTab('hardware'));
|
|
this.register('3', 'Switch to Live Demo tab', () => this.switchTab('demo'));
|
|
this.register('4', 'Switch to Architecture tab', () => this.switchTab('architecture'));
|
|
this.register('5', 'Switch to Performance tab', () => this.switchTab('performance'));
|
|
this.register('6', 'Switch to Applications tab', () => this.switchTab('applications'));
|
|
this.register('7', 'Switch to Sensing tab', () => this.switchTab('sensing'));
|
|
this.register('8', 'Switch to Training tab', () => this.switchTab('training'));
|
|
this.register('p', 'Toggle performance monitor', () => this.togglePerfMonitor());
|
|
this.register('t', 'Toggle dark/light theme', () => this.toggleTheme());
|
|
}
|
|
|
|
register(key, description, handler) {
|
|
this.shortcuts.set(key, { description, handler });
|
|
}
|
|
|
|
init() {
|
|
document.addEventListener('keydown', (e) => this.handleKeydown(e));
|
|
this.createOverlay();
|
|
}
|
|
|
|
handleKeydown(e) {
|
|
if (!this.enabled) return;
|
|
|
|
// Ignore when typing in inputs
|
|
const tag = e.target.tagName.toLowerCase();
|
|
if (tag === 'input' || tag === 'textarea' || tag === 'select' || e.target.isContentEditable) {
|
|
if (e.key === 'Escape') {
|
|
e.target.blur();
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Ignore modified keys (except shift for '?')
|
|
if (e.ctrlKey || e.altKey || e.metaKey) return;
|
|
|
|
const shortcut = this.shortcuts.get(e.key);
|
|
if (shortcut) {
|
|
e.preventDefault();
|
|
shortcut.handler();
|
|
}
|
|
}
|
|
|
|
switchTab(tabId) {
|
|
const tabManager = this.app?.getComponent?.('tabManager');
|
|
if (tabManager) {
|
|
tabManager.switchToTab(tabId);
|
|
}
|
|
}
|
|
|
|
togglePerfMonitor() {
|
|
const event = new CustomEvent('toggle-perf-monitor');
|
|
document.dispatchEvent(event);
|
|
}
|
|
|
|
toggleTheme() {
|
|
const event = new CustomEvent('toggle-theme');
|
|
document.dispatchEvent(event);
|
|
}
|
|
|
|
closeAll() {
|
|
if (this.helpVisible) {
|
|
this.hideHelp();
|
|
}
|
|
}
|
|
|
|
createOverlay() {
|
|
this.overlay = document.createElement('div');
|
|
this.overlay.className = 'shortcuts-overlay';
|
|
this.overlay.setAttribute('role', 'dialog');
|
|
this.overlay.setAttribute('aria-label', 'Keyboard shortcuts');
|
|
this.overlay.setAttribute('aria-modal', 'true');
|
|
this.overlay.innerHTML = this.buildHelpHTML();
|
|
this.overlay.addEventListener('click', (e) => {
|
|
if (e.target === this.overlay) this.hideHelp();
|
|
});
|
|
document.body.appendChild(this.overlay);
|
|
}
|
|
|
|
buildHelpHTML() {
|
|
const groups = [
|
|
{
|
|
title: 'Navigation',
|
|
items: Array.from(this.shortcuts.entries())
|
|
.filter(([key]) => /^[1-8]$/.test(key))
|
|
},
|
|
{
|
|
title: 'Actions',
|
|
items: Array.from(this.shortcuts.entries())
|
|
.filter(([key]) => /^[a-z]$/.test(key))
|
|
},
|
|
{
|
|
title: 'General',
|
|
items: Array.from(this.shortcuts.entries())
|
|
.filter(([key]) => !/^[1-8a-z]$/.test(key))
|
|
}
|
|
];
|
|
|
|
return `
|
|
<div class="shortcuts-panel">
|
|
<div class="shortcuts-header">
|
|
<h2>Keyboard Shortcuts</h2>
|
|
<button class="shortcuts-close" aria-label="Close">×</button>
|
|
</div>
|
|
<div class="shortcuts-body">
|
|
${groups.map(group => `
|
|
<div class="shortcuts-group">
|
|
<h3>${group.title}</h3>
|
|
${group.items.map(([key, { description }]) => `
|
|
<div class="shortcut-row">
|
|
<kbd>${this.formatKey(key)}</kbd>
|
|
<span>${description}</span>
|
|
</div>
|
|
`).join('')}
|
|
</div>
|
|
`).join('')}
|
|
</div>
|
|
</div>
|
|
`;
|
|
}
|
|
|
|
formatKey(key) {
|
|
const map = { Escape: 'Esc', '?': '?' };
|
|
return map[key] || key.toUpperCase();
|
|
}
|
|
|
|
toggleHelp() {
|
|
this.helpVisible ? this.hideHelp() : this.showHelp();
|
|
}
|
|
|
|
showHelp() {
|
|
this.overlay.classList.add('visible');
|
|
this.helpVisible = true;
|
|
// Focus close button
|
|
const closeBtn = this.overlay.querySelector('.shortcuts-close');
|
|
if (closeBtn) {
|
|
closeBtn.onclick = () => this.hideHelp();
|
|
closeBtn.focus();
|
|
}
|
|
}
|
|
|
|
hideHelp() {
|
|
this.overlay.classList.remove('visible');
|
|
this.helpVisible = false;
|
|
}
|
|
|
|
dispose() {
|
|
if (this.overlay?.parentNode) {
|
|
this.overlay.parentNode.removeChild(this.overlay);
|
|
}
|
|
}
|
|
}
|