From a5a0a2a9a1cd629954fb41b7f01d682bc470c60e Mon Sep 17 00:00:00 2001 From: Niklas Gruhn Date: Mon, 2 Sep 2024 13:30:36 +0200 Subject: [PATCH] perf: BarcodeDetector polyfill only if no native support Still using polyfill for `QrcodeDropZone` and `QrcodeCapture` even on platforms with native `BarcodeDetector` is available, because on some of them `BarcodeDetector.detect` does not support `Blob` / `File` inputs. See: #447 --- .gitignore | 3 ++ src/misc/scanner.ts | 60 ++++++++++++++++++++++++++++++------ src/misc/shimGetUserMedia.ts | 4 +-- src/misc/util.ts | 2 +- 4 files changed, 56 insertions(+), 13 deletions(-) diff --git a/.gitignore b/.gitignore index 3ff7c80a..18084c6d 100644 --- a/.gitignore +++ b/.gitignore @@ -25,6 +25,9 @@ # Icon must end with two \r Icon +# VitePress +.vitepress/cache +.vitepress/dist # Thumbnails ._* diff --git a/src/misc/scanner.ts b/src/misc/scanner.ts index c2644b59..feb0266a 100644 --- a/src/misc/scanner.ts +++ b/src/misc/scanner.ts @@ -1,10 +1,44 @@ -import { type DetectedBarcode, type BarcodeFormat, BarcodeDetector } from 'barcode-detector/pure' +import { type DetectedBarcode, type BarcodeFormat, BarcodeDetector, type BarcodeDetectorOptions } from 'barcode-detector/pure' import { eventOn } from './callforth' import { DropImageFetchError } from './errors' +declare global { + interface Window { + BarcodeDetector?: typeof BarcodeDetector + } +} + +/** + * Singleton `BarcodeDetector` instance used by `QrcodeStream`. This is firtly to avoid + * the overhead of creating a new instances for scanning each frame. And secondly, the + * instances can seamlessly be replaced in the middle of the scanning process, if the + * `formats` prop of `QrcodeStream` is changed. + * + * This instance is not used by `QrcodeCapture` and `QrcodeDropZone`, because it may not + * have the right `formats` configured. For these components we create one-off `BarcodeDetector` + * instances because it does not happen so frequently anyway (see: `processFile`/`processUrl`). + */ let barcodeDetector: BarcodeDetector -export const setScanningFormats = (formats: BarcodeFormat[]) => { - barcodeDetector = new BarcodeDetector({ formats }) + +/** + * Seamlessly updates the set of used barcode formats during scanning. + */ +export function setScanningFormats(formats: BarcodeFormat[]) { + // Only use the `BarcodeDetector` polyfill if the API is not supported natively. + // + // Note, that we can't just monkey patch the API on load, i.e. + // + // globalThis.BarcodeDetector ??= BarcodeDetector + // + // because that is not SSR compatible. If the polyfill is applied during SSR, then + // it's actually missing at runtime. Thus, we have to check the API support at runtime: + if (window.BarcodeDetector === undefined) { + console.debug('[vue-qrcode-reader] BarcodeDetector not available: will use polyfill.') + barcodeDetector = new BarcodeDetector({ formats }) + } else { + console.debug('[vue-qrcode-reader] BarcodeDetector available: will use native API.') + barcodeDetector = new window.BarcodeDetector({ formats }) + } } type ScanHandler = (_: DetectedBarcode[]) => void @@ -28,7 +62,7 @@ export const keepScanning = async ( } ) => { console.debug('[vue-qrcode-reader] start scanning') - barcodeDetector = new BarcodeDetector({ formats }) + setScanningFormats(formats) const processFrame = (state: { lastScanned: number; contentBefore: string[]; lastScanHadContent: boolean }) => @@ -140,9 +174,12 @@ export const processFile = async ( file: File, formats: BarcodeFormat[] = ['qr_code'] ): Promise => { - const barcodeDetector = new BarcodeDetector({ - formats - }) + // To scan files/urls we use one-off `BarcodeDetector` instnaces, + // since we don't scan as often as camera frames. Note, that we + // always use the polyfill. This is because (at the time of writing) + // some browser/OS combinations don't support `Blob`/`File` inputs + // into the `detect` function. + const barcodeDetector = new BarcodeDetector({ formats }) return await barcodeDetector.detect(file) } @@ -151,9 +188,12 @@ export const processUrl = async ( url: string, formats: BarcodeFormat[] = ['qr_code'] ): Promise => { - const barcodeDetector = new BarcodeDetector({ - formats - }) + // To scan files/urls we use one-off `BarcodeDetector` instnaces, + // since we don't scan as often as camera frames. Note, that we + // always use the polyfill. This is because (at the time of writing) + // some browser/OS combinations don't support `Blob`/`File` inputs + // into the `detect` function. + const barcodeDetector = new BarcodeDetector({ formats }) const image = await imageElementFromUrl(url) diff --git a/src/misc/shimGetUserMedia.ts b/src/misc/shimGetUserMedia.ts index 7b73c253..c65bae3e 100644 --- a/src/misc/shimGetUserMedia.ts +++ b/src/misc/shimGetUserMedia.ts @@ -8,9 +8,9 @@ import { shimGetUserMedia as safariShim } from 'webrtc-adapter/dist/safari/safar import { detectBrowser } from 'webrtc-adapter/dist/utils' import { StreamApiNotSupportedError } from './errors' -import { indempotent } from './util' +import { idempotent } from './util' -export default indempotent(() => { +export default idempotent(() => { const browserDetails = detectBrowser(window) switch (browserDetails.browser) { diff --git a/src/misc/util.ts b/src/misc/util.ts index d2baf68f..f67fac2f 100644 --- a/src/misc/util.ts +++ b/src/misc/util.ts @@ -2,7 +2,7 @@ * Takes a function `action` and returns a new function, that behaves * like action but when called a second time does nothing. */ -export const indempotent = (action: (x: any) => T) => { +export const idempotent = (action: (x: any) => T) => { let called = false let result: T | undefined = undefined