| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  | import React, { useCallback, useRef, useState } from "react"; | 
					
						
							|  |  |  | import { Button, StyleSheet, Text, View } from "react-native"; | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   Camera, | 
					
						
							|  |  |  |   useCameraPermission, | 
					
						
							|  |  |  |   useCameraDevice, | 
					
						
							|  |  |  |   useCameraFormat, | 
					
						
							|  |  |  |   PhotoFile, | 
					
						
							|  |  |  |   VideoFile, | 
					
						
							|  |  |  |   CameraRuntimeError, | 
					
						
							|  |  |  |   Orientation, | 
					
						
							| 
									
										
										
										
											2024-02-01 00:58:23 -07:00
										 |  |  |   // @ts-ignore
 | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  | } from "react-native-vision-camera"; | 
					
						
							|  |  |  | import { RecordingButton } from "./capture-button"; | 
					
						
							|  |  |  | import { useIsForeground } from "./is-foreground"; | 
					
						
							| 
									
										
										
										
											2024-02-01 19:43:44 -07:00
										 |  |  | import { useIsFocused } from "@react-navigation/native"; | 
					
						
							| 
									
										
										
										
											2024-01-31 15:55:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  | export default function CameraScreen(): React.ReactElement { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |   const camera = useRef<Camera>(null); | 
					
						
							|  |  |  |   const { hasPermission, requestPermission } = useCameraPermission(); | 
					
						
							| 
									
										
										
										
											2024-02-01 19:43:44 -07:00
										 |  |  |   const [isCameraInitialized, setIsCameraInitialized] = | 
					
						
							|  |  |  |     useState<boolean>(false); | 
					
						
							| 
									
										
										
										
											2024-01-30 00:15:23 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-01 11:42:50 -08:00
										 |  |  |   const isForeground = useIsForeground(); | 
					
						
							| 
									
										
										
										
											2024-01-31 15:55:33 -08:00
										 |  |  |   const isFocused = useIsFocused(); | 
					
						
							| 
									
										
										
										
											2024-02-01 11:42:50 -08:00
										 |  |  |   const isActive = isForeground && isFocused; | 
					
						
							| 
									
										
										
										
											2024-01-30 14:08:46 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const onError = useCallback((error: CameraRuntimeError) => { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     console.error(error); | 
					
						
							|  |  |  |   }, []); | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const onInitialized = useCallback(() => { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     console.log("Camera initialized!"); | 
					
						
							|  |  |  |     setIsCameraInitialized(true); | 
					
						
							|  |  |  |   }, []); | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |   const onMediaCaptured = useCallback((media: PhotoFile | VideoFile) => { | 
					
						
							|  |  |  |     console.log(`Media captured! ${JSON.stringify(media)}`); | 
					
						
							|  |  |  |   }, []); | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-01 19:43:44 -07:00
										 |  |  |   const onVideoChunkReady = useCallback((event) => { | 
					
						
							|  |  |  |     console.log(`Chunk ready in react-native`, event.nativeEvent); | 
					
						
							|  |  |  |   }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |   if (!hasPermission) { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     requestPermission(); | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |     // Error handling in case they refuse to give permission
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |   const device = useCameraDevice("back"); | 
					
						
							| 
									
										
										
										
											2024-01-30 14:08:46 -08:00
										 |  |  |   const format = useCameraFormat(device, [ | 
					
						
							|  |  |  |     { videoResolution: { width: 3048, height: 2160 } }, | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     { fps: 60 }, | 
					
						
							|  |  |  |   ]); // this sets as a target
 | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |   //Orientation detection
 | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |   const [orientation, setOrientation] = useState<Orientation>("portrait"); | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const toggleOrientation = () => { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     setOrientation( | 
					
						
							|  |  |  |       (currentOrientation) => | 
					
						
							|  |  |  |         currentOrientation === "landscape-left" ? "portrait" : "landscape-left", // Can adjust this and the type to match what we want
 | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (device === null) { | 
					
						
							| 
									
										
										
										
											2024-02-01 19:43:44 -07:00
										 |  |  |     console.log(device); | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     return ( | 
					
						
							|  |  |  |       <Text> | 
					
						
							|  |  |  |         Camera not available. Does user have permissions: {hasPermission} | 
					
						
							|  |  |  |       </Text> | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |   } | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     hasPermission && ( | 
					
						
							|  |  |  |       <View style={styles.container}> | 
					
						
							|  |  |  |         <Camera | 
					
						
							|  |  |  |           ref={camera} | 
					
						
							|  |  |  |           style={StyleSheet.absoluteFill} | 
					
						
							|  |  |  |           device={device} | 
					
						
							| 
									
										
										
										
											2024-01-30 14:08:46 -08:00
										 |  |  |           format={format} | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |           onInitialized={onInitialized} | 
					
						
							| 
									
										
										
										
											2024-01-30 14:08:46 -08:00
										 |  |  |           onError={onError} | 
					
						
							| 
									
										
										
										
											2024-02-01 19:43:44 -07:00
										 |  |  |           onVideoChunkReady={onVideoChunkReady} | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |           video={true} | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |           orientation={orientation} // TODO: #60
 | 
					
						
							| 
									
										
										
										
											2024-01-30 14:08:46 -08:00
										 |  |  |           isActive={isActive} | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |         /> | 
					
						
							|  |  |  |         <RecordingButton | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |           style={[ | 
					
						
							|  |  |  |             styles.captureButton, | 
					
						
							|  |  |  |             orientation === "portrait" ? styles.portrait : styles.landscape, | 
					
						
							|  |  |  |           ]} | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |           camera={camera} | 
					
						
							|  |  |  |           onMediaCaptured={onMediaCaptured} | 
					
						
							|  |  |  |           enabled={isCameraInitialized} | 
					
						
							|  |  |  |         /> | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |         <View | 
					
						
							|  |  |  |           style={[ | 
					
						
							|  |  |  |             styles.button, | 
					
						
							|  |  |  |             orientation === "portrait" | 
					
						
							|  |  |  |               ? styles.togglePortrait | 
					
						
							|  |  |  |               : styles.toggleLandscape, | 
					
						
							|  |  |  |           ]} | 
					
						
							|  |  |  |         > | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |           <Button | 
					
						
							|  |  |  |             title="Toggle Orientation" | 
					
						
							|  |  |  |             onPress={toggleOrientation} | 
					
						
							|  |  |  |             color="#841584" | 
					
						
							|  |  |  |             accessibilityLabel="Toggle camera orientation" | 
					
						
							|  |  |  |           /> | 
					
						
							|  |  |  |         </View> | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |       </View> | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |   ); | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const styles = StyleSheet.create({ | 
					
						
							|  |  |  |   container: { | 
					
						
							|  |  |  |     flex: 1, | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     backgroundColor: "black", | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |   }, | 
					
						
							|  |  |  |   captureButton: { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     position: "absolute", | 
					
						
							|  |  |  |     alignSelf: "center", | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |   button: { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     position: "absolute", | 
					
						
							|  |  |  |     alignSelf: "center", | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |   }, | 
					
						
							|  |  |  |   togglePortrait: { | 
					
						
							| 
									
										
										
										
											2024-01-30 17:05:43 -08:00
										 |  |  |     bottom: 110, // needs refined
 | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |   }, | 
					
						
							|  |  |  |   toggleLandscape: { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     transform: [{ rotate: "90deg" }], | 
					
						
							|  |  |  |     bottom: "43%", // Should come from SafeAreaProvider, hardcoded right now, should roughly appear above the button
 | 
					
						
							|  |  |  |     left: 50, // needs refined
 | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |   }, | 
					
						
							|  |  |  |   portrait: { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     bottom: 20, // needs refined
 | 
					
						
							| 
									
										
										
										
											2024-01-30 16:35:11 -08:00
										 |  |  |   }, | 
					
						
							|  |  |  |   landscape: { | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  |     bottom: "40%", // Should come from SafeAreaProvider
 | 
					
						
							|  |  |  |     left: 20, // needs refined
 | 
					
						
							| 
									
										
										
										
											2024-01-29 19:01:45 -08:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2024-01-31 22:45:00 -07:00
										 |  |  | }); |