272 lines
37 KiB
JavaScript
272 lines
37 KiB
JavaScript
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, ViewChild, } from "@angular/core";
|
|
import { toCanvas, toDataURL, toString, } from "qrcode";
|
|
import * as i0 from "@angular/core";
|
|
import * as i1 from "@angular/platform-browser";
|
|
class QRCodeComponent {
|
|
constructor(renderer, sanitizer) {
|
|
this.renderer = renderer;
|
|
this.sanitizer = sanitizer;
|
|
this.allowEmptyString = false;
|
|
this.colorDark = "#000000ff";
|
|
this.colorLight = "#ffffffff";
|
|
this.cssClass = "qrcode";
|
|
this.elementType = "canvas";
|
|
this.errorCorrectionLevel = "M";
|
|
this.margin = 4;
|
|
this.qrdata = "";
|
|
this.scale = 4;
|
|
this.width = 10;
|
|
this.qrCodeURL = new EventEmitter();
|
|
this.context = null;
|
|
}
|
|
async ngOnChanges() {
|
|
await this.createQRCode();
|
|
}
|
|
isValidQrCodeText(data) {
|
|
if (this.allowEmptyString === false) {
|
|
return !(typeof data === "undefined" ||
|
|
data === "" ||
|
|
data === "null" ||
|
|
data === null);
|
|
}
|
|
return !(typeof data === "undefined");
|
|
}
|
|
toDataURL(qrCodeConfig) {
|
|
return new Promise((resolve, reject) => {
|
|
toDataURL(this.qrdata, qrCodeConfig, (err, url) => {
|
|
if (err) {
|
|
reject(err);
|
|
}
|
|
else {
|
|
resolve(url);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
toCanvas(canvas, qrCodeConfig) {
|
|
return new Promise((resolve, reject) => {
|
|
toCanvas(canvas, this.qrdata, qrCodeConfig, (error) => {
|
|
if (error) {
|
|
reject(error);
|
|
}
|
|
else {
|
|
resolve("success");
|
|
}
|
|
});
|
|
});
|
|
}
|
|
toSVG(qrCodeConfig) {
|
|
return new Promise((resolve, reject) => {
|
|
toString(this.qrdata, qrCodeConfig, (err, url) => {
|
|
if (err) {
|
|
reject(err);
|
|
}
|
|
else {
|
|
resolve(url);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
renderElement(element) {
|
|
for (const node of this.qrcElement.nativeElement.childNodes) {
|
|
this.renderer.removeChild(this.qrcElement.nativeElement, node);
|
|
}
|
|
this.renderer.appendChild(this.qrcElement.nativeElement, element);
|
|
}
|
|
async createQRCode() {
|
|
if (this.version && this.version > 40) {
|
|
console.warn("[angularx-qrcode] max value for `version` is 40");
|
|
this.version = 40;
|
|
}
|
|
else if (this.version && this.version < 1) {
|
|
console.warn("[angularx-qrcode]`min value for `version` is 1");
|
|
this.version = 1;
|
|
}
|
|
else if (this.version !== undefined && isNaN(this.version)) {
|
|
console.warn("[angularx-qrcode] version should be a number, defaulting to auto.");
|
|
this.version = undefined;
|
|
}
|
|
try {
|
|
if (!this.isValidQrCodeText(this.qrdata)) {
|
|
throw new Error("[angularx-qrcode] Field `qrdata` is empty, set 'allowEmptyString=\"true\"' to overwrite this behaviour.");
|
|
}
|
|
if (this.isValidQrCodeText(this.qrdata) && this.qrdata === "") {
|
|
this.qrdata = " ";
|
|
}
|
|
const config = {
|
|
color: {
|
|
dark: this.colorDark,
|
|
light: this.colorLight,
|
|
},
|
|
errorCorrectionLevel: this.errorCorrectionLevel,
|
|
margin: this.margin,
|
|
scale: this.scale,
|
|
version: this.version,
|
|
width: this.width,
|
|
};
|
|
const centerImageSrc = this.imageSrc;
|
|
const centerImageHeight = this.imageHeight || 40;
|
|
const centerImageWidth = this.imageWidth || 40;
|
|
switch (this.elementType) {
|
|
case "canvas": {
|
|
const canvasElement = this.renderer.createElement("canvas");
|
|
this.context = canvasElement.getContext("2d");
|
|
this.toCanvas(canvasElement, config)
|
|
.then(() => {
|
|
if (this.ariaLabel) {
|
|
this.renderer.setAttribute(canvasElement, "aria-label", `${this.ariaLabel}`);
|
|
}
|
|
if (this.title) {
|
|
this.renderer.setAttribute(canvasElement, "title", `${this.title}`);
|
|
}
|
|
if (centerImageSrc && this.context) {
|
|
this.centerImage = new Image(centerImageWidth, centerImageHeight);
|
|
if (centerImageSrc !== this.centerImage.src) {
|
|
this.centerImage.src = centerImageSrc;
|
|
}
|
|
if (centerImageHeight !== this.centerImage.height) {
|
|
this.centerImage.height = centerImageHeight;
|
|
}
|
|
if (centerImageWidth !== this.centerImage.width) {
|
|
this.centerImage.width = centerImageWidth;
|
|
}
|
|
const centerImage = this.centerImage;
|
|
if (centerImage) {
|
|
centerImage.onload = () => {
|
|
this.context?.drawImage(centerImage, canvasElement.width / 2 - centerImageWidth / 2, canvasElement.height / 2 - centerImageHeight / 2, centerImageWidth, centerImageHeight);
|
|
};
|
|
}
|
|
}
|
|
this.renderElement(canvasElement);
|
|
this.emitQRCodeURL(canvasElement);
|
|
})
|
|
.catch((e) => {
|
|
console.error("[angularx-qrcode] canvas error:", e);
|
|
});
|
|
break;
|
|
}
|
|
case "svg": {
|
|
const svgParentElement = this.renderer.createElement("div");
|
|
this.toSVG(config)
|
|
.then((svgString) => {
|
|
this.renderer.setProperty(svgParentElement, "innerHTML", svgString);
|
|
const svgElement = svgParentElement.firstChild;
|
|
this.renderer.setAttribute(svgElement, "height", `${this.width}`);
|
|
this.renderer.setAttribute(svgElement, "width", `${this.width}`);
|
|
this.renderElement(svgElement);
|
|
this.emitQRCodeURL(svgElement);
|
|
})
|
|
.catch((e) => {
|
|
console.error("[angularx-qrcode] svg error:", e);
|
|
});
|
|
break;
|
|
}
|
|
case "url":
|
|
case "img":
|
|
default: {
|
|
const imgElement = this.renderer.createElement("img");
|
|
this.toDataURL(config)
|
|
.then((dataUrl) => {
|
|
if (this.alt) {
|
|
imgElement.setAttribute("alt", this.alt);
|
|
}
|
|
if (this.ariaLabel) {
|
|
imgElement.setAttribute("aria-label", this.ariaLabel);
|
|
}
|
|
imgElement.setAttribute("src", dataUrl);
|
|
if (this.title) {
|
|
imgElement.setAttribute("title", this.title);
|
|
}
|
|
this.renderElement(imgElement);
|
|
this.emitQRCodeURL(imgElement);
|
|
})
|
|
.catch((e) => {
|
|
console.error("[angularx-qrcode] img/url error:", e);
|
|
});
|
|
}
|
|
}
|
|
}
|
|
catch (e) {
|
|
console.error("[angularx-qrcode] Error generating QR Code:", e.message);
|
|
}
|
|
}
|
|
emitQRCodeURL(element) {
|
|
const className = element.constructor.name;
|
|
if (className === SVGSVGElement.name) {
|
|
const svgHTML = element.outerHTML;
|
|
const blob = new Blob([svgHTML], { type: "image/svg+xml" });
|
|
const urlSvg = URL.createObjectURL(blob);
|
|
const urlSanitized = this.sanitizer.bypassSecurityTrustUrl(urlSvg);
|
|
this.qrCodeURL.emit(urlSanitized);
|
|
return;
|
|
}
|
|
let urlImage = "";
|
|
if (className === HTMLCanvasElement.name) {
|
|
urlImage = element.toDataURL("image/png");
|
|
}
|
|
if (className === HTMLImageElement.name) {
|
|
urlImage = element.src;
|
|
}
|
|
fetch(urlImage)
|
|
.then((urlResponse) => urlResponse.blob())
|
|
.then((blob) => URL.createObjectURL(blob))
|
|
.then((url) => this.sanitizer.bypassSecurityTrustUrl(url))
|
|
.then((urlSanitized) => {
|
|
this.qrCodeURL.emit(urlSanitized);
|
|
})
|
|
.catch((error) => {
|
|
console.error("[angularx-qrcode] Error when fetching image/png URL: " + error);
|
|
});
|
|
}
|
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: QRCodeComponent, deps: [{ token: i0.Renderer2 }, { token: i1.DomSanitizer }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.6", type: QRCodeComponent, selector: "qrcode", inputs: { allowEmptyString: "allowEmptyString", colorDark: "colorDark", colorLight: "colorLight", cssClass: "cssClass", elementType: "elementType", errorCorrectionLevel: "errorCorrectionLevel", imageSrc: "imageSrc", imageHeight: "imageHeight", imageWidth: "imageWidth", margin: "margin", qrdata: "qrdata", scale: "scale", version: "version", width: "width", alt: "alt", ariaLabel: "ariaLabel", title: "title" }, outputs: { qrCodeURL: "qrCodeURL" }, viewQueries: [{ propertyName: "qrcElement", first: true, predicate: ["qrcElement"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: `<div #qrcElement [class]="cssClass"></div>`, isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
}
|
|
export { QRCodeComponent };
|
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: QRCodeComponent, decorators: [{
|
|
type: Component,
|
|
args: [{
|
|
selector: "qrcode",
|
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
template: `<div #qrcElement [class]="cssClass"></div>`,
|
|
}]
|
|
}], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i1.DomSanitizer }]; }, propDecorators: { allowEmptyString: [{
|
|
type: Input
|
|
}], colorDark: [{
|
|
type: Input
|
|
}], colorLight: [{
|
|
type: Input
|
|
}], cssClass: [{
|
|
type: Input
|
|
}], elementType: [{
|
|
type: Input
|
|
}], errorCorrectionLevel: [{
|
|
type: Input
|
|
}], imageSrc: [{
|
|
type: Input
|
|
}], imageHeight: [{
|
|
type: Input
|
|
}], imageWidth: [{
|
|
type: Input
|
|
}], margin: [{
|
|
type: Input
|
|
}], qrdata: [{
|
|
type: Input
|
|
}], scale: [{
|
|
type: Input
|
|
}], version: [{
|
|
type: Input
|
|
}], width: [{
|
|
type: Input
|
|
}], alt: [{
|
|
type: Input
|
|
}], ariaLabel: [{
|
|
type: Input
|
|
}], title: [{
|
|
type: Input
|
|
}], qrCodeURL: [{
|
|
type: Output
|
|
}], qrcElement: [{
|
|
type: ViewChild,
|
|
args: ["qrcElement", { static: true }]
|
|
}] } });
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angularx-qrcode.component.js","sourceRoot":"","sources":["../../../../projects/angularx-qrcode/src/lib/angularx-qrcode.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EAET,YAAY,EACZ,KAAK,EAEL,MAAM,EAEN,SAAS,GACV,MAAM,eAAe,CAAA;AAEtB,OAAO,EAKL,QAAQ,EACR,SAAS,EACT,QAAQ,GACT,MAAM,QAAQ,CAAA;;;AAGf,MAKa,eAAe;IA6B1B,YAAoB,QAAmB,EAAU,SAAuB;QAApD,aAAQ,GAAR,QAAQ,CAAW;QAAU,cAAS,GAAT,SAAS,CAAc;QA5BxD,qBAAgB,GAAG,KAAK,CAAA;QACxB,cAAS,GAAG,WAAW,CAAA;QACvB,eAAU,GAAG,WAAW,CAAA;QACxB,aAAQ,GAAG,QAAQ,CAAA;QACnB,gBAAW,GAAsB,QAAQ,CAAA;QAElD,yBAAoB,GAA+B,GAAG,CAAA;QAI7C,WAAM,GAAG,CAAC,CAAA;QACV,WAAM,GAAG,EAAE,CAAA;QACX,UAAK,GAAG,CAAC,CAAA;QAET,UAAK,GAAG,EAAE,CAAA;QAOhB,cAAS,GAAG,IAAI,YAAY,EAAW,CAAA;QAI1C,YAAO,GAAoC,IAAI,CAAA;IAGqB,CAAC;IAErE,KAAK,CAAC,WAAW;QACtB,MAAM,IAAI,CAAC,YAAY,EAAE,CAAA;IAC3B,CAAC;IAES,iBAAiB,CAAC,IAAmB;QAC7C,IAAI,IAAI,CAAC,gBAAgB,KAAK,KAAK,EAAE;YACnC,OAAO,CAAC,CACN,OAAO,IAAI,KAAK,WAAW;gBAC3B,IAAI,KAAK,EAAE;gBACX,IAAI,KAAK,MAAM;gBACf,IAAI,KAAK,IAAI,CACd,CAAA;SACF;QACD,OAAO,CAAC,CAAC,OAAO,IAAI,KAAK,WAAW,CAAC,CAAA;IACvC,CAAC;IAEO,SAAS,CAAC,YAAoC;QACpD,OAAO,IAAI,OAAO,CAChB,CACE,OAAwC,EACxC,MAAuC,EACvC,EAAE;YACF,SAAS,CACP,IAAI,CAAC,MAAM,EACX,YAAY,EACZ,CAAC,GAA6B,EAAE,GAAW,EAAE,EAAE;gBAC7C,IAAI,GAAG,EAAE;oBACP,MAAM,CAAC,GAAG,CAAC,CAAA;iBACZ;qBAAM;oBACL,OAAO,CAAC,GAAG,CAAC,CAAA;iBACb;YACH,CAAC,CACF,CAAA;QACH,CAAC,CACF,CAAA;IACH,CAAC;IAEO,QAAQ,CACd,MAAyB,EACzB,YAAoC;QAEpC,OAAO,IAAI,OAAO,CAChB,CACE,OAAwC,EACxC,MAAuC,EACvC,EAAE;YACF,QAAQ,CACN,MAAM,EACN,IAAI,CAAC,MAAM,EACX,YAAY,EACZ,CAAC,KAA+B,EAAE,EAAE;gBAClC,IAAI,KAAK,EAAE;oBACT,MAAM,CAAC,KAAK,CAAC,CAAA;iBACd;qBAAM;oBACL,OAAO,CAAC,SAAS,CAAC,CAAA;iBACnB;YACH,CAAC,CACF,CAAA;QACH,CAAC,CACF,CAAA;IACH,CAAC;IAEO,KAAK,CAAC,YAAmC;QAC/C,OAAO,IAAI,OAAO,CAChB,CACE,OAAwC,EACxC,MAAuC,EACvC,EAAE;YACF,QAAQ,CACN,IAAI,CAAC,MAAM,EACX,YAAY,EACZ,CAAC,GAA6B,EAAE,GAAW,EAAE,EAAE;gBAC7C,IAAI,GAAG,EAAE;oBACP,MAAM,CAAC,GAAG,CAAC,CAAA;iBACZ;qBAAM;oBACL,OAAO,CAAC,GAAG,CAAC,CAAA;iBACb;YACH,CAAC,CACF,CAAA;QACH,CAAC,CACF,CAAA;IACH,CAAC;IAEO,aAAa,CAAC,OAAgB;QACpC,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,UAAU,EAAE;YAC3D,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,IAAI,CAAC,CAAA;SAC/D;QACD,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,OAAO,CAAC,CAAA;IACnE,CAAC;IAEO,KAAK,CAAC,YAAY;QAExB,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,OAAO,GAAG,EAAE,EAAE;YACrC,OAAO,CAAC,IAAI,CAAC,iDAAiD,CAAC,CAAA;YAC/D,IAAI,CAAC,OAAO,GAAG,EAAE,CAAA;SAClB;aAAM,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,OAAO,GAAG,CAAC,EAAE;YAC3C,OAAO,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAA;YAC9D,IAAI,CAAC,OAAO,GAAG,CAAC,CAAA;SACjB;aAAM,IAAI,IAAI,CAAC,OAAO,KAAK,SAAS,IAAI,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YAC5D,OAAO,CAAC,IAAI,CACV,mEAAmE,CACpE,CAAA;YACD,IAAI,CAAC,OAAO,GAAG,SAAS,CAAA;SACzB;QAED,IAAI;YACF,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE;gBACxC,MAAM,IAAI,KAAK,CACb,yGAAyG,CAC1G,CAAA;aACF;YAGD,IAAI,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,MAAM,KAAK,EAAE,EAAE;gBAC7D,IAAI,CAAC,MAAM,GAAG,GAAG,CAAA;aAClB;YAED,MAAM,MAAM,GAAG;gBACb,KAAK,EAAE;oBACL,IAAI,EAAE,IAAI,CAAC,SAAS;oBACpB,KAAK,EAAE,IAAI,CAAC,UAAU;iBACvB;gBACD,oBAAoB,EAAE,IAAI,CAAC,oBAAoB;gBAC/C,MAAM,EAAE,IAAI,CAAC,MAAM;gBACnB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,OAAO,EAAE,IAAI,CAAC,OAAO;gBACrB,KAAK,EAAE,IAAI,CAAC,KAAK;aAClB,CAAA;YAED,MAAM,cAAc,GAAG,IAAI,CAAC,QAAQ,CAAA;YACpC,MAAM,iBAAiB,GAAG,IAAI,CAAC,WAAW,IAAI,EAAE,CAAA;YAChD,MAAM,gBAAgB,GAAG,IAAI,CAAC,UAAU,IAAI,EAAE,CAAA;YAE9C,QAAQ,IAAI,CAAC,WAAW,EAAE;gBACxB,KAAK,QAAQ,CAAC,CAAC;oBACb,MAAM,aAAa,GACjB,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAA;oBACvC,IAAI,CAAC,OAAO,GAAG,aAAa,CAAC,UAAU,CAAC,IAAI,CAAC,CAAA;oBAC7C,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;yBACjC,IAAI,CAAC,GAAG,EAAE;wBACT,IAAI,IAAI,CAAC,SAAS,EAAE;4BAClB,IAAI,CAAC,QAAQ,CAAC,YAAY,CACxB,aAAa,EACb,YAAY,EACZ,GAAG,IAAI,CAAC,SAAS,EAAE,CACpB,CAAA;yBACF;wBACD,IAAI,IAAI,CAAC,KAAK,EAAE;4BACd,IAAI,CAAC,QAAQ,CAAC,YAAY,CACxB,aAAa,EACb,OAAO,EACP,GAAG,IAAI,CAAC,KAAK,EAAE,CAChB,CAAA;yBACF;wBAED,IAAI,cAAc,IAAI,IAAI,CAAC,OAAO,EAAE;4BAClC,IAAI,CAAC,WAAW,GAAG,IAAI,KAAK,CAC1B,gBAAgB,EAChB,iBAAiB,CAClB,CAAA;4BAED,IAAI,cAAc,KAAK,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE;gCAC3C,IAAI,CAAC,WAAW,CAAC,GAAG,GAAG,cAAc,CAAA;6BACtC;4BAED,IAAI,iBAAiB,KAAK,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE;gCACjD,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,iBAAiB,CAAA;6BAC5C;4BAED,IAAI,gBAAgB,KAAK,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE;gCAC/C,IAAI,CAAC,WAAW,CAAC,KAAK,GAAG,gBAAgB,CAAA;6BAC1C;4BAED,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAA;4BAEpC,IAAI,WAAW,EAAE;gCACf,WAAW,CAAC,MAAM,GAAG,GAAG,EAAE;oCACxB,IAAI,CAAC,OAAO,EAAE,SAAS,CACrB,WAAW,EACX,aAAa,CAAC,KAAK,GAAG,CAAC,GAAG,gBAAgB,GAAG,CAAC,EAC9C,aAAa,CAAC,MAAM,GAAG,CAAC,GAAG,iBAAiB,GAAG,CAAC,EAChD,gBAAgB,EAChB,iBAAiB,CAClB,CAAA;gCACH,CAAC,CAAA;6BACF;yBACF;wBAED,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,CAAA;wBACjC,IAAI,CAAC,aAAa,CAAC,aAAkC,CAAC,CAAA;oBACxD,CAAC,CAAC;yBACD,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE;wBACX,OAAO,CAAC,KAAK,CAAC,iCAAiC,EAAE,CAAC,CAAC,CAAA;oBACrD,CAAC,CAAC,CAAA;oBACJ,MAAK;iBACN;gBACD,KAAK,KAAK,CAAC,CAAC;oBACV,MAAM,gBAAgB,GACpB,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC,CAAA;oBACpC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;yBACf,IAAI,CAAC,CAAC,SAAiB,EAAE,EAAE;wBAC1B,IAAI,CAAC,QAAQ,CAAC,WAAW,CACvB,gBAAgB,EAChB,WAAW,EACX,SAAS,CACV,CAAA;wBACD,MAAM,UAAU,GAAG,gBAAgB,CAAC,UAA2B,CAAA;wBAC/D,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,UAAU,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,CAAA;wBACjE,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,UAAU,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,CAAA;wBAChE,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAA;wBAC9B,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAA;oBAChC,CAAC,CAAC;yBACD,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE;wBACX,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,CAAC,CAAC,CAAA;oBAClD,CAAC,CAAC,CAAA;oBACJ,MAAK;iBACN;gBACD,KAAK,KAAK,CAAC;gBACX,KAAK,KAAK,CAAC;gBACX,OAAO,CAAC,CAAC;oBACP,MAAM,UAAU,GACd,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC,CAAA;oBACpC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC;yBACnB,IAAI,CAAC,CAAC,OAAe,EAAE,EAAE;wBACxB,IAAI,IAAI,CAAC,GAAG,EAAE;4BACZ,UAAU,CAAC,YAAY,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,CAAC,CAAA;yBACzC;wBACD,IAAI,IAAI,CAAC,SAAS,EAAE;4BAClB,UAAU,CAAC,YAAY,CAAC,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,CAAA;yBACtD;wBACD,UAAU,CAAC,YAAY,CAAC,KAAK,EAAE,OAAO,CAAC,CAAA;wBACvC,IAAI,IAAI,CAAC,KAAK,EAAE;4BACd,UAAU,CAAC,YAAY,CAAC,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,CAAA;yBAC7C;wBACD,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAA;wBAC9B,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAA;oBAChC,CAAC,CAAC;yBACD,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE;wBACX,OAAO,CAAC,KAAK,CAAC,kCAAkC,EAAE,CAAC,CAAC,CAAA;oBACtD,CAAC,CAAC,CAAA;iBACL;aACF;SACF;QAAC,OAAO,CAAa,EAAE;YACtB,OAAO,CAAC,KAAK,CAAC,6CAA6C,EAAE,CAAC,CAAC,OAAO,CAAC,CAAA;SACxE;IACH,CAAC;IAED,aAAa,CAAC,OAA6D;QACzE,MAAM,SAAS,GAAG,OAAO,CAAC,WAAW,CAAC,IAAI,CAAA;QAC1C,IAAI,SAAS,KAAK,aAAa,CAAC,IAAI,EAAE;YACpC,MAAM,OAAO,GAAG,OAAO,CAAC,SAAS,CAAA;YACjC,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,OAAO,CAAC,EAAE,EAAE,IAAI,EAAE,eAAe,EAAE,CAAC,CAAA;YAC3D,MAAM,MAAM,GAAG,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAA;YACxC,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,sBAAsB,CAAC,MAAM,CAAC,CAAA;YAClE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,YAAY,CAAC,CAAA;YACjC,OAAM;SACP;QAED,IAAI,QAAQ,GAAG,EAAE,CAAA;QAEjB,IAAI,SAAS,KAAK,iBAAiB,CAAC,IAAI,EAAE;YACxC,QAAQ,GAAI,OAA6B,CAAC,SAAS,CAAC,WAAW,CAAC,CAAA;SACjE;QAED,IAAI,SAAS,KAAK,gBAAgB,CAAC,IAAI,EAAE;YACvC,QAAQ,GAAI,OAA4B,CAAC,GAAG,CAAA;SAC7C;QAED,KAAK,CAAC,QAAQ,CAAC;aACZ,IAAI,CAAC,CAAC,WAAqB,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC;aACnD,IAAI,CAAC,CAAC,IAAU,EAAE,EAAE,CAAC,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;aAC/C,IAAI,CAAC,CAAC,GAAW,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,sBAAsB,CAAC,GAAG,CAAC,CAAC;aACjE,IAAI,CAAC,CAAC,YAAqB,EAAE,EAAE;YAC9B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,YAAY,CAAC,CAAA;QACnC,CAAC,CAAC;aACD,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;YACf,OAAO,CAAC,KAAK,CACX,uDAAuD,GAAG,KAAK,CAChE,CAAA;QACH,CAAC,CAAC,CAAA;IACN,CAAC;8GAvTU,eAAe;kGAAf,eAAe,4nBAFhB,4CAA4C;;SAE3C,eAAe;2FAAf,eAAe;kBAL3B,SAAS;mBAAC;oBACT,QAAQ,EAAE,QAAQ;oBAClB,eAAe,EAAE,uBAAuB,CAAC,MAAM;oBAC/C,QAAQ,EAAE,4CAA4C;iBACvD;2HAEiB,gBAAgB;sBAA/B,KAAK;gBACU,SAAS;sBAAxB,KAAK;gBACU,UAAU;sBAAzB,KAAK;gBACU,QAAQ;sBAAvB,KAAK;gBACU,WAAW;sBAA1B,KAAK;gBAEC,oBAAoB;sBAD1B,KAAK;gBAEU,QAAQ;sBAAvB,KAAK;gBACU,WAAW;sBAA1B,KAAK;gBACU,UAAU;sBAAzB,KAAK;gBACU,MAAM;sBAArB,KAAK;gBACU,MAAM;sBAArB,KAAK;gBACU,KAAK;sBAApB,KAAK;gBACU,OAAO;sBAAtB,KAAK;gBACU,KAAK;sBAApB,KAAK;gBAGU,GAAG;sBAAlB,KAAK;gBACU,SAAS;sBAAxB,KAAK;gBACU,KAAK;sBAApB,KAAK;gBAEI,SAAS;sBAAlB,MAAM;gBAE2C,UAAU;sBAA3D,SAAS;uBAAC,YAAY,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  Renderer2,\n  ViewChild,\n} from \"@angular/core\"\nimport { DomSanitizer, SafeUrl } from \"@angular/platform-browser\"\nimport {\n  QRCodeErrorCorrectionLevel,\n  QRCodeRenderersOptions,\n  QRCodeToDataURLOptions,\n  QRCodeToStringOptions,\n  toCanvas,\n  toDataURL,\n  toString,\n} from \"qrcode\"\nimport { QRCodeVersion, QRCodeElementType, FixMeLater } from \"./types\"\n\n@Component({\n  selector: \"qrcode\",\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  template: `<div #qrcElement [class]=\"cssClass\"></div>`,\n})\nexport class QRCodeComponent implements OnChanges {\n  @Input() public allowEmptyString = false\n  @Input() public colorDark = \"#000000ff\"\n  @Input() public colorLight = \"#ffffffff\"\n  @Input() public cssClass = \"qrcode\"\n  @Input() public elementType: QRCodeElementType = \"canvas\"\n  @Input()\n  public errorCorrectionLevel: QRCodeErrorCorrectionLevel = \"M\"\n  @Input() public imageSrc?: string\n  @Input() public imageHeight?: number\n  @Input() public imageWidth?: number\n  @Input() public margin = 4\n  @Input() public qrdata = \"\"\n  @Input() public scale = 4\n  @Input() public version?: QRCodeVersion\n  @Input() public width = 10\n\n  // Accessibility features introduced in 13.0.4+\n  @Input() public alt?: string\n  @Input() public ariaLabel?: string\n  @Input() public title?: string\n\n  @Output() qrCodeURL = new EventEmitter<SafeUrl>()\n\n  @ViewChild(\"qrcElement\", { static: true }) public qrcElement!: ElementRef\n\n  public context: CanvasRenderingContext2D | null = null\n  private centerImage?: HTMLImageElement\n\n  constructor(private renderer: Renderer2, private sanitizer: DomSanitizer) {}\n\n  public async ngOnChanges(): Promise<void> {\n    await this.createQRCode()\n  }\n\n  protected isValidQrCodeText(data: string | null): boolean {\n    if (this.allowEmptyString === false) {\n      return !(\n        typeof data === \"undefined\" ||\n        data === \"\" ||\n        data === \"null\" ||\n        data === null\n      )\n    }\n    return !(typeof data === \"undefined\")\n  }\n\n  private toDataURL(qrCodeConfig: QRCodeToDataURLOptions): Promise<FixMeLater> {\n    return new Promise(\n      (\n        resolve: (arg: FixMeLater) => FixMeLater,\n        reject: (arg: FixMeLater) => FixMeLater\n      ) => {\n        toDataURL(\n          this.qrdata,\n          qrCodeConfig,\n          (err: Error | null | undefined, url: string) => {\n            if (err) {\n              reject(err)\n            } else {\n              resolve(url)\n            }\n          }\n        )\n      }\n    )\n  }\n\n  private toCanvas(\n    canvas: HTMLCanvasElement,\n    qrCodeConfig: QRCodeRenderersOptions\n  ): Promise<FixMeLater> {\n    return new Promise(\n      (\n        resolve: (arg: FixMeLater) => FixMeLater,\n        reject: (arg: FixMeLater) => FixMeLater\n      ) => {\n        toCanvas(\n          canvas,\n          this.qrdata,\n          qrCodeConfig,\n          (error: Error | null | undefined) => {\n            if (error) {\n              reject(error)\n            } else {\n              resolve(\"success\")\n            }\n          }\n        )\n      }\n    )\n  }\n\n  private toSVG(qrCodeConfig: QRCodeToStringOptions): Promise<FixMeLater> {\n    return new Promise(\n      (\n        resolve: (arg: FixMeLater) => FixMeLater,\n        reject: (arg: FixMeLater) => FixMeLater\n      ) => {\n        toString(\n          this.qrdata,\n          qrCodeConfig,\n          (err: Error | null | undefined, url: string) => {\n            if (err) {\n              reject(err)\n            } else {\n              resolve(url)\n            }\n          }\n        )\n      }\n    )\n  }\n\n  private renderElement(element: Element): void {\n    for (const node of this.qrcElement.nativeElement.childNodes) {\n      this.renderer.removeChild(this.qrcElement.nativeElement, node)\n    }\n    this.renderer.appendChild(this.qrcElement.nativeElement, element)\n  }\n\n  private async createQRCode(): Promise<void> {\n    // Set sensitive defaults\n    if (this.version && this.version > 40) {\n      console.warn(\"[angularx-qrcode] max value for `version` is 40\")\n      this.version = 40\n    } else if (this.version && this.version < 1) {\n      console.warn(\"[angularx-qrcode]`min value for `version` is 1\")\n      this.version = 1\n    } else if (this.version !== undefined && isNaN(this.version)) {\n      console.warn(\n        \"[angularx-qrcode] version should be a number, defaulting to auto.\"\n      )\n      this.version = undefined\n    }\n\n    try {\n      if (!this.isValidQrCodeText(this.qrdata)) {\n        throw new Error(\n          \"[angularx-qrcode] Field `qrdata` is empty, set 'allowEmptyString=\\\"true\\\"' to overwrite this behaviour.\"\n        )\n      }\n\n      // This is a workaround to allow an empty string as qrdata\n      if (this.isValidQrCodeText(this.qrdata) && this.qrdata === \"\") {\n        this.qrdata = \" \"\n      }\n\n      const config = {\n        color: {\n          dark: this.colorDark,\n          light: this.colorLight,\n        },\n        errorCorrectionLevel: this.errorCorrectionLevel,\n        margin: this.margin,\n        scale: this.scale,\n        version: this.version,\n        width: this.width,\n      }\n\n      const centerImageSrc = this.imageSrc\n      const centerImageHeight = this.imageHeight || 40\n      const centerImageWidth = this.imageWidth || 40\n\n      switch (this.elementType) {\n        case \"canvas\": {\n          const canvasElement: HTMLCanvasElement =\n            this.renderer.createElement(\"canvas\")\n          this.context = canvasElement.getContext(\"2d\")\n          this.toCanvas(canvasElement, config)\n            .then(() => {\n              if (this.ariaLabel) {\n                this.renderer.setAttribute(\n                  canvasElement,\n                  \"aria-label\",\n                  `${this.ariaLabel}`\n                )\n              }\n              if (this.title) {\n                this.renderer.setAttribute(\n                  canvasElement,\n                  \"title\",\n                  `${this.title}`\n                )\n              }\n\n              if (centerImageSrc && this.context) {\n                this.centerImage = new Image(\n                  centerImageWidth,\n                  centerImageHeight\n                )\n\n                if (centerImageSrc !== this.centerImage.src) {\n                  this.centerImage.src = centerImageSrc\n                }\n\n                if (centerImageHeight !== this.centerImage.height) {\n                  this.centerImage.height = centerImageHeight\n                }\n\n                if (centerImageWidth !== this.centerImage.width) {\n                  this.centerImage.width = centerImageWidth\n                }\n\n                const centerImage = this.centerImage\n\n                if (centerImage) {\n                  centerImage.onload = () => {\n                    this.context?.drawImage(\n                      centerImage,\n                      canvasElement.width / 2 - centerImageWidth / 2,\n                      canvasElement.height / 2 - centerImageHeight / 2,\n                      centerImageWidth,\n                      centerImageHeight\n                    )\n                  }\n                }\n              }\n\n              this.renderElement(canvasElement)\n              this.emitQRCodeURL(canvasElement as HTMLCanvasElement)\n            })\n            .catch((e) => {\n              console.error(\"[angularx-qrcode] canvas error:\", e)\n            })\n          break\n        }\n        case \"svg\": {\n          const svgParentElement: HTMLElement =\n            this.renderer.createElement(\"div\")\n          this.toSVG(config)\n            .then((svgString: string) => {\n              this.renderer.setProperty(\n                svgParentElement,\n                \"innerHTML\",\n                svgString\n              )\n              const svgElement = svgParentElement.firstChild as SVGSVGElement\n              this.renderer.setAttribute(svgElement, \"height\", `${this.width}`)\n              this.renderer.setAttribute(svgElement, \"width\", `${this.width}`)\n              this.renderElement(svgElement)\n              this.emitQRCodeURL(svgElement)\n            })\n            .catch((e) => {\n              console.error(\"[angularx-qrcode] svg error:\", e)\n            })\n          break\n        }\n        case \"url\":\n        case \"img\":\n        default: {\n          const imgElement: HTMLImageElement =\n            this.renderer.createElement(\"img\")\n          this.toDataURL(config)\n            .then((dataUrl: string) => {\n              if (this.alt) {\n                imgElement.setAttribute(\"alt\", this.alt)\n              }\n              if (this.ariaLabel) {\n                imgElement.setAttribute(\"aria-label\", this.ariaLabel)\n              }\n              imgElement.setAttribute(\"src\", dataUrl)\n              if (this.title) {\n                imgElement.setAttribute(\"title\", this.title)\n              }\n              this.renderElement(imgElement)\n              this.emitQRCodeURL(imgElement)\n            })\n            .catch((e) => {\n              console.error(\"[angularx-qrcode] img/url error:\", e)\n            })\n        }\n      }\n    } catch (e: FixMeLater) {\n      console.error(\"[angularx-qrcode] Error generating QR Code:\", e.message)\n    }\n  }\n\n  emitQRCodeURL(element: HTMLCanvasElement | HTMLImageElement | SVGSVGElement) {\n    const className = element.constructor.name\n    if (className === SVGSVGElement.name) {\n      const svgHTML = element.outerHTML\n      const blob = new Blob([svgHTML], { type: \"image/svg+xml\" })\n      const urlSvg = URL.createObjectURL(blob)\n      const urlSanitized = this.sanitizer.bypassSecurityTrustUrl(urlSvg)\n      this.qrCodeURL.emit(urlSanitized)\n      return\n    }\n\n    let urlImage = \"\"\n\n    if (className === HTMLCanvasElement.name) {\n      urlImage = (element as HTMLCanvasElement).toDataURL(\"image/png\")\n    }\n\n    if (className === HTMLImageElement.name) {\n      urlImage = (element as HTMLImageElement).src\n    }\n\n    fetch(urlImage)\n      .then((urlResponse: Response) => urlResponse.blob())\n      .then((blob: Blob) => URL.createObjectURL(blob))\n      .then((url: string) => this.sanitizer.bypassSecurityTrustUrl(url))\n      .then((urlSanitized: SafeUrl) => {\n        this.qrCodeURL.emit(urlSanitized)\n      })\n      .catch((error) => {\n        console.error(\n          \"[angularx-qrcode] Error when fetching image/png URL: \" + error\n        )\n      })\n  }\n}\n"]}
|