ngx-captcha
Version:
Dynamic captcha (Google reCaptcha) implementation for Angular
127 lines • 16.9 kB
JavaScript
import { Injectable, NgZone } from "@angular/core";
import * as i0 from "@angular/core";
export class ScriptService {
constructor(zone) {
this.zone = zone;
this.scriptElemId = "ngx-catpcha-script";
/**
* Name of the global google recaptcha script
*/
this.windowGrecaptcha = "grecaptcha";
/**
* Name of enterpise property in the global google recaptcha script
*/
this.windowGrecaptchaEnterprise = "enterprise";
/**
* Name of the global callback
*/
this.windowOnLoadCallbackProperty = "ngx_captcha_onload_callback";
/**
* Name of the global callback for enterprise
*/
this.windowOnLoadEnterpriseCallbackProperty = "ngx_captcha_onload_enterprise_callback";
this.globalDomain = "recaptcha.net";
this.defaultDomain = "google.com";
this.enterpriseApi = "enterprise.js";
this.defaultApi = "api.js";
}
registerCaptchaScript(config, render, onLoad, language) {
if (this.grecaptchaScriptLoaded(config.useEnterprise)) {
// recaptcha script is already loaded
// just call the callback
if (config.useEnterprise) {
this.zone.run(() => {
onLoad(window[this.windowGrecaptcha][this.windowGrecaptchaEnterprise]);
});
}
else {
this.zone.run(() => {
onLoad(window[this.windowGrecaptcha]);
});
}
return;
}
// we need to patch the callback through global variable, otherwise callback is not accessible
// note: https://github.com/Enngage/ngx-captcha/issues/2
if (config.useEnterprise) {
window[this.getCallbackName(true)] = ((() => this.zone.run(onLoad.bind(this, window[this.windowGrecaptcha][this.windowGrecaptchaEnterprise]))));
}
else {
window[this.getCallbackName(false)] = ((() => this.zone.run(onLoad.bind(this, window[this.windowGrecaptcha]))));
}
// prepare script elem
const scriptElem = document.createElement("script");
scriptElem.id = this.scriptElemId;
scriptElem.innerHTML = "";
scriptElem.src = this.getCaptchaScriptUrl(config, render, language);
scriptElem.async = true;
scriptElem.defer = true;
// add script to header
document.getElementsByTagName("head")[0].appendChild(scriptElem);
}
cleanup() {
const elem = document.getElementById(this.scriptElemId);
if (elem) {
elem.remove();
}
window[this.getCallbackName()] = undefined;
window[this.windowGrecaptcha] = undefined;
}
/**
* Indicates if google recaptcha script is available and ready to be used
*/
grecaptchaScriptLoaded(useEnterprise) {
if (!window[this.getCallbackName(useEnterprise)] ||
!window[this.windowGrecaptcha]) {
return false;
}
else if (useEnterprise &&
window[this.windowGrecaptcha][this.windowGrecaptchaEnterprise]) {
return true;
// if only enterprise script is loaded we need to check some v3's method
}
else if (window[this.windowGrecaptcha].execute) {
return true;
}
return false;
}
/**
* Gets global callback name
* @param useEnterprise Optional flag for enterprise script
* @private
*/
getCallbackName(useEnterprise) {
return useEnterprise
? this.windowOnLoadEnterpriseCallbackProperty
: this.windowOnLoadCallbackProperty;
}
/**
* Gets language param used in script url
*/
getLanguageParam(hl) {
if (!hl) {
return "";
}
return `&hl=${hl}`;
}
/**
* Url to google api script
*/
getCaptchaScriptUrl(config, render, language) {
const domain = config.useGlobalDomain
? this.globalDomain
: this.defaultDomain;
const api = config.useEnterprise ? this.enterpriseApi : this.defaultApi;
const callback = this.getCallbackName(config.useEnterprise);
return `https://www.${domain}/recaptcha/${api}?onload=${callback}&render=${render}${this.getLanguageParam(language)}`;
}
}
/** @nocollapse */ ScriptService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: ScriptService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
/** @nocollapse */ ScriptService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: ScriptService, providedIn: "root" });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: ScriptService, decorators: [{
type: Injectable,
args: [{
providedIn: "root",
}]
}], ctorParameters: function () { return [{ type: i0.NgZone }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"script.service.js","sourceRoot":"","sources":["../../../../src/lib/services/script.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from \"@angular/core\";\r\nimport { RecaptchaConfiguration } from \"../models/recaptcha-configuration\";\r\n\r\n@Injectable({\r\n  providedIn: \"root\",\r\n})\r\nexport class ScriptService {\r\n  private readonly scriptElemId: string = \"ngx-catpcha-script\";\r\n\r\n  /**\r\n   * Name of the global google recaptcha script\r\n   */\r\n  protected readonly windowGrecaptcha = \"grecaptcha\";\r\n\r\n  /**\r\n   * Name of enterpise property in the global google recaptcha script\r\n   */\r\n  protected readonly windowGrecaptchaEnterprise = \"enterprise\";\r\n\r\n  /**\r\n   * Name of the global callback\r\n   */\r\n  protected readonly windowOnLoadCallbackProperty =\r\n    \"ngx_captcha_onload_callback\";\r\n\r\n  /**\r\n   * Name of the global callback for enterprise\r\n   */\r\n  protected readonly windowOnLoadEnterpriseCallbackProperty =\r\n    \"ngx_captcha_onload_enterprise_callback\";\r\n\r\n  protected readonly globalDomain: string = \"recaptcha.net\";\r\n\r\n  protected readonly defaultDomain: string = \"google.com\";\r\n\r\n  protected readonly enterpriseApi: string = \"enterprise.js\";\r\n\r\n  protected readonly defaultApi: string = \"api.js\";\r\n\r\n  constructor(protected zone: NgZone) {}\r\n\r\n  registerCaptchaScript(\r\n    config: RecaptchaConfiguration,\r\n    render: string,\r\n    onLoad: (grecaptcha: any) => void,\r\n    language?: string\r\n  ): void {\r\n    if (this.grecaptchaScriptLoaded(config.useEnterprise)) {\r\n      // recaptcha script is already loaded\r\n      // just call the callback\r\n      if (config.useEnterprise) {\r\n        this.zone.run(() => {\r\n          onLoad(\r\n            (window as any)[this.windowGrecaptcha][\r\n              this.windowGrecaptchaEnterprise\r\n            ]\r\n          );\r\n        });\r\n      } else {\r\n        this.zone.run(() => {\r\n          onLoad((window as any)[this.windowGrecaptcha]);\r\n        });\r\n      }\r\n      return;\r\n    }\r\n\r\n    // we need to patch the callback through global variable, otherwise callback is not accessible\r\n    // note: https://github.com/Enngage/ngx-captcha/issues/2\r\n    if (config.useEnterprise) {\r\n      (window as any)[this.getCallbackName(true)] = <any>(\r\n        (() =>\r\n          this.zone.run(\r\n            onLoad.bind(\r\n              this,\r\n              (window as any)[this.windowGrecaptcha][\r\n                this.windowGrecaptchaEnterprise\r\n              ]\r\n            )\r\n          ))\r\n      );\r\n    } else {\r\n      (window as any)[this.getCallbackName(false)] = <any>(\r\n        (() =>\r\n          this.zone.run(\r\n            onLoad.bind(this, (window as any)[this.windowGrecaptcha])\r\n          ))\r\n      );\r\n    }\r\n\r\n    // prepare script elem\r\n    const scriptElem = document.createElement(\"script\");\r\n    scriptElem.id = this.scriptElemId;\r\n    scriptElem.innerHTML = \"\";\r\n    scriptElem.src = this.getCaptchaScriptUrl(config, render, language);\r\n    scriptElem.async = true;\r\n    scriptElem.defer = true;\r\n\r\n    // add script to header\r\n    document.getElementsByTagName(\"head\")[0].appendChild(scriptElem);\r\n  }\r\n\r\n  cleanup(): void {\r\n    const elem = document.getElementById(this.scriptElemId);\r\n\r\n    if (elem) {\r\n      elem.remove();\r\n    }\r\n    (window as any)[this.getCallbackName()] = undefined;\r\n    (window as any)[this.windowGrecaptcha] = undefined;\r\n  }\r\n\r\n  /**\r\n   * Indicates if google recaptcha script is available and ready to be used\r\n   */\r\n  private grecaptchaScriptLoaded(useEnterprise?: boolean): boolean {\r\n    if (\r\n      !(window as any)[this.getCallbackName(useEnterprise)] ||\r\n      !(window as any)[this.windowGrecaptcha]\r\n    ) {\r\n      return false;\r\n    } else if (\r\n      useEnterprise &&\r\n      (window as any)[this.windowGrecaptcha][this.windowGrecaptchaEnterprise]\r\n    ) {\r\n      return true;\r\n      // if only enterprise script is loaded we need to check some v3's method\r\n    } else if ((window as any)[this.windowGrecaptcha].execute) {\r\n      return true;\r\n    }\r\n    return false;\r\n  }\r\n\r\n  /**\r\n   * Gets global callback name\r\n   * @param useEnterprise Optional flag for enterprise script\r\n   * @private\r\n   */\r\n  private getCallbackName(useEnterprise?: boolean): string {\r\n    return useEnterprise\r\n      ? this.windowOnLoadEnterpriseCallbackProperty\r\n      : this.windowOnLoadCallbackProperty;\r\n  }\r\n\r\n  /**\r\n   * Gets language param used in script url\r\n   */\r\n  private getLanguageParam(hl?: string): string {\r\n    if (!hl) {\r\n      return \"\";\r\n    }\r\n\r\n    return `&hl=${hl}`;\r\n  }\r\n\r\n  /**\r\n   * Url to google api script\r\n   */\r\n  private getCaptchaScriptUrl(\r\n    config: RecaptchaConfiguration,\r\n    render: string,\r\n    language?: string\r\n  ): string {\r\n    const domain = config.useGlobalDomain\r\n      ? this.globalDomain\r\n      : this.defaultDomain;\r\n    const api = config.useEnterprise ? this.enterpriseApi : this.defaultApi;\r\n    const callback = this.getCallbackName(config.useEnterprise);\r\n\r\n    return `https://www.${domain}/recaptcha/${api}?onload=${callback}&render=${render}${this.getLanguageParam(\r\n      language\r\n    )}`;\r\n  }\r\n}\r\n"]}