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
161 lines
5.2 KiB
JavaScript
161 lines
5.2 KiB
JavaScript
// Screenshot Tool - Capture current tab view as PNG
|
|
// Uses html2canvas-like approach with native Canvas API
|
|
|
|
import { toastManager } from './toast.js';
|
|
|
|
export class ScreenshotTool {
|
|
constructor() {
|
|
this.capturing = false;
|
|
}
|
|
|
|
init() {
|
|
document.addEventListener('take-screenshot', () => this.capture());
|
|
}
|
|
|
|
async capture() {
|
|
if (this.capturing) return;
|
|
this.capturing = true;
|
|
|
|
const activeTab = document.querySelector('.tab-content.active');
|
|
if (!activeTab) {
|
|
toastManager.warning('No active tab to capture');
|
|
this.capturing = false;
|
|
return;
|
|
}
|
|
|
|
try {
|
|
// Flash effect
|
|
this.flashEffect();
|
|
|
|
// Try native ClipboardItem API first (modern browsers)
|
|
if (typeof ClipboardItem !== 'undefined') {
|
|
await this.captureToClipboard(activeTab);
|
|
toastManager.success('Screenshot copied to clipboard', { duration: 3000 });
|
|
} else {
|
|
// Fallback: download as file
|
|
await this.captureToFile(activeTab);
|
|
toastManager.success('Screenshot saved as file', { duration: 3000 });
|
|
}
|
|
} catch (err) {
|
|
console.error('Screenshot failed:', err);
|
|
// Fallback: capture visible canvases + basic layout
|
|
try {
|
|
await this.captureCanvasFallback(activeTab);
|
|
toastManager.success('Screenshot saved (canvas only)', { duration: 3000 });
|
|
} catch {
|
|
toastManager.error('Screenshot failed. Try using browser\'s built-in screenshot tool.');
|
|
}
|
|
}
|
|
|
|
this.capturing = false;
|
|
}
|
|
|
|
async captureToClipboard(element) {
|
|
const canvas = await this.renderToCanvas(element);
|
|
const blob = await new Promise(resolve => canvas.toBlob(resolve, 'image/png'));
|
|
await navigator.clipboard.write([
|
|
new ClipboardItem({ 'image/png': blob })
|
|
]);
|
|
}
|
|
|
|
async captureToFile(element) {
|
|
const canvas = await this.renderToCanvas(element);
|
|
const dataUrl = canvas.toDataURL('image/png');
|
|
const link = document.createElement('a');
|
|
link.href = dataUrl;
|
|
link.download = `ruview-screenshot-${this.timestamp()}.png`;
|
|
link.click();
|
|
}
|
|
|
|
async captureCanvasFallback(element) {
|
|
// Find any canvas elements and merge them
|
|
const canvases = element.querySelectorAll('canvas');
|
|
if (canvases.length === 0) throw new Error('No canvas elements found');
|
|
|
|
const firstCanvas = canvases[0];
|
|
const mergedCanvas = document.createElement('canvas');
|
|
mergedCanvas.width = firstCanvas.width || 800;
|
|
mergedCanvas.height = firstCanvas.height || 600;
|
|
const ctx = mergedCanvas.getContext('2d');
|
|
|
|
// Dark background
|
|
ctx.fillStyle = '#1f2121';
|
|
ctx.fillRect(0, 0, mergedCanvas.width, mergedCanvas.height);
|
|
|
|
canvases.forEach(c => {
|
|
try { ctx.drawImage(c, 0, 0); } catch { /* tainted canvas */ }
|
|
});
|
|
|
|
// Add timestamp watermark
|
|
ctx.fillStyle = 'rgba(255,255,255,0.3)';
|
|
ctx.font = '12px monospace';
|
|
ctx.fillText(`RuView - ${new Date().toLocaleString()}`, 10, mergedCanvas.height - 10);
|
|
|
|
const dataUrl = mergedCanvas.toDataURL('image/png');
|
|
const link = document.createElement('a');
|
|
link.href = dataUrl;
|
|
link.download = `ruview-screenshot-${this.timestamp()}.png`;
|
|
link.click();
|
|
}
|
|
|
|
async renderToCanvas(element) {
|
|
// Simple DOM-to-canvas renderer for basic content
|
|
const rect = element.getBoundingClientRect();
|
|
const canvas = document.createElement('canvas');
|
|
const scale = window.devicePixelRatio || 1;
|
|
canvas.width = rect.width * scale;
|
|
canvas.height = rect.height * scale;
|
|
const ctx = canvas.getContext('2d');
|
|
ctx.scale(scale, scale);
|
|
|
|
// Render background
|
|
const styles = getComputedStyle(element);
|
|
ctx.fillStyle = styles.backgroundColor || '#1f2121';
|
|
ctx.fillRect(0, 0, rect.width, rect.height);
|
|
|
|
// Render existing canvases
|
|
const canvases = element.querySelectorAll('canvas');
|
|
canvases.forEach(c => {
|
|
const cRect = c.getBoundingClientRect();
|
|
const x = cRect.left - rect.left;
|
|
const y = cRect.top - rect.top;
|
|
try { ctx.drawImage(c, x, y, cRect.width, cRect.height); } catch { /* tainted */ }
|
|
});
|
|
|
|
// Render text content
|
|
ctx.fillStyle = styles.color || '#e0e0e0';
|
|
ctx.font = `14px ${styles.fontFamily || 'sans-serif'}`;
|
|
let textY = 30;
|
|
element.querySelectorAll('h2, h3, .stat-value, .metric-label').forEach(el => {
|
|
const text = el.textContent.trim();
|
|
if (text && textY < rect.height - 20) {
|
|
const elStyles = getComputedStyle(el);
|
|
ctx.font = `${elStyles.fontWeight} ${elStyles.fontSize} ${styles.fontFamily || 'sans-serif'}`;
|
|
ctx.fillStyle = elStyles.color;
|
|
ctx.fillText(text, 20, textY);
|
|
textY += parseInt(elStyles.fontSize) + 8;
|
|
}
|
|
});
|
|
|
|
// Watermark
|
|
ctx.fillStyle = 'rgba(255,255,255,0.15)';
|
|
ctx.font = '11px monospace';
|
|
ctx.fillText(`RuView - ${new Date().toLocaleString()}`, 10, rect.height - 10);
|
|
|
|
return canvas;
|
|
}
|
|
|
|
flashEffect() {
|
|
const flash = document.createElement('div');
|
|
flash.className = 'screenshot-flash';
|
|
document.body.appendChild(flash);
|
|
flash.addEventListener('animationend', () => flash.remove());
|
|
}
|
|
|
|
timestamp() {
|
|
return new Date().toISOString().replace(/[:.]/g, '-').slice(0, 19);
|
|
}
|
|
|
|
dispose() {}
|
|
}
|