// (C) Copyright 2015 Moodle Pty Ltd. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. import { Injectable } from '@angular/core'; import { makeSingleton } from '@singletons'; import { CoreCanceledError } from '@classes/errors/cancelederror'; import { CorePromisedValue } from '@classes/promised-value'; import { QRScanner } from '@features/native/plugins'; import { CoreModals } from './modals'; import { CorePlatform } from './platform'; import { Subscription } from 'rxjs'; import { CoreCustomURLSchemes } from './urlschemes'; /** * Handles qr scan services. */ @Injectable({ providedIn: 'root' }) export class CoreQRScanService { protected qrScanData?: {deferred: CorePromisedValue; observable: Subscription}; protected initialColorSchemeContent = 'light dark'; /** * Check whether the app can scan QR codes. * * @returns Whether the app can scan QR codes. */ canScanQR(): boolean { return CorePlatform.isMobile(); } /** * Open a modal to scan a QR code. * * @param title Title of the modal. Defaults to "QR reader". * @returns Promise resolved with the captured text or undefined if cancelled or error. */ async scanQR(title?: string): Promise { const { CoreViewerQRScannerComponent } = await import('@features/viewer/components/qr-scanner/qr-scanner'); return CoreModals.openModal({ component: CoreViewerQRScannerComponent, cssClass: 'core-modal-fullscreen', componentProps: { title, }, }); } /** * Scan a QR code and handle the URL if it's a custom URL scheme. * * @param title Title of the modal. Defaults to "QR reader". * @returns Promise resolved with the captured text or undefined if cancelled, error or URL handled. */ async scanQRWithUrlHandling(title?: string): Promise { // Scan for a QR code. const text = await CoreQRScan.scanQR(title); if (!text) { return; } if (CoreCustomURLSchemes.isCustomURL(text)) { // Is a custom URL scheme, handle it. try { await CoreCustomURLSchemes.handleCustomURL(text); } catch (error) { CoreCustomURLSchemes.treatHandleCustomURLError(error); } return; } return text; } /** * Start scanning for a QR code. * * @returns Promise resolved with the QR string, rejected if error or cancelled. */ async startScanQR(): Promise { if (!CorePlatform.isMobile()) { return Promise.reject('QRScanner isn\'t available in browser.'); } // Ask the user for permission to use the camera. // The scan method also does this, but since it returns an Observable we wouldn't be able to detect if the user denied. try { const status = await QRScanner.prepare(); if (!status.authorized) { // No access to the camera, reject. In android this shouldn't happen, denying access passes through catch. throw new Error('The user denied camera access.'); } if (this.qrScanData && this.qrScanData.deferred) { // Already scanning. return this.qrScanData.deferred; } // Start scanning. this.qrScanData = { deferred: new CorePromisedValue(), // When text is received, stop scanning and return the text. observable: QRScanner.scan().subscribe(text => this.stopScanQR(text, false)), }; // Show the camera. try { await QRScanner.show(); document.body.classList.add('core-scanning-qr'); // Set color-scheme to 'normal', otherwise the camera isn't seen in Android. const colorSchemeMeta = document.querySelector('meta[name="color-scheme"]'); if (colorSchemeMeta) { this.initialColorSchemeContent = colorSchemeMeta.getAttribute('content') || this.initialColorSchemeContent; colorSchemeMeta.setAttribute('content', 'normal'); } return this.qrScanData.deferred; } catch (error) { this.stopScanQR(error, true); throw error; } } catch (error) { // eslint-disable-next-line @typescript-eslint/naming-convention error.message = error.message || (error as { _message?: string })._message; throw error; } } /** * Stop scanning for QR code. If no param is provided, the app will consider the user cancelled. * * @param data If success, the text of the QR code. If error, the error object or message. Undefined for cancelled. * @param error True if the data belongs to an error, false otherwise. */ stopScanQR(data?: string | Error, error?: boolean): void { if (!this.qrScanData) { // Not scanning. return; } // Hide camera preview. document.body.classList.remove('core-scanning-qr'); // Set color-scheme to the initial value. document.querySelector('meta[name="color-scheme"]')?.setAttribute('content', this.initialColorSchemeContent); QRScanner.hide(); QRScanner.destroy(); this.qrScanData.observable.unsubscribe(); // Stop scanning. if (error) { this.qrScanData.deferred.reject(typeof data === 'string' ? new Error(data) : data); } else if (data !== undefined) { this.qrScanData.deferred.resolve(data as string); } else { this.qrScanData.deferred.reject(new CoreCanceledError()); } delete this.qrScanData; } } export const CoreQRScan = makeSingleton(CoreQRScanService);