diff --git a/package.json b/package.json index 3fa4ac35..62a1e34f 100644 --- a/package.json +++ b/package.json @@ -145,6 +145,7 @@ "react-native-tor": "^0.1.8", "react-native-unimodules": "^0.14.10", "react-native-vector-icons": "9.2.0", + "react-native-vision-camera": "4.3.2", "react-native-web": "0.19.12", "react-native-web-linear-gradient": "^1.1.2", "react-native-web-maps": "^0.3.0", diff --git a/src/components/Camera.tsx b/src/components/Camera.tsx index b5614831..6710ca8e 100644 --- a/src/components/Camera.tsx +++ b/src/components/Camera.tsx @@ -1,61 +1,57 @@ -import React, { ReactNode, useState, useEffect } from "react"; +import React, { ReactNode, useEffect } from "react"; import { StyleProp, ViewStyle, InteractionManager, StyleSheet } from "react-native"; -import { Camera, CameraType } from "react-native-camera-kit"; +import { + Camera, + useCodeScanner, + useCameraDevice, + useCameraPermission, +} from "react-native-vision-camera"; import Container from "./Container"; -import { PLATFORM } from "../utils/constants"; - -let ReactNativePermissions: any; -if (PLATFORM !== "macos") { - ReactNativePermissions = require("react-native-permissions"); -} export interface ICamera { active?: boolean; children?: ReactNode | JSX.Element; - cameraType?: keyof CameraType; + cameraType?: keyof CameraType; // TODO(hsjoberg) onRead?: (text: string) => void; onNotAuthorized?: () => void; // TODO(hsjoberg): style?: StyleProp; } -export default function CameraComponent({ - cameraType, - children, - onNotAuthorized, - onRead, - style, - active, -}: ICamera) { - const [start, setStart] = useState(false); +export default function CameraComponent({ children, onNotAuthorized, onRead, active }: ICamera) { + const device = useCameraDevice("back"); + const { hasPermission, requestPermission } = useCameraPermission(); + const codeScanner = useCodeScanner({ + codeTypes: ["qr"], + onCodeScanned: (codes) => { + if (codes.length >= 0) { + onRead?.(codes[0].value ?? ""); + } + }, + }); active = active ?? true; useEffect(() => { - InteractionManager.runAfterInteractions(async () => { - const permission = - PLATFORM === "ios" - ? ReactNativePermissions.PERMISSIONS.IOS.CAMERA - : ReactNativePermissions.PERMISSIONS.ANDROID.CAMERA; - const result = await ReactNativePermissions.request(permission); - - if (result === "granted" || result === "limited") { - console.log("Camera permission not granted"); - setStart(true); - } else { - console.log("Camera permission granted"); + (async () => { + if (hasPermission === false) { + console.log("Does not have camera permission"); + if (await !requestPermission()) { + // TODO fix await + onNotAuthorized?.(); + } } - }); - }, []); + })(); + }, [requestPermission, hasPermission]); - if (!start || !active) { + if (!active || !hasPermission || !device) { return {children ?? <>}; } return ( <> onRead?.(event.nativeEvent.codeStringValue)} + style={StyleSheet.absoluteFill} + codeScanner={codeScanner} + device={device} + isActive={active} /> {children} diff --git a/src/components/FooterNav.tsx b/src/components/FooterNav.tsx index d559659f..fde1cd7f 100644 --- a/src/components/FooterNav.tsx +++ b/src/components/FooterNav.tsx @@ -20,9 +20,7 @@ export default function FooterNav() {