CameraPage.tsx 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. import { useState, useEffect, useRef } from 'react';
  2. import { useParams } from 'react-router-dom';
  3. import { useQuery } from '@tanstack/react-query';
  4. import { RefreshCw, AlertTriangle, Camera, Maximize, Minimize } from 'lucide-react';
  5. import { api } from '../api/client';
  6. export function CameraPage() {
  7. const { printerId } = useParams<{ printerId: string }>();
  8. const id = parseInt(printerId || '0', 10);
  9. const [streamMode, setStreamMode] = useState<'stream' | 'snapshot'>('stream');
  10. const [streamError, setStreamError] = useState(false);
  11. const [streamLoading, setStreamLoading] = useState(true);
  12. const [imageKey, setImageKey] = useState(Date.now());
  13. const [transitioning, setTransitioning] = useState(false);
  14. const [isFullscreen, setIsFullscreen] = useState(false);
  15. const imgRef = useRef<HTMLImageElement>(null);
  16. const containerRef = useRef<HTMLDivElement>(null);
  17. // Fetch printer info for the title
  18. const { data: printer } = useQuery({
  19. queryKey: ['printer', id],
  20. queryFn: () => api.getPrinter(id),
  21. enabled: id > 0,
  22. });
  23. // Update document title
  24. useEffect(() => {
  25. if (printer) {
  26. document.title = `${printer.name} - Camera`;
  27. }
  28. return () => {
  29. document.title = 'Bambuddy';
  30. };
  31. }, [printer]);
  32. // Cleanup on unmount
  33. useEffect(() => {
  34. return () => {
  35. if (imgRef.current) {
  36. imgRef.current.src = '';
  37. }
  38. };
  39. }, []);
  40. // Auto-hide loading after timeout
  41. useEffect(() => {
  42. if (streamLoading && !transitioning) {
  43. const timeout = streamMode === 'stream' ? 3000 : 20000;
  44. const timer = setTimeout(() => {
  45. setStreamLoading(false);
  46. }, timeout);
  47. return () => clearTimeout(timer);
  48. }
  49. }, [streamMode, streamLoading, imageKey, transitioning]);
  50. // Fullscreen change listener
  51. useEffect(() => {
  52. const handleFullscreenChange = () => {
  53. setIsFullscreen(!!document.fullscreenElement);
  54. };
  55. document.addEventListener('fullscreenchange', handleFullscreenChange);
  56. return () => document.removeEventListener('fullscreenchange', handleFullscreenChange);
  57. }, []);
  58. const handleStreamError = () => {
  59. setStreamError(true);
  60. setStreamLoading(false);
  61. };
  62. const handleStreamLoad = () => {
  63. setStreamLoading(false);
  64. setStreamError(false);
  65. };
  66. const switchToMode = (newMode: 'stream' | 'snapshot') => {
  67. if (streamMode === newMode || transitioning) return;
  68. setTransitioning(true);
  69. setStreamLoading(true);
  70. setStreamError(false);
  71. if (imgRef.current) {
  72. imgRef.current.src = '';
  73. }
  74. setTimeout(() => {
  75. setStreamMode(newMode);
  76. setImageKey(Date.now());
  77. setTransitioning(false);
  78. }, 100);
  79. };
  80. const refresh = () => {
  81. if (transitioning) return;
  82. setTransitioning(true);
  83. setStreamLoading(true);
  84. setStreamError(false);
  85. if (imgRef.current) {
  86. imgRef.current.src = '';
  87. }
  88. setTimeout(() => {
  89. setImageKey(Date.now());
  90. setTransitioning(false);
  91. }, 100);
  92. };
  93. const toggleFullscreen = () => {
  94. if (!containerRef.current) return;
  95. if (document.fullscreenElement) {
  96. document.exitFullscreen();
  97. } else {
  98. containerRef.current.requestFullscreen();
  99. }
  100. };
  101. const currentUrl = transitioning
  102. ? ''
  103. : streamMode === 'stream'
  104. ? `/api/v1/printers/${id}/camera/stream?fps=10&t=${imageKey}`
  105. : `/api/v1/printers/${id}/camera/snapshot?t=${imageKey}`;
  106. const isDisabled = streamLoading || transitioning;
  107. if (!id) {
  108. return (
  109. <div className="min-h-screen bg-black flex items-center justify-center">
  110. <p className="text-white">Invalid printer ID</p>
  111. </div>
  112. );
  113. }
  114. return (
  115. <div ref={containerRef} className="min-h-screen bg-black flex flex-col">
  116. {/* Header */}
  117. <div className="flex items-center justify-between px-4 py-2 bg-bambu-dark-secondary border-b border-bambu-dark-tertiary">
  118. <h1 className="text-sm font-medium text-white flex items-center gap-2">
  119. <Camera className="w-4 h-4" />
  120. {printer?.name || `Printer ${id}`}
  121. </h1>
  122. <div className="flex items-center gap-2">
  123. {/* Mode toggle */}
  124. <div className="flex bg-bambu-dark rounded p-0.5">
  125. <button
  126. onClick={() => switchToMode('stream')}
  127. disabled={isDisabled}
  128. className={`px-3 py-1 text-xs rounded transition-colors ${
  129. streamMode === 'stream'
  130. ? 'bg-bambu-green text-white'
  131. : 'text-bambu-gray hover:text-white disabled:opacity-50'
  132. }`}
  133. >
  134. Live
  135. </button>
  136. <button
  137. onClick={() => switchToMode('snapshot')}
  138. disabled={isDisabled}
  139. className={`px-3 py-1 text-xs rounded transition-colors ${
  140. streamMode === 'snapshot'
  141. ? 'bg-bambu-green text-white'
  142. : 'text-bambu-gray hover:text-white disabled:opacity-50'
  143. }`}
  144. >
  145. Snapshot
  146. </button>
  147. </div>
  148. <button
  149. onClick={refresh}
  150. disabled={isDisabled}
  151. className="p-1.5 hover:bg-bambu-dark-tertiary rounded disabled:opacity-50"
  152. title={streamMode === 'stream' ? 'Restart stream' : 'Refresh snapshot'}
  153. >
  154. <RefreshCw className={`w-4 h-4 text-bambu-gray ${isDisabled ? 'animate-spin' : ''}`} />
  155. </button>
  156. <button
  157. onClick={toggleFullscreen}
  158. className="p-1.5 hover:bg-bambu-dark-tertiary rounded"
  159. title={isFullscreen ? 'Exit fullscreen' : 'Fullscreen'}
  160. >
  161. {isFullscreen ? (
  162. <Minimize className="w-4 h-4 text-bambu-gray" />
  163. ) : (
  164. <Maximize className="w-4 h-4 text-bambu-gray" />
  165. )}
  166. </button>
  167. </div>
  168. </div>
  169. {/* Video area */}
  170. <div className="flex-1 flex items-center justify-center p-2">
  171. <div className="relative w-full h-full flex items-center justify-center">
  172. {(streamLoading || transitioning) && (
  173. <div className="absolute inset-0 flex items-center justify-center bg-black/50 z-10">
  174. <div className="text-center">
  175. <RefreshCw className="w-8 h-8 text-bambu-gray animate-spin mx-auto mb-2" />
  176. <p className="text-sm text-bambu-gray">
  177. {streamMode === 'stream' ? 'Connecting to camera...' : 'Capturing snapshot...'}
  178. </p>
  179. </div>
  180. </div>
  181. )}
  182. {streamError && (
  183. <div className="absolute inset-0 flex items-center justify-center bg-black z-10">
  184. <div className="text-center p-4">
  185. <AlertTriangle className="w-12 h-12 text-orange-400 mx-auto mb-3" />
  186. <p className="text-white mb-2">Camera unavailable</p>
  187. <p className="text-xs text-bambu-gray mb-4 max-w-md">
  188. Make sure the printer is powered on and connected.
  189. </p>
  190. <button
  191. onClick={refresh}
  192. className="px-4 py-2 bg-bambu-green text-white rounded hover:bg-bambu-green/80 transition-colors"
  193. >
  194. Retry
  195. </button>
  196. </div>
  197. </div>
  198. )}
  199. <img
  200. ref={imgRef}
  201. key={imageKey}
  202. src={currentUrl}
  203. alt="Camera stream"
  204. className="max-w-full max-h-full object-contain"
  205. onError={currentUrl ? handleStreamError : undefined}
  206. onLoad={currentUrl ? handleStreamLoad : undefined}
  207. />
  208. </div>
  209. </div>
  210. </div>
  211. );
  212. }