SpoolBuddyLayout.tsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. import { useState, useEffect, useRef, useCallback, useMemo } from 'react';
  2. import { Outlet, useNavigate, useLocation } from 'react-router-dom';
  3. import { useQuery, useQueries } from '@tanstack/react-query';
  4. import { useTranslation } from 'react-i18next';
  5. import { SpoolBuddyTopBar } from './SpoolBuddyTopBar';
  6. import { SpoolBuddyBottomNav } from './SpoolBuddyBottomNav';
  7. import { SpoolBuddyStatusBar } from './SpoolBuddyStatusBar';
  8. import { SpoolBuddyQuickMenu } from './SpoolBuddyQuickMenu';
  9. import { useSpoolBuddyState } from '../../hooks/useSpoolBuddyState';
  10. import { api, spoolbuddyApi, type Printer, type PrinterStatus } from '../../api/client';
  11. import { VirtualKeyboard } from '../VirtualKeyboard';
  12. export function SpoolBuddyLayout() {
  13. const [selectedPrinterId, setSelectedPrinterId] = useState<number | null>(null);
  14. const [alert, setAlert] = useState<{ type: 'warning' | 'error' | 'info'; message: string } | null>(null);
  15. const [blanked, setBlanked] = useState(false);
  16. const [displayBrightness, setDisplayBrightness] = useState(100);
  17. const [displayBlankTimeout, setDisplayBlankTimeout] = useState(0);
  18. const lastActivityRef = useRef(Date.now());
  19. const { i18n } = useTranslation();
  20. const navigate = useNavigate();
  21. const location = useLocation();
  22. const sbState = useSpoolBuddyState();
  23. // Sync language from backend settings (kiosk has its own browser with empty localStorage)
  24. const { data: appSettings } = useQuery({
  25. queryKey: ['settings'],
  26. queryFn: api.getSettings,
  27. });
  28. useEffect(() => {
  29. if (appSettings?.language && appSettings.language !== i18n.language) {
  30. i18n.changeLanguage(appSettings.language);
  31. }
  32. }, [appSettings?.language, i18n]);
  33. // Query device data to initialize display settings on any page
  34. const { data: devices = [] } = useQuery({
  35. queryKey: ['spoolbuddy-devices'],
  36. queryFn: () => spoolbuddyApi.getDevices(),
  37. refetchInterval: 30000,
  38. });
  39. const device = devices[0];
  40. const effectiveDeviceOnline = sbState.deviceOnline || Boolean(device?.online);
  41. const sbStateForUi = useMemo(
  42. () => ({ ...sbState, deviceOnline: effectiveDeviceOnline }),
  43. [sbState, effectiveDeviceOnline]
  44. );
  45. // Sync display settings from device on initial load
  46. const initializedRef = useRef(false);
  47. useEffect(() => {
  48. if (device && !initializedRef.current) {
  49. setDisplayBrightness(device.display_brightness);
  50. setDisplayBlankTimeout(device.display_blank_timeout);
  51. initializedRef.current = true;
  52. }
  53. }, [device]);
  54. // Force dark theme on mount, restore on unmount
  55. useEffect(() => {
  56. const root = document.documentElement;
  57. const hadDark = root.classList.contains('dark');
  58. root.classList.add('dark');
  59. return () => {
  60. if (!hadDark) root.classList.remove('dark');
  61. };
  62. }, []);
  63. // Auto-check for SpoolBuddy daemon updates
  64. const { data: updateCheck } = useQuery({
  65. queryKey: ['spoolbuddy-update-check', device?.device_id],
  66. queryFn: () => device ? spoolbuddyApi.checkDaemonUpdate(device.device_id) : Promise.resolve(null),
  67. enabled: !!device,
  68. refetchInterval: 5 * 60 * 1000, // re-check every 5 minutes
  69. staleTime: 0,
  70. });
  71. // Update alert based on device state and available updates.
  72. // Only clear alerts that the layout itself set (not alerts from child pages).
  73. const layoutAlertRef = useRef<string | null>(null);
  74. useEffect(() => {
  75. if (!effectiveDeviceOnline) {
  76. const msg = 'SpoolBuddy device disconnected';
  77. setAlert({ type: 'warning', message: msg });
  78. layoutAlertRef.current = msg;
  79. } else if (updateCheck?.update_available && updateCheck.latest_version) {
  80. const msg = `Update available: v${updateCheck.latest_version}`;
  81. setAlert({ type: 'info', message: msg });
  82. layoutAlertRef.current = msg;
  83. } else if (layoutAlertRef.current) {
  84. setAlert(null);
  85. layoutAlertRef.current = null;
  86. }
  87. }, [effectiveDeviceOnline, updateCheck?.update_available, updateCheck?.latest_version]);
  88. // Track user activity for screen blank
  89. const resetActivity = useCallback(() => {
  90. lastActivityRef.current = Date.now();
  91. setBlanked(false);
  92. }, []);
  93. useEffect(() => {
  94. window.addEventListener('pointerdown', resetActivity);
  95. window.addEventListener('keydown', resetActivity);
  96. return () => {
  97. window.removeEventListener('pointerdown', resetActivity);
  98. window.removeEventListener('keydown', resetActivity);
  99. };
  100. }, [resetActivity]);
  101. // Auto-navigate to dashboard when a NEW tag is detected (transition from no-tag to tag)
  102. const tagDetected = Boolean(sbState.matchedSpool || sbState.unknownTagUid);
  103. const prevTagDetected = useRef(false);
  104. useEffect(() => {
  105. if (tagDetected && !prevTagDetected.current) {
  106. resetActivity();
  107. if (location.pathname !== '/spoolbuddy') {
  108. navigate('/spoolbuddy');
  109. }
  110. }
  111. prevTagDetected.current = tagDetected;
  112. }, [tagDetected, location.pathname, navigate, resetActivity]);
  113. // Screen blank timer
  114. useEffect(() => {
  115. if (displayBlankTimeout <= 0) return;
  116. const interval = setInterval(() => {
  117. if (Date.now() - lastActivityRef.current >= displayBlankTimeout * 1000) {
  118. setBlanked(true);
  119. }
  120. }, 1000);
  121. return () => clearInterval(interval);
  122. }, [displayBlankTimeout]);
  123. // Online printers list for swipe-to-switch
  124. const { data: printers = [] } = useQuery({
  125. queryKey: ['printers'],
  126. queryFn: () => api.getPrinters(),
  127. });
  128. const statusQueries = useQueries({
  129. queries: printers.map((printer: Printer) => ({
  130. queryKey: ['printerStatus', printer.id],
  131. queryFn: () => api.getPrinterStatus(printer.id),
  132. refetchInterval: 10000,
  133. select: (data: PrinterStatus) => ({ connected: data?.connected }),
  134. })),
  135. });
  136. const onlinePrinters = useMemo(() => {
  137. return printers.filter((_: Printer, i: number) => statusQueries[i]?.data?.connected);
  138. }, [printers, statusQueries]);
  139. // Swipe left/right to cycle through online printers
  140. const touchStartRef = useRef<{ x: number; y: number } | null>(null);
  141. const swipeLockedRef = useRef(false);
  142. const SWIPE_THRESHOLD = 50;
  143. const rootRef = useRef<HTMLDivElement>(null);
  144. const handleTouchStart = useCallback((e: React.TouchEvent) => {
  145. touchStartRef.current = { x: e.touches[0].clientX, y: e.touches[0].clientY };
  146. swipeLockedRef.current = false;
  147. }, []);
  148. const handleTouchEnd = useCallback((e: React.TouchEvent) => {
  149. if (!touchStartRef.current) return;
  150. const dx = e.changedTouches[0].clientX - touchStartRef.current.x;
  151. const dy = e.changedTouches[0].clientY - touchStartRef.current.y;
  152. const startY = touchStartRef.current.y;
  153. touchStartRef.current = null;
  154. swipeLockedRef.current = false;
  155. // Vertical swipe: open/close quick menu
  156. if (Math.abs(dy) >= SWIPE_THRESHOLD && Math.abs(dy) > Math.abs(dx)) {
  157. if (dy > 0 && startY < 80) {
  158. // Swipe down from top area → open quick menu
  159. setQuickMenuOpen(true);
  160. }
  161. return;
  162. }
  163. // Horizontal swipe: cycle printers
  164. if (onlinePrinters.length < 2) return;
  165. if (Math.abs(dx) < SWIPE_THRESHOLD || Math.abs(dy) > Math.abs(dx)) return;
  166. const currentIdx = onlinePrinters.findIndex((p: Printer) => p.id === selectedPrinterId);
  167. const nextIdx = dx < 0
  168. ? (currentIdx + 1) % onlinePrinters.length // swipe left → next
  169. : (currentIdx - 1 + onlinePrinters.length) % onlinePrinters.length; // swipe right → prev
  170. setSelectedPrinterId(onlinePrinters[nextIdx].id);
  171. }, [onlinePrinters, selectedPrinterId, setSelectedPrinterId]);
  172. // Block browser back/forward swipe gesture with non-passive touchmove listener
  173. useEffect(() => {
  174. const el = rootRef.current;
  175. if (!el) return;
  176. const onTouchMove = (e: TouchEvent) => {
  177. if (!touchStartRef.current) return;
  178. const dx = Math.abs(e.touches[0].clientX - touchStartRef.current.x);
  179. const dy = Math.abs(e.touches[0].clientY - touchStartRef.current.y);
  180. // Once locked as horizontal, prevent default for the rest of this gesture
  181. if (swipeLockedRef.current) { e.preventDefault(); return; }
  182. if (dx > 10 && dx > dy) { swipeLockedRef.current = true; e.preventDefault(); }
  183. };
  184. el.addEventListener('touchmove', onTouchMove, { passive: false });
  185. return () => el.removeEventListener('touchmove', onTouchMove);
  186. }, []);
  187. // Track virtual keyboard visibility to hide bottom bars
  188. const [keyboardVisible, setKeyboardVisible] = useState(false);
  189. // Quick menu (swipe down to open)
  190. const [quickMenuOpen, setQuickMenuOpen] = useState(false);
  191. // CSS brightness filter (software dimming)
  192. const brightnessStyle = displayBrightness < 100
  193. ? { filter: `brightness(${displayBrightness / 100})` } as const
  194. : undefined;
  195. return (
  196. <>
  197. <div
  198. ref={rootRef}
  199. data-spoolbuddy-kiosk
  200. className="w-screen h-screen bg-bambu-dark text-white flex flex-col overflow-hidden"
  201. style={{ ...brightnessStyle, overscrollBehaviorX: 'none' }}
  202. onTouchStart={handleTouchStart}
  203. onTouchEnd={handleTouchEnd}
  204. >
  205. <SpoolBuddyTopBar
  206. selectedPrinterId={selectedPrinterId}
  207. onPrinterChange={setSelectedPrinterId}
  208. deviceOnline={effectiveDeviceOnline}
  209. />
  210. <main className="flex-1 overflow-y-auto">
  211. <Outlet context={{
  212. selectedPrinterId, setSelectedPrinterId, sbState: sbStateForUi, setAlert,
  213. displayBrightness, setDisplayBrightness,
  214. displayBlankTimeout, setDisplayBlankTimeout,
  215. }} />
  216. </main>
  217. {!keyboardVisible && <SpoolBuddyStatusBar alert={alert} />}
  218. {!keyboardVisible && <SpoolBuddyBottomNav />}
  219. <VirtualKeyboard onVisibilityChange={setKeyboardVisible} />
  220. </div>
  221. {/* Quick menu (swipe down from top) */}
  222. <SpoolBuddyQuickMenu
  223. isOpen={quickMenuOpen}
  224. onClose={() => setQuickMenuOpen(false)}
  225. deviceId={device?.device_id ?? null}
  226. deviceOnline={effectiveDeviceOnline}
  227. />
  228. {/* Screen blank overlay — touch to wake */}
  229. {blanked && (
  230. <div
  231. className="fixed inset-0 bg-black z-[9999]"
  232. onPointerDown={(e) => { e.stopPropagation(); resetActivity(); }}
  233. />
  234. )}
  235. </>
  236. );
  237. }
  238. // Hook for child pages to access shared context
  239. export interface SpoolBuddyOutletContext {
  240. selectedPrinterId: number | null;
  241. setSelectedPrinterId: (id: number) => void;
  242. sbState: ReturnType<typeof useSpoolBuddyState>;
  243. setAlert: (alert: { type: 'warning' | 'error' | 'info'; message: string } | null) => void;
  244. displayBrightness: number;
  245. setDisplayBrightness: (brightness: number) => void;
  246. displayBlankTimeout: number;
  247. setDisplayBlankTimeout: (timeout: number) => void;
  248. }