| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310 |
- import { useEffect, useMemo, useState } from 'react';
- import { useParams, useSearchParams } from 'react-router-dom';
- import { useQuery, useQueryClient } from '@tanstack/react-query';
- import { useTranslation } from 'react-i18next';
- import { Layers, Clock, Timer, Printer } from 'lucide-react';
- import { api } from '../api/client';
- import type { PrinterStatus } from '../api/client';
- import { formatDuration, formatETA, type TimeFormat } from '../utils/date';
- type TFunction = (key: string, options?: Record<string, unknown>) => string;
- type OverlaySize = 'small' | 'medium' | 'large';
- interface OverlayConfig {
- size: OverlaySize;
- fps: number;
- showCamera: boolean;
- showProgress: boolean;
- showLayers: boolean;
- showEta: boolean;
- showFilename: boolean;
- showStatus: boolean;
- showPrinter: boolean;
- }
- function parseConfig(params: URLSearchParams): OverlayConfig {
- const show = params.get('show')?.split(',') || ['progress', 'layers', 'eta', 'filename', 'status'];
- // Parse FPS (default 15, max 30, min 1)
- const fpsParam = parseInt(params.get('fps') || '15', 10);
- const fps = Math.min(Math.max(isNaN(fpsParam) ? 15 : fpsParam, 1), 30);
- // Parse camera toggle (default true, set camera=false to hide)
- const cameraParam = params.get('camera');
- const showCamera = cameraParam !== 'false' && cameraParam !== '0';
- return {
- size: (params.get('size') as OverlaySize) || 'medium',
- fps,
- showCamera,
- showProgress: show.includes('progress'),
- showLayers: show.includes('layers'),
- showEta: show.includes('eta'),
- showFilename: show.includes('filename'),
- showStatus: show.includes('status'),
- showPrinter: show.includes('printer'),
- };
- }
- function getStatusText(status: PrinterStatus, t: TFunction): string {
- if (status.stg_cur_name) return status.stg_cur_name;
- switch (status.state) {
- case 'RUNNING': return t('streamOverlay.status.printing');
- case 'PAUSE': return t('streamOverlay.status.paused');
- case 'FINISH': return t('streamOverlay.status.finished');
- case 'FAILED': return t('streamOverlay.status.failed');
- case 'IDLE': return t('streamOverlay.status.idle');
- default: return status.state || t('streamOverlay.status.unknown');
- }
- }
- function getSizeClasses(size: OverlaySize) {
- switch (size) {
- case 'small':
- return {
- container: 'p-3',
- text: 'text-sm',
- textLarge: 'text-lg',
- progressHeight: 'h-2',
- icon: 'w-3 h-3',
- gap: 'gap-2',
- logoHeight: 'h-12',
- };
- case 'large':
- return {
- container: 'p-6',
- text: 'text-xl',
- textLarge: 'text-3xl',
- progressHeight: 'h-4',
- icon: 'w-6 h-6',
- gap: 'gap-4',
- logoHeight: 'h-24',
- };
- case 'medium':
- default:
- return {
- container: 'p-4',
- text: 'text-base',
- textLarge: 'text-xl',
- progressHeight: 'h-3',
- icon: 'w-4 h-4',
- gap: 'gap-3',
- logoHeight: 'h-16',
- };
- }
- }
- export function StreamOverlayPage() {
- const { printerId } = useParams<{ printerId: string }>();
- const [searchParams] = useSearchParams();
- const { t } = useTranslation();
- const queryClient = useQueryClient();
- const id = parseInt(printerId || '0', 10);
- const [imageKey, setImageKey] = useState(Date.now());
- const config = useMemo(() => parseConfig(searchParams), [searchParams]);
- const sizes = getSizeClasses(config.size);
- // Fetch printer info
- const { data: printer } = useQuery({
- queryKey: ['printer', id],
- queryFn: () => api.getPrinter(id),
- enabled: id > 0,
- });
- // Fetch printer status with polling
- const { data: status } = useQuery({
- queryKey: ['printerStatus', id],
- queryFn: () => api.getPrinterStatus(id),
- enabled: id > 0,
- refetchInterval: 2000,
- });
- // Fetch settings info
- const { data: settings } = useQuery({
- queryKey: ['settings'],
- queryFn: api.getSettings,
- });
- const timeFormat: TimeFormat = settings?.time_format || 'system';
- // WebSocket for real-time updates
- useEffect(() => {
- if (!id) return;
- const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
- const wsUrl = `${protocol}//${window.location.host}/api/v1/ws`;
- const ws = new WebSocket(wsUrl);
- ws.onmessage = (event) => {
- try {
- const data = JSON.parse(event.data);
- if (data.type === 'printer_status' && data.printer_id === id) {
- queryClient.setQueryData(['printerStatus', id], data.status);
- }
- } catch {
- // Ignore parse errors
- }
- };
- ws.onerror = () => {
- // WebSocket error - polling will continue as fallback
- };
- return () => {
- ws.close();
- };
- }, [id, queryClient]);
- // Update document title
- useEffect(() => {
- document.title = printer ? `${printer.name} - ${t('streamOverlay.title')}` : t('streamOverlay.title');
- return () => {
- document.title = 'Bambuddy';
- };
- }, [printer, t]);
- // Refresh stream on error
- const handleStreamError = () => {
- setTimeout(() => {
- setImageKey(Date.now());
- }, 3000);
- };
- if (!id) {
- return (
- <div className="min-h-screen bg-black flex items-center justify-center">
- <p className="text-white">{t('streamOverlay.invalidPrinterId')}</p>
- </div>
- );
- }
- if (!status) {
- return (
- <div className="min-h-screen bg-black flex items-center justify-center">
- <p className="text-gray-400">{t('common.loading')}</p>
- </div>
- );
- }
- const isPrinting = status.state === 'RUNNING' || status.state === 'PAUSE';
- const progress = status.progress || 0;
- const streamUrl = `/api/v1/printers/${id}/camera/stream?fps=${config.fps}&t=${imageKey}`;
- return (
- <div className="min-h-screen bg-black relative overflow-hidden">
- {/* Camera feed - fullscreen background (optional) */}
- {config.showCamera && (
- <img
- key={imageKey}
- src={streamUrl}
- alt={t('streamOverlay.cameraStream')}
- className="absolute inset-0 w-full h-full object-contain"
- onError={handleStreamError}
- />
- )}
- {/* Bambuddy logo - top right */}
- <a
- href="https://github.com/maziggy/bambuddy"
- target="_blank"
- rel="noopener noreferrer"
- className="absolute top-4 right-4 z-10"
- >
- <img
- src="/img/bambuddy_logo_dark_transparent.png"
- alt="Bambuddy"
- className={`${sizes.logoHeight} object-contain drop-shadow-lg hover:scale-105 transition-transform`}
- />
- </a>
- {/* Status overlay - bottom */}
- <div className="absolute bottom-0 left-0 right-0 z-10 bg-gradient-to-t from-black/80 via-black/60 to-transparent">
- <div className={`${sizes.container}`}>
- {/* Printer name */}
- {config.showPrinter && printer && (
- <div className={`flex items-center ${sizes.gap} mb-2`}>
- <Printer className={`${sizes.icon} text-white/70`} />
- <span className={`${sizes.text} text-white font-medium`}>{printer.name}</span>
- </div>
- )}
- {/* Filename */}
- {config.showFilename && status.current_print && (
- <div className={`${sizes.textLarge} text-white font-semibold mb-2 truncate drop-shadow-md`}>
- {status.current_print.replace(/\.gcode\.3mf$|\.3mf$|\.gcode$/i, '')}
- </div>
- )}
- {/* Status text */}
- {config.showStatus && (
- <div className={`${sizes.text} text-white/70 mb-2`}>
- {getStatusText(status, t)}
- </div>
- )}
- {/* Progress bar */}
- {config.showProgress && isPrinting && (
- <div className="mb-3">
- <div className={`flex items-center justify-between mb-1 ${sizes.text}`}>
- <span className="text-white/70">{t('streamOverlay.progress')}</span>
- <span className="text-white font-bold">{Math.round(progress)}%</span>
- </div>
- <div className={`w-full bg-white/20 rounded-full ${sizes.progressHeight}`}>
- <div
- className={`bg-bambu-green ${sizes.progressHeight} rounded-full transition-all duration-500`}
- style={{ width: `${progress}%` }}
- />
- </div>
- </div>
- )}
- {/* Stats row */}
- {isPrinting && (config.showLayers || config.showEta) && (
- <div className={`flex items-center ${sizes.gap} flex-wrap`}>
- {/* Layers */}
- {config.showLayers && status.layer_num != null && status.total_layers != null && status.total_layers > 0 && (
- <div className={`flex items-center ${sizes.gap} text-white/70`}>
- <Layers className={sizes.icon} />
- <span className={sizes.text}>
- <span className="text-white">{status.layer_num}</span>
- <span className="mx-1">/</span>
- <span>{status.total_layers}</span>
- </span>
- </div>
- )}
- {/* Remaining time */}
- {config.showEta && status.remaining_time != null && status.remaining_time > 0 && (
- <>
- <div className={`flex items-center ${sizes.gap} text-white/70`}>
- <Timer className={sizes.icon} />
- <span className={`${sizes.text} text-white`}>
- {formatDuration(status.remaining_time * 60)}
- </span>
- </div>
- <div className={`flex items-center ${sizes.gap} text-white/70`}>
- <Clock className={sizes.icon} />
- <span className={`${sizes.text} text-white`}>
- {t('streamOverlay.eta')} {formatETA(status.remaining_time, timeFormat, t)}
- </span>
- </div>
- </>
- )}
- </div>
- )}
- {/* Idle state */}
- {!isPrinting && (
- <div className={`${sizes.text} text-white/70 py-2`}>
- {status.connected ? t('streamOverlay.printerIdle') : t('streamOverlay.printerOffline')}
- </div>
- )}
- </div>
- </div>
- </div>
- );
- }
|