UNPKG

ngx-captcha

Version:

Dynamic captcha (Google reCaptcha) implementation for Angular

127 lines 16.9 kB
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"]}