Skip to content

Commit

Permalink
Address feedback
Browse files Browse the repository at this point in the history
* Improve error handling
* Improve torch switch-off handling
* Remove unnecessary react fragements
  • Loading branch information
GovernmentPlates committed Aug 20, 2024
1 parent 11aa451 commit 85b5b2c
Show file tree
Hide file tree
Showing 2 changed files with 30 additions and 20 deletions.
30 changes: 19 additions & 11 deletions src/Components/QrScanner/QrScannerPlugin.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
// file = QrScannerPlugin.jsx
import {MutableRefObject, useEffect, useRef, useState} from 'react';
import {MutableRefObject, useEffect, useRef, useState, useCallback} from 'react';
import {ArrowUpTrayIcon} from '@heroicons/react/24/solid';
import {Html5Qrcode, Html5QrcodeScannerState, Html5QrcodeSupportedFormats} from 'html5-qrcode';
import {useLogError} from '../../hooks/useError';
import {checkCameraPermissions} from '../../utils/media';
import {TorchButton} from './TorchButton';
import classes from './QrScanner.module.css';
Expand Down Expand Up @@ -53,18 +54,24 @@ export default function QrScannerPlugin({
const aspectRatio = calcAspectRatio();
const html5CustomScanner: MutableRefObject<Html5Qrcode | null> = useRef(null);
const [canUseCamera, setCanUseCamera] = useState(true);
const logError = useLogError();

// Turn off the torch (if it is on) when navigating away from the scan page
async function switchOffTorch(html5CustomScanner: MutableRefObject<Html5Qrcode | null>) {
try {
const track = html5CustomScanner?.current?.getRunningTrackCameraCapabilities();
if (track && track.torchFeature().value()) {
await track.torchFeature().apply(false);
const switchOffTorch = useCallback(
async function switchOffTorch(html5CustomScanner: MutableRefObject<Html5Qrcode | null>) {
try {
const track = html5CustomScanner?.current?.getRunningTrackCameraCapabilities();
if (track && track.torchFeature().value()) {
await track.torchFeature().apply(false);
}
} catch (error) {
// This raises an error about invalid tracks - we have to catch it! (blame the library)
console.warn('Failed to disable torch:', error);
logError(`Failed to disable torch: ${error}`);
}
} catch (error) {
console.warn('Failed to disable torch:', error);
}
}
},
[logError]
);

useEffect(() => {
const showQRCode = async () => {
Expand Down Expand Up @@ -97,8 +104,8 @@ export default function QrScannerPlugin({

return () => {
const stopQrScanner = async () => {
await switchOffTorch(html5CustomScanner);
if (html5CustomScanner.current?.isScanning) {
switchOffTorch(html5CustomScanner);
await html5CustomScanner.current.stop();
}
html5CustomScanner.current?.clear();
Expand All @@ -117,6 +124,7 @@ export default function QrScannerPlugin({
qrCodeSuccessCallback,
qrCodeErrorCallback,
onPermRefused,
switchOffTorch,
]);

return (
Expand Down
20 changes: 11 additions & 9 deletions src/Components/QrScanner/TorchButton.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import {MutableRefObject, useState, useEffect} from 'react';
import {BoltIcon, BoltSlashIcon, ExclamationCircleIcon} from '@heroicons/react/24/solid';
import {Html5Qrcode} from 'html5-qrcode';
import PropTypes from 'prop-types';
import {useLogError} from '../../hooks/useError';

interface TorchButtonProps {
html5CustomScanner: MutableRefObject<Html5Qrcode | null>;
Expand All @@ -11,6 +12,7 @@ interface TorchButtonProps {
export function TorchButton({html5CustomScanner, canUseCamera}: TorchButtonProps) {
const [torchOn, setTorchOn] = useState(false);
const [torchUnavailable, setTorchUnavailable] = useState(false);
const logError = useLogError();

useEffect(() => {
const toggleTorch = async () => {
Expand All @@ -25,29 +27,29 @@ export function TorchButton({html5CustomScanner, canUseCamera}: TorchButtonProps
} catch (error) {
setTorchUnavailable(true);
console.warn('Failed to toggle torch:', error);
logError(`Failed to toggle torch: ${error}`);
}
};

toggleTorch();
}, [torchOn, html5CustomScanner]);
}, [torchOn, html5CustomScanner, logError]);

if (!canUseCamera) {
return;
}

if (torchUnavailable) {
return (
<>
<div className="fit-content flex justify-center gap-1 bg-yellow-500 py-3 text-center text-amber-900">
<span className="flex items-center">
<ExclamationCircleIcon className="mr-1 h-6 w-6" />
Your device's torch is unavailable
</span>
</div>
</>
<div className="fit-content flex justify-center gap-1 bg-yellow-500 py-3 text-center text-amber-900">
<span className="flex items-center">
<ExclamationCircleIcon className="mr-1 h-6 w-6" />
Your device's torch is unavailable
</span>
</div>
);
}

// TODO: Feedback (make it so that it feels like a button - dark on-press for 100 ms)
return (
<div
onClick={() => setTorchOn(prev => !prev)}
Expand Down

0 comments on commit 85b5b2c

Please sign in to comment.