UNPKG

@microblink/photopay-in-browser-sdk

Version:

A simple payment barcode scanning library for WebAssembly-enabled browsers.

533 lines (427 loc) 16.3 kB
/** * Copyright (c) Microblink Ltd. All rights reserved. */ import * as PhotoPaySDK from '@microblink/photopay-in-browser-sdk'; import { AvailableRecognizers, CameraEntry, CameraExperience, EventReady, VideoRecognitionConfiguration, ImageRecognitionConfiguration, ImageRecognitionType, RecognizerInstance, RecognitionEvent, RecognitionStatus, RecognitionResults, SdkSettings, SDKError } from './data-structures'; import * as ErrorTypes from './error-structures'; const _IS_IMAGE_CAPTURE = false; export interface CheckConclusion { status: boolean; message?: string; } export async function getCameraDevices(): Promise<Array<CameraEntry>> { const devices = await PhotoPaySDK.getCameraDevices(); const allDevices = devices.frontCameras.concat(devices.backCameras); const finalEntries = allDevices.map((el: PhotoPaySDK.SelectedCamera) => { return { prettyName: el.label, details: el } }); return finalEntries; } export class SdkService { private sdk: PhotoPaySDK.WasmSDK; private eventEmitter$: HTMLAnchorElement; private cancelInitiatedFromOutside: boolean = false; private recognizerName: string; public videoRecognizer: PhotoPaySDK.VideoRecognizer; public showOverlay: boolean = false; constructor() { this.eventEmitter$ = document.createElement('a'); } public delete() { this.sdk?.delete(); } public initialize(licenseKey: string, sdkSettings: SdkSettings): Promise<EventReady|SDKError> { const loadSettings = new PhotoPaySDK.WasmSDKLoadSettings(licenseKey); loadSettings.allowHelloMessage = sdkSettings.allowHelloMessage; loadSettings.engineLocation = sdkSettings.engineLocation; if (sdkSettings.wasmType) { loadSettings.wasmType = sdkSettings.wasmType; } return new Promise((resolve) => { PhotoPaySDK.loadWasmModule(loadSettings) .then((sdk: PhotoPaySDK.WasmSDK) => { this.sdk = sdk; this.showOverlay = sdk.showOverlay; resolve(new EventReady(this.sdk)); }) .catch(error => { resolve(new SDKError(ErrorTypes.componentErrors.sdkLoadFailed, error)); }); }); } public checkRecognizers(recognizers: Array<string>): CheckConclusion { if (!recognizers || !recognizers.length) { return { status: false, message: 'There are no provided recognizers!' } } for (const recognizer of recognizers) { if (!this.isRecognizerAvailable(recognizer)) { return { status: false, message: `Recognizer "${ recognizer }" doesn't exist!` } } } return { status: true } } public getDesiredCameraExperience(_recognizers: Array<string> = [], _recognizerOptions: any = {}): CameraExperience { return CameraExperience.Barcode; } public async scanFromCamera( configuration: VideoRecognitionConfiguration, eventCallback: (ev: RecognitionEvent) => void ): Promise<void> { eventCallback({ status: RecognitionStatus.Preparing }); this.cancelInitiatedFromOutside = false; // Prepare terminate mechanism before recognizer and runner instances are created this.eventEmitter$.addEventListener('terminate', async () => { this.videoRecognizer?.cancelRecognition?.(); window.setTimeout(() => this.videoRecognizer?.releaseVideoFeed?.(), 1); if (recognizerRunner) { try { await recognizerRunner.delete(); } catch (error) { // Psst, this error should not happen. } } for (const recognizer of recognizers) { if (!recognizer) { continue; } if (recognizer.recognizer?.objectHandle > -1) { recognizer.recognizer.delete?.(); } if (recognizer.successFrame?.objectHandle > -1) { recognizer.successFrame.delete?.(); } } }); // Prepare recognizers and runner const recognizers = await this.createRecognizers( configuration.recognizers, configuration.recognizerOptions, configuration.successFrame ); const recognizerRunner = await this.createRecognizerRunner( recognizers, eventCallback ); try { this.videoRecognizer = await PhotoPaySDK.VideoRecognizer.createVideoRecognizerFromCameraStream( configuration.cameraFeed, recognizerRunner, configuration.cameraId ); eventCallback({ status: RecognitionStatus.Ready }); await this.videoRecognizer.setVideoRecognitionMode(PhotoPaySDK.VideoRecognitionMode.Recognition); this.videoRecognizer.startRecognition( async (recognitionState: PhotoPaySDK.RecognizerResultState) => { this.videoRecognizer.pauseRecognition(); eventCallback({ status: RecognitionStatus.Processing }); if (recognitionState !== PhotoPaySDK.RecognizerResultState.Empty) { for (const recognizer of recognizers) { const results = await recognizer.recognizer.getResult(); this.recognizerName = recognizer.recognizer.recognizerName; if (!results || results.state === PhotoPaySDK.RecognizerResultState.Empty) { eventCallback({ status: RecognitionStatus.EmptyResultState, data: { initiatedByUser: this.cancelInitiatedFromOutside, recognizerName: this.recognizerName } }); } else { const recognitionResults: RecognitionResults = { recognizer: results, recognizerName: this.recognizerName } if (recognizer.successFrame) { const successFrameResults = await recognizer.successFrame.getResult(); if (successFrameResults && successFrameResults.state !== PhotoPaySDK.RecognizerResultState.Empty) { recognitionResults.successFrame = successFrameResults; } } recognitionResults.imageCapture = _IS_IMAGE_CAPTURE; const scanData: any = { result: recognitionResults, initiatedByUser: this.cancelInitiatedFromOutside, imageCapture: _IS_IMAGE_CAPTURE } eventCallback({ status: RecognitionStatus.ScanSuccessful, data: scanData }); break; } } } else { eventCallback({ status: RecognitionStatus.EmptyResultState, data: { initiatedByUser: this.cancelInitiatedFromOutside, recognizerName: '' } }); } window.setTimeout(() => void this.cancelRecognition(), 400); }, configuration.recognitionTimeout); } catch (error) { if (error && error.details?.reason) { const reason = error.details?.reason; switch (reason) { case PhotoPaySDK.NotSupportedReason.MediaDevicesNotSupported: eventCallback({ status: RecognitionStatus.NoSupportForMediaDevices }); break; case PhotoPaySDK.NotSupportedReason.CameraNotFound: eventCallback({ status: RecognitionStatus.CameraNotFound }); break; case PhotoPaySDK.NotSupportedReason.CameraNotAllowed: eventCallback({ status: RecognitionStatus.CameraNotAllowed }); break; case PhotoPaySDK.NotSupportedReason.CameraInUse: eventCallback({ status: RecognitionStatus.CameraInUse }); break; default: eventCallback({ status: RecognitionStatus.UnableToAccessCamera }); } console.warn('VideoRecognizerError', error.name, '[' + reason + ']:', error.message); } else { eventCallback({ status: RecognitionStatus.UnknownError }); } void this.cancelRecognition(); } } public async flipCamera(): Promise<void> { await this.videoRecognizer.flipCamera(); } public isCameraFlipped(): boolean { if (!this.videoRecognizer) { return false; } return this.videoRecognizer.isCameraFlipped(); } public isScanFromImageAvailable(_recognizers: Array<string> = [], _recognizerOptions: any = {}): boolean { return true; } public getScanFromImageType(_recognizers: Array<string> = [], _recognizerOptions: any = {}): ImageRecognitionType { return ImageRecognitionType.Single; } public async scanFromImage( configuration: ImageRecognitionConfiguration, eventCallback: (ev: RecognitionEvent) => void ): Promise<void> { eventCallback({ status: RecognitionStatus.Preparing }); const recognizers = await this.createRecognizers( configuration.recognizers, configuration.recognizerOptions ); const recognizerRunner = await this.createRecognizerRunner( recognizers, eventCallback ); const handleTerminate = async () => { this.eventEmitter$.removeEventListener('terminate', handleTerminate); if (recognizerRunner) { try { await recognizerRunner.delete(); } catch (error) { // Psst, this error should not happen. } } for (const recognizer of recognizers) { if (!recognizer) { continue; } if (recognizer.recognizer?.objectHandle > -1) { recognizer.recognizer.delete?.(); } } this.eventEmitter$.dispatchEvent(new Event('terminate:done')); }; this.eventEmitter$.addEventListener('terminate', handleTerminate); // Get image file if (!configuration.file || !RegExp(/^image\//).exec(configuration.file.type)) { eventCallback({ status: RecognitionStatus.NoImageFileFound }); window.setTimeout(() => void this.cancelRecognition(), 500); return; } const file = configuration.file; const imageElement = new Image(); imageElement.src = URL.createObjectURL(file); await imageElement.decode(); const imageFrame = PhotoPaySDK.captureFrame(imageElement); // Get results eventCallback({ status: RecognitionStatus.Processing }); const processResult = await recognizerRunner.processImage(imageFrame); if (processResult !== PhotoPaySDK.RecognizerResultState.Empty) { for (const recognizer of recognizers) { const results = await recognizer.recognizer.getResult(); if (!results || results.state === PhotoPaySDK.RecognizerResultState.Empty) { eventCallback({ status: RecognitionStatus.EmptyResultState, data: { initiatedByUser: this.cancelInitiatedFromOutside, recognizerName: recognizer.name } }); } else { const recognitionResults: RecognitionResults = { recognizer: results, imageCapture: _IS_IMAGE_CAPTURE, recognizerName: recognizer.name }; eventCallback({ status: RecognitionStatus.ScanSuccessful, data: recognitionResults }); break; } } } else { eventCallback({ status: RecognitionStatus.EmptyResultState, data: { initiatedByUser: this.cancelInitiatedFromOutside, recognizerName: '' } }); } window.setTimeout(() => void this.cancelRecognition(), 500); } public async stopRecognition() { void await this.cancelRecognition(true); } public async resumeRecognition(): Promise<void> { this.videoRecognizer.resumeRecognition(true); } public changeCameraDevice(camera: PhotoPaySDK.SelectedCamera): Promise<boolean> { return new Promise((resolve) => { this.videoRecognizer.changeCameraDevice(camera) .then(() => resolve(true)) .catch(() => resolve(false)); }); } ////////////////////////////////////////////////////////////////////////////// // // PRIVATE METHODS private isRecognizerAvailable(recognizer: string): boolean { return !!AvailableRecognizers[recognizer]; } private async createRecognizers( recognizers: Array<string>, recognizerOptions?: any, successFrame: boolean = false ): Promise<Array<RecognizerInstance>> { const pureRecognizers = []; for (const recognizer of recognizers) { const instance = await PhotoPaySDK[AvailableRecognizers[recognizer]](this.sdk); pureRecognizers.push(instance); } if (recognizerOptions && Object.keys(recognizerOptions).length > 0) { for (const recognizer of pureRecognizers) { const settings = await recognizer.currentSettings(); let updated = false; if ( !recognizerOptions[recognizer.recognizerName] || Object.keys(recognizerOptions[recognizer.recognizerName]).length < 1 ) { continue; } for (const [key, value] of Object.entries(recognizerOptions[recognizer.recognizerName])) { if (key in settings) { settings[key] = value; updated = true; } } if (updated) { await recognizer.updateSettings(settings); } } } const recognizerInstances = []; for (let i = 0; i < pureRecognizers.length; ++i) { const recognizer = pureRecognizers[i]; const instance: RecognizerInstance = { name: recognizers[i], recognizer } if (successFrame) { const successFrameGrabber = await PhotoPaySDK.createSuccessFrameGrabberRecognizer(this.sdk, recognizer); instance.successFrame = successFrameGrabber; } recognizerInstances.push(instance) } return recognizerInstances; } private async createRecognizerRunner( recognizers: Array<RecognizerInstance>, eventCallback: (ev: RecognitionEvent) => void ): Promise<PhotoPaySDK.RecognizerRunner> { const metadataCallbacks: PhotoPaySDK.MetadataCallbacks = { onDetectionFailed: () => eventCallback({ status: RecognitionStatus.DetectionFailed }), onQuadDetection: (quad: PhotoPaySDK.Displayable) => { eventCallback({ status: RecognitionStatus.DetectionStatusChange, data: quad }); const detectionStatus = quad.detectionStatus; switch (detectionStatus) { case PhotoPaySDK.DetectionStatus.Fail: eventCallback({ status: RecognitionStatus.DetectionStatusSuccess }); break; case PhotoPaySDK.DetectionStatus.Success: eventCallback({ status: RecognitionStatus.DetectionStatusSuccess }); break; case PhotoPaySDK.DetectionStatus.CameraTooHigh: eventCallback({ status: RecognitionStatus.DetectionStatusCameraTooHigh }); break; case PhotoPaySDK.DetectionStatus.FallbackSuccess: eventCallback({ status: RecognitionStatus.DetectionStatusFallbackSuccess }); break; case PhotoPaySDK.DetectionStatus.Partial: eventCallback({ status: RecognitionStatus.DetectionStatusPartial }); break; case PhotoPaySDK.DetectionStatus.CameraAtAngle: eventCallback({ status: RecognitionStatus.DetectionStatusCameraAtAngle }); break; case PhotoPaySDK.DetectionStatus.CameraTooNear: eventCallback({ status: RecognitionStatus.DetectionStatusCameraTooNear }); break; case PhotoPaySDK.DetectionStatus.DocumentTooCloseToEdge: eventCallback({ status: RecognitionStatus.DetectionStatusDocumentTooCloseToEdge }); break; default: // Send nothing } } } const recognizerRunner = await PhotoPaySDK.createRecognizerRunner( this.sdk, recognizers.map((el: RecognizerInstance) => el.successFrame || el.recognizer), false, metadataCallbacks ); return recognizerRunner; } private async cancelRecognition(initiatedFromOutside: boolean = false): Promise<void> { this.cancelInitiatedFromOutside = initiatedFromOutside; this.eventEmitter$.dispatchEvent(new Event('terminate')); } }