html5-qrcode
Version:
A cross platform HTML5 QR Code & bar code scanner
156 lines (141 loc) • 5.87 kB
text/typescript
/**
* @fileoverview
* {@interface QrcodeDecoder} wrapper around ZXing library.
*
* @author mebjas <minhazav@gmail.com>
*
* ZXing library forked from https://github.com/zxing-js/library.
*
* The word "QR Code" is registered trademark of DENSO WAVE INCORPORATED
* http://www.denso-wave.com/qrcode/faqpatent-e.html
*/
import * as ZXing from "../third_party/zxing-js.umd";
import {
QrcodeResult,
QrcodeResultDebugData,
QrcodeResultFormat,
Html5QrcodeSupportedFormats,
Logger,
QrcodeDecoderAsync
} from "./core";
/**
* ZXing based Code decoder.
*/
export class ZXingHtml5QrcodeDecoder implements QrcodeDecoderAsync {
private readonly formatMap: Map<Html5QrcodeSupportedFormats, any>
= new Map([
[Html5QrcodeSupportedFormats.QR_CODE, ZXing.BarcodeFormat.QR_CODE ],
[Html5QrcodeSupportedFormats.AZTEC, ZXing.BarcodeFormat.AZTEC ],
[Html5QrcodeSupportedFormats.CODABAR, ZXing.BarcodeFormat.CODABAR ],
[Html5QrcodeSupportedFormats.CODE_39, ZXing.BarcodeFormat.CODE_39 ],
[Html5QrcodeSupportedFormats.CODE_93, ZXing.BarcodeFormat.CODE_93 ],
[
Html5QrcodeSupportedFormats.CODE_128,
ZXing.BarcodeFormat.CODE_128 ],
[
Html5QrcodeSupportedFormats.DATA_MATRIX,
ZXing.BarcodeFormat.DATA_MATRIX ],
[
Html5QrcodeSupportedFormats.MAXICODE,
ZXing.BarcodeFormat.MAXICODE ],
[Html5QrcodeSupportedFormats.ITF, ZXing.BarcodeFormat.ITF ],
[Html5QrcodeSupportedFormats.EAN_13, ZXing.BarcodeFormat.EAN_13 ],
[Html5QrcodeSupportedFormats.EAN_8, ZXing.BarcodeFormat.EAN_8 ],
[Html5QrcodeSupportedFormats.PDF_417, ZXing.BarcodeFormat.PDF_417 ],
[Html5QrcodeSupportedFormats.RSS_14, ZXing.BarcodeFormat.RSS_14 ],
[
Html5QrcodeSupportedFormats.RSS_EXPANDED,
ZXing.BarcodeFormat.RSS_EXPANDED ],
[Html5QrcodeSupportedFormats.UPC_A, ZXing.BarcodeFormat.UPC_A ],
[Html5QrcodeSupportedFormats.UPC_E, ZXing.BarcodeFormat.UPC_E ],
[
Html5QrcodeSupportedFormats.UPC_EAN_EXTENSION,
ZXing.BarcodeFormat.UPC_EAN_EXTENSION ]
]);
private readonly reverseFormatMap: Map<any, Html5QrcodeSupportedFormats>
= this.createReverseFormatMap();
private hints: Map<any, any>;
private verbose: boolean;
private logger: Logger;
public constructor(
requestedFormats: Array<Html5QrcodeSupportedFormats>,
verbose: boolean,
logger: Logger) {
if (!ZXing) {
throw "Use html5qrcode.min.js without edit, ZXing not found.";
}
this.verbose = verbose;
this.logger = logger;
const formats = this.createZXingFormats(requestedFormats);
const hints = new Map();
hints.set(ZXing.DecodeHintType.POSSIBLE_FORMATS, formats);
// TODO(minhazav): Make this configurable by developers.
hints.set(ZXing.DecodeHintType.TRY_HARDER, false);
this.hints = hints;
}
decodeAsync(canvas: HTMLCanvasElement): Promise<QrcodeResult> {
return new Promise((resolve, reject) => {
try {
resolve(this.decode(canvas));
} catch (error) {
reject(error);
}
});
}
private decode(canvas: HTMLCanvasElement): QrcodeResult {
// Note: Earlier we used to instantiate the zxingDecoder once as state
// of this class and use it for each scans. There seems to be some
// stateful bug in ZXing library around RSS_14 likely due to
// https://github.com/zxing-js/library/issues/211.
// Recreating a new instance per scan doesn't lead to performance issues
// and temporarily mitigates this issue.
// TODO(mebjas): Properly fix this issue in ZXing library.
const zxingDecoder = new ZXing.MultiFormatReader(
this.verbose, this.hints);
const luminanceSource
= new ZXing.HTMLCanvasElementLuminanceSource(canvas);
const binaryBitmap
= new ZXing.BinaryBitmap(
new ZXing.HybridBinarizer(luminanceSource));
let result = zxingDecoder.decode(binaryBitmap);
return {
text: result.text,
format: QrcodeResultFormat.create(
this.toHtml5QrcodeSupportedFormats(result.format)),
debugData: this.createDebugData()
};
}
private createReverseFormatMap(): Map<any, Html5QrcodeSupportedFormats> {
let result = new Map();
this.formatMap.forEach(
(value: any, key: Html5QrcodeSupportedFormats, _) => {
result.set(value, key);
});
return result;
}
private toHtml5QrcodeSupportedFormats(zxingFormat: any)
: Html5QrcodeSupportedFormats {
if (!this.reverseFormatMap.has(zxingFormat)) {
throw `reverseFormatMap doesn't have ${zxingFormat}`;
}
return this.reverseFormatMap.get(zxingFormat)!;
}
private createZXingFormats(
requestedFormats: Array<Html5QrcodeSupportedFormats>):
Array<any> {
let zxingFormats = [];
for (const requestedFormat of requestedFormats) {
if (this.formatMap.has(requestedFormat)) {
zxingFormats.push(
this.formatMap.get(requestedFormat));
} else {
this.logger.logError(`${requestedFormat} is not supported by`
+ "ZXingHtml5QrcodeShim");
}
}
return zxingFormats;
}
private createDebugData(): QrcodeResultDebugData {
return { decoderName: "zxing-js" };
}
}