@dsivd/prestations-ng 16.4.9-beta.1 → 16.4.9-beta.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -10,6 +10,10 @@
10
10
 
11
11
  ### Added
12
12
 
13
+ - [recaptcha.service.ts](projects/prestations-ng/src/sdk-recaptcha/recaptcha.service.ts)
14
+
15
+ - added `CAPTCHA_URL_TOKEN = new InjectionToken<string>` to be able to configure the captcha script url
16
+
13
17
  - [foehn-transmit-waiting-modal.component.ts](projects/prestations-ng/src/foehn-page/foehn-transmit-waiting-modal/foehn-transmit-waiting-modal.component.ts)
14
18
  - [foehn-transmit-waiting-modal.component.html](projects/prestations-ng/src/foehn-page/foehn-transmit-waiting-modal/foehn-transmit-waiting-modal.component.html)
15
19
 
@@ -21,6 +21,8 @@ export class SdkEpaymentComponent {
21
21
  this.platformFailureMessageDicoKey = 'epayment-form.errors.default';
22
22
  this.failureModalDisplayed = false;
23
23
  this.redirectModalDisplayed = false;
24
+ // Make it easier for testing purpose
25
+ this.window = window;
24
26
  this.failureMessageDicoKeyByCode = new Map([
25
27
  ['back', 'epayment-form.errors.back'],
26
28
  ['cancel', 'epayment-form.errors.cancel'],
@@ -44,7 +46,7 @@ export class SdkEpaymentComponent {
44
46
  // eslint-disable-next-line rxjs-angular/prefer-async-pipe
45
47
  .subscribe(redirectUrl => {
46
48
  this.redirectModalDisplayed = true;
47
- window.location.href = redirectUrl;
49
+ this.window.location.href = redirectUrl;
48
50
  });
49
51
  }
50
52
  handleFailure(failureCode) {
@@ -72,4 +74,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
72
74
  }], ctorParameters: function () { return [{ type: i1.EPaymentService }, { type: i2.ActivatedRoute }, { type: i3.GesdemHandlerService }, { type: i4.GesdemEventService }]; }, propDecorators: { baseUrl: [{
73
75
  type: Input
74
76
  }] } });
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sdk-epayment.component.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/sdk-epayment/sdk-epayment.component.ts","../../../../projects/prestations-ng/src/sdk-epayment/sdk-epayment.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAU,MAAM,eAAe,CAAC;AAEzD,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;;;;;;;;AAMvC;;;GAGG;AAOH,MAAM,OAAO,oBAAoB;IAgB7B,YACY,eAAgC,EACvB,KAAqB,EACrB,aAAmC,EACnC,kBAAsC;QAH/C,oBAAe,GAAf,eAAe,CAAiB;QACvB,UAAK,GAAL,KAAK,CAAgB;QACrB,kBAAa,GAAb,aAAa,CAAsB;QACnC,uBAAkB,GAAlB,kBAAkB,CAAoB;QAlB3D,YAAO,GAAG,MAAM,CAAC;QAEjB,kCAA6B,GAAG,8BAA8B,CAAC;QAC/D,0BAAqB,GAAG,KAAK,CAAC;QAE9B,2BAAsB,GAAG,KAAK,CAAC;QAEvB,gCAA2B,GAAG,IAAI,GAAG,CAAC;YAC1C,CAAC,MAAM,EAAE,2BAA2B,CAAC;YACrC,CAAC,QAAQ,EAAE,6BAA6B,CAAC;YACzC,CAAC,SAAS,EAAE,8BAA8B,CAAC;YAC3C,CAAC,OAAO,EAAE,oCAAoC,CAAC;SAClD,CAAC,CAAC;IAOA,CAAC;IAEJ,QAAQ;QACJ,2EAA2E;QAC3E,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;IAC/D,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,SAAiB;QACpB,IAAI,CAAC,eAAe;aACf,oBAAoB,CAAC,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC;YAC9C,0DAA0D;aACzD,SAAS,CAAC,WAAW,CAAC,EAAE;YACrB,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;YACnC,MAAM,CAAC,QAAQ,CAAC,IAAI,GAAG,WAAW,CAAC;QACvC,CAAC,CAAC,CAAC;IACX,CAAC;IAEO,aAAa,CAAC,WAAmB;QACrC,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE;YACtB,OAAO;SACV;QAED,IAAI,CAAC,6BAA6B;YAC9B,IAAI,CAAC,2BAA2B,CAAC,GAAG,CAAC,WAAW,CAAC;gBACjD,IAAI,CAAC,6BAA6B,CAAC;QAEvC,IAAI,CAAC,kBAAkB;aAClB,4BAA4B,EAAE;aAC9B,IAAI,CAAC,KAAK,EAAE,CAAC;YACd,0DAA0D;aACzD,SAAS,CAAC,GAAG,EAAE;YACZ,IAAI,CAAC,eAAe,CAAC,oBAAoB,CACrC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,EAC9C,IAAI,CAAC,OAAO,CACf,CAAC;QACN,CAAC,CAAC,CAAC;QAEP,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;IACtC,CAAC;;kHAjEQ,oBAAoB;sGAApB,oBAAoB,qFClBjC,+tCA6BA;4FDXa,oBAAoB;kBANhC,SAAS;+BAEI,eAAe;uMAMzB,OAAO;sBADN,KAAK","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\nimport { ActivatedRoute } from '@angular/router';\nimport { first } from 'rxjs/operators';\n\nimport { GesdemEventService } from '../gesdem/gesdem-event.service';\nimport { GesdemHandlerService } from '../gesdem/gesdem-handler.service';\nimport { EPaymentService } from './sdk-epayment.service';\n\n/**\n * Composant invisible à insérer dans une page SDK permettant d'effectuer des paiement sur la plateforme de paiement de\n * Postfinance. La méthode payNow() déclenche le paiement.\n */\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: 'epayment-form',\n    templateUrl: './sdk-epayment.component.html',\n    styleUrls: ['./sdk-epayment.component.css']\n})\nexport class SdkEpaymentComponent implements OnInit {\n    @Input()\n    baseUrl = 'api/';\n\n    platformFailureMessageDicoKey = 'epayment-form.errors.default';\n    failureModalDisplayed = false;\n\n    redirectModalDisplayed = false;\n\n    private failureMessageDicoKeyByCode = new Map([\n        ['back', 'epayment-form.errors.back'],\n        ['cancel', 'epayment-form.errors.cancel'],\n        ['refused', 'epayment-form.errors.refused'],\n        ['error', 'epayment-form.errors.unknown-error']\n    ]);\n\n    constructor(\n        private ePaymentService: EPaymentService,\n        private readonly route: ActivatedRoute,\n        private readonly gesDemService: GesdemHandlerService,\n        private readonly gesdemEventService: GesdemEventService\n    ) {}\n\n    ngOnInit(): void {\n        // En cas d'échec, la plateforme nous renvoie dans la page avec une raison.\n        this.handleFailure(this.route.snapshot.queryParams.REASON);\n    }\n\n    /**\n     * Déclenche le processus de paiement électronique. Le paiement est d'abord préparé avec GesDem, puis on navigue\n     * vers la plateforme de paiement.\n     *\n     * @param reference La référence de la demande en cours\n     */\n    payNow(reference: string): void {\n        this.ePaymentService\n            .createPendingPayment(reference, this.baseUrl)\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(redirectUrl => {\n                this.redirectModalDisplayed = true;\n                window.location.href = redirectUrl;\n            });\n    }\n\n    private handleFailure(failureCode: string): void {\n        if (!failureCode?.length) {\n            return;\n        }\n\n        this.platformFailureMessageDicoKey =\n            this.failureMessageDicoKeyByCode.get(failureCode) ||\n            this.platformFailureMessageDicoKey;\n\n        this.gesdemEventService\n            .formInitializationObservable()\n            .pipe(first())\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(() => {\n                this.ePaymentService.cancelPendingPayment(\n                    this.gesDemService.lastResponse.meta.reference,\n                    this.baseUrl\n                );\n            });\n\n        this.failureModalDisplayed = true;\n    }\n}\n","<foehn-modal\n    [id]=\"'epayment-form-error-modal'\"\n    [isModalVisible]=\"failureModalDisplayed\"\n    [modalHeaderText]=\"'epayment-form.errors.title' | fromDictionary\"\n    [modalBodyText]=\"platformFailureMessageDicoKey | fromDictionary\"\n    (click)=\"failureModalDisplayed = false\"\n></foehn-modal>\n\n<foehn-modal\n    [id]=\"'epayment-form-redirection-modal'\"\n    [isModalVisible]=\"redirectModalDisplayed\"\n    [modalHeaderText]=\"'epayment-form.redirection.title' | fromDictionary\"\n    [closeable]=\"false\"\n>\n    <!-- Need to add a tabindex=\"0\" (here on modal-body-content) otherwise cdkTrapFocus will not work because no closing button -->\n    <div class=\"modal-body-content\" tabindex=\"0\">\n        <p [innerHTML]=\"'epayment-form.redirection.text' | fromDictionary\"></p>\n        <div class=\"progress\" aria-hidden=\"true\">\n            <div\n                role=\"progressbar\"\n                class=\"progress-bar bg-success progress-bar-animated progress-bar-striped w-100\"\n            ></div>\n        </div>\n        <small\n            class=\"form-text text-secondary\"\n            [innerHTML]=\"'epayment-form.redirection.help' | fromDictionary\"\n        ></small>\n    </div>\n</foehn-modal>\n"]}
77
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sdk-epayment.component.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/sdk-epayment/sdk-epayment.component.ts","../../../../projects/prestations-ng/src/sdk-epayment/sdk-epayment.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAU,MAAM,eAAe,CAAC;AAEzD,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;;;;;;;;AAMvC;;;GAGG;AAOH,MAAM,OAAO,oBAAoB;IAmB7B,YACY,eAAgC,EACvB,KAAqB,EACrB,aAAmC,EACnC,kBAAsC;QAH/C,oBAAe,GAAf,eAAe,CAAiB;QACvB,UAAK,GAAL,KAAK,CAAgB;QACrB,kBAAa,GAAb,aAAa,CAAsB;QACnC,uBAAkB,GAAlB,kBAAkB,CAAoB;QArB3D,YAAO,GAAG,MAAM,CAAC;QAEjB,kCAA6B,GAAG,8BAA8B,CAAC;QAC/D,0BAAqB,GAAG,KAAK,CAAC;QAE9B,2BAAsB,GAAG,KAAK,CAAC;QAE/B,qCAAqC;QAC7B,WAAM,GAAG,MAAM,CAAC;QAEhB,gCAA2B,GAAG,IAAI,GAAG,CAAC;YAC1C,CAAC,MAAM,EAAE,2BAA2B,CAAC;YACrC,CAAC,QAAQ,EAAE,6BAA6B,CAAC;YACzC,CAAC,SAAS,EAAE,8BAA8B,CAAC;YAC3C,CAAC,OAAO,EAAE,oCAAoC,CAAC;SAClD,CAAC,CAAC;IAOA,CAAC;IAEJ,QAAQ;QACJ,2EAA2E;QAC3E,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;IAC/D,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,SAAiB;QACpB,IAAI,CAAC,eAAe;aACf,oBAAoB,CAAC,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC;YAC9C,0DAA0D;aACzD,SAAS,CAAC,WAAW,CAAC,EAAE;YACrB,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;YACnC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,GAAG,WAAW,CAAC;QAC5C,CAAC,CAAC,CAAC;IACX,CAAC;IAEO,aAAa,CAAC,WAAmB;QACrC,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE;YACtB,OAAO;SACV;QAED,IAAI,CAAC,6BAA6B;YAC9B,IAAI,CAAC,2BAA2B,CAAC,GAAG,CAAC,WAAW,CAAC;gBACjD,IAAI,CAAC,6BAA6B,CAAC;QAEvC,IAAI,CAAC,kBAAkB;aAClB,4BAA4B,EAAE;aAC9B,IAAI,CAAC,KAAK,EAAE,CAAC;YACd,0DAA0D;aACzD,SAAS,CAAC,GAAG,EAAE;YACZ,IAAI,CAAC,eAAe,CAAC,oBAAoB,CACrC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,EAC9C,IAAI,CAAC,OAAO,CACf,CAAC;QACN,CAAC,CAAC,CAAC;QAEP,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;IACtC,CAAC;;kHApEQ,oBAAoB;sGAApB,oBAAoB,qFClBjC,+tCA6BA;4FDXa,oBAAoB;kBANhC,SAAS;+BAEI,eAAe;uMAMzB,OAAO;sBADN,KAAK","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\nimport { ActivatedRoute } from '@angular/router';\nimport { first } from 'rxjs/operators';\n\nimport { GesdemEventService } from '../gesdem/gesdem-event.service';\nimport { GesdemHandlerService } from '../gesdem/gesdem-handler.service';\nimport { EPaymentService } from './sdk-epayment.service';\n\n/**\n * Composant invisible à insérer dans une page SDK permettant d'effectuer des paiement sur la plateforme de paiement de\n * Postfinance. La méthode payNow() déclenche le paiement.\n */\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: 'epayment-form',\n    templateUrl: './sdk-epayment.component.html',\n    styleUrls: ['./sdk-epayment.component.css']\n})\nexport class SdkEpaymentComponent implements OnInit {\n    @Input()\n    baseUrl = 'api/';\n\n    platformFailureMessageDicoKey = 'epayment-form.errors.default';\n    failureModalDisplayed = false;\n\n    redirectModalDisplayed = false;\n\n    // Make it easier for testing purpose\n    private window = window;\n\n    private failureMessageDicoKeyByCode = new Map([\n        ['back', 'epayment-form.errors.back'],\n        ['cancel', 'epayment-form.errors.cancel'],\n        ['refused', 'epayment-form.errors.refused'],\n        ['error', 'epayment-form.errors.unknown-error']\n    ]);\n\n    constructor(\n        private ePaymentService: EPaymentService,\n        private readonly route: ActivatedRoute,\n        private readonly gesDemService: GesdemHandlerService,\n        private readonly gesdemEventService: GesdemEventService\n    ) {}\n\n    ngOnInit(): void {\n        // En cas d'échec, la plateforme nous renvoie dans la page avec une raison.\n        this.handleFailure(this.route.snapshot.queryParams.REASON);\n    }\n\n    /**\n     * Déclenche le processus de paiement électronique. Le paiement est d'abord préparé avec GesDem, puis on navigue\n     * vers la plateforme de paiement.\n     *\n     * @param reference La référence de la demande en cours\n     */\n    payNow(reference: string): void {\n        this.ePaymentService\n            .createPendingPayment(reference, this.baseUrl)\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(redirectUrl => {\n                this.redirectModalDisplayed = true;\n                this.window.location.href = redirectUrl;\n            });\n    }\n\n    private handleFailure(failureCode: string): void {\n        if (!failureCode?.length) {\n            return;\n        }\n\n        this.platformFailureMessageDicoKey =\n            this.failureMessageDicoKeyByCode.get(failureCode) ||\n            this.platformFailureMessageDicoKey;\n\n        this.gesdemEventService\n            .formInitializationObservable()\n            .pipe(first())\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(() => {\n                this.ePaymentService.cancelPendingPayment(\n                    this.gesDemService.lastResponse.meta.reference,\n                    this.baseUrl\n                );\n            });\n\n        this.failureModalDisplayed = true;\n    }\n}\n","<foehn-modal\n    [id]=\"'epayment-form-error-modal'\"\n    [isModalVisible]=\"failureModalDisplayed\"\n    [modalHeaderText]=\"'epayment-form.errors.title' | fromDictionary\"\n    [modalBodyText]=\"platformFailureMessageDicoKey | fromDictionary\"\n    (click)=\"failureModalDisplayed = false\"\n></foehn-modal>\n\n<foehn-modal\n    [id]=\"'epayment-form-redirection-modal'\"\n    [isModalVisible]=\"redirectModalDisplayed\"\n    [modalHeaderText]=\"'epayment-form.redirection.title' | fromDictionary\"\n    [closeable]=\"false\"\n>\n    <!-- Need to add a tabindex=\"0\" (here on modal-body-content) otherwise cdkTrapFocus will not work because no closing button -->\n    <div class=\"modal-body-content\" tabindex=\"0\">\n        <p [innerHTML]=\"'epayment-form.redirection.text' | fromDictionary\"></p>\n        <div class=\"progress\" aria-hidden=\"true\">\n            <div\n                role=\"progressbar\"\n                class=\"progress-bar bg-success progress-bar-animated progress-bar-striped w-100\"\n            ></div>\n        </div>\n        <small\n            class=\"form-text text-secondary\"\n            [innerHTML]=\"'epayment-form.redirection.help' | fromDictionary\"\n        ></small>\n    </div>\n</foehn-modal>\n"]}
@@ -1,6 +1,6 @@
1
1
  import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
- export const RECAPTCHA_URL = 'https://www.recaptcha.net/recaptcha/api.js?onload=reCaptchaLoad&render=explicit';
3
+ import * as i1 from "../recaptcha.service";
4
4
  export const SCRIPT_ID = 'prestations-ng-captcha-script';
5
5
  /**
6
6
  * Integration with Google Recaptcha.
@@ -10,7 +10,8 @@ export const SCRIPT_ID = 'prestations-ng-captcha-script';
10
10
  *
11
11
  */
12
12
  export class GrecaptchaComponent {
13
- constructor() {
13
+ constructor(recaptchaService) {
14
+ this.recaptchaService = recaptchaService;
14
15
  this.loaded = new EventEmitter();
15
16
  this.scriptError = new EventEmitter();
16
17
  this.tokenOnSuccess = new EventEmitter();
@@ -37,6 +38,7 @@ export class GrecaptchaComponent {
37
38
  }
38
39
  const config = {
39
40
  sitekey: this.siteKey,
41
+ theme: 'light',
40
42
  callback: (token) => this.tokenOnSuccess.emit(token),
41
43
  'expired-callback': () => this.reset()
42
44
  };
@@ -46,7 +48,7 @@ export class GrecaptchaComponent {
46
48
  appendScriptTag() {
47
49
  const script = document.createElement('script');
48
50
  script.id = SCRIPT_ID;
49
- script.src = RECAPTCHA_URL;
51
+ script.src = this.recaptchaService.getCaptchaScriptUrl();
50
52
  script.async = true;
51
53
  script.defer = true;
52
54
  script.onerror = () => this.scriptError.emit(true);
@@ -83,12 +85,12 @@ export class GrecaptchaComponent {
83
85
  }
84
86
  }
85
87
  }
86
- GrecaptchaComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GrecaptchaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
88
+ GrecaptchaComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GrecaptchaComponent, deps: [{ token: i1.RecaptchaService }], target: i0.ɵɵFactoryTarget.Component });
87
89
  GrecaptchaComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: GrecaptchaComponent, selector: "app-grecaptcha", inputs: { siteKey: "siteKey" }, outputs: { loaded: "loaded", scriptError: "scriptError", tokenOnSuccess: "tokenOnSuccess", destroyed: "destroyed" }, viewQueries: [{ propertyName: "recaptchaContainer", first: true, predicate: ["recaptchaContainer"], descendants: true, static: true }, { propertyName: "recaptchaScriptContainer", first: true, predicate: ["recaptchaScriptContainer"], descendants: true, static: true }], ngImport: i0, template: "<div #recaptchaScriptContainer></div>\n<div #recaptchaContainer></div>\n" });
88
90
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GrecaptchaComponent, decorators: [{
89
91
  type: Component,
90
92
  args: [{ selector: 'app-grecaptcha', template: "<div #recaptchaScriptContainer></div>\n<div #recaptchaContainer></div>\n" }]
91
- }], propDecorators: { siteKey: [{
93
+ }], ctorParameters: function () { return [{ type: i1.RecaptchaService }]; }, propDecorators: { siteKey: [{
92
94
  type: Input
93
95
  }], loaded: [{
94
96
  type: Output
@@ -105,4 +107,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
105
107
  type: ViewChild,
106
108
  args: ['recaptchaScriptContainer', { static: true }]
107
109
  }] } });
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"grecaptcha.component.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/sdk-recaptcha/grecaptcha/grecaptcha.component.ts","../../../../../projects/prestations-ng/src/sdk-recaptcha/grecaptcha/grecaptcha.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,SAAS,EAET,YAAY,EACZ,KAAK,EAGL,MAAM,EACN,SAAS,EACZ,MAAM,eAAe,CAAC;;AAEvB,MAAM,CAAC,MAAM,aAAa,GACtB,iFAAiF,CAAC;AACtF,MAAM,CAAC,MAAM,SAAS,GAAG,+BAA+B,CAAC;AAUzD;;;;;;GAMG;AAMH,MAAM,OAAO,mBAAmB;IALhC;QAUI,WAAM,GAAG,IAAI,YAAY,EAAW,CAAC;QAGrC,gBAAW,GAAG,IAAI,YAAY,EAAW,CAAC;QAG1C,mBAAc,GAAG,IAAI,YAAY,EAAiB,CAAC;QAGnD,cAAS,GAAG,IAAI,YAAY,EAAW,CAAC;KA8F3C;IApFG,QAAQ;QACJ,0EAA0E;QAC1E,IAAI,CAAC,WAAW,EAAE,CAAC;IACvB,CAAC;IAED,WAAW;QACP,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;IACxB,CAAC;IAED,KAAK;QACD,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAExB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACnC,CAAC;IAEO,aAAa;QACjB,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACf,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,CAAC;SACvD;QAED,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE;YACpB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;SAC/D;QAED,MAAM,MAAM,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,QAAQ,EAAE,CAAC,KAAa,EAAE,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC;YAC5D,kBAAkB,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE;SACzC,CAAC;QAEF,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,UAAU,CAAC,MAAM,CACpC,IAAI,CAAC,kBAAkB,CAAC,aAAa,EACrC,MAAM,CACT,CAAC;QAEF,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC3B,CAAC;IAEO,eAAe;QACnB,MAAM,MAAM,GAAG,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QAChD,MAAM,CAAC,EAAE,GAAG,SAAS,CAAC;QACtB,MAAM,CAAC,GAAG,GAAG,aAAa,CAAC;QAC3B,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC;QACpB,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC;QACpB,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACnD,IAAI,CAAC,wBAAwB,CAAC,aAAa,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;IACpE,CAAC;IAEO,YAAY;QAChB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAE7B,MAAM,CAAC,aAAa,GAAG,IAAI,CAAC;IAChC,CAAC;IAEO,WAAW;QACf,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAExB,wCAAwC;QACxC,MAAM,WAAW,GAAG,QAAQ,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC;QACvD,IAAI,WAAW,EAAE;YACb,IAAI,CAAC,aAAa,EAAE,CAAC;YACrB,OAAO;SACV;QAED,4CAA4C;QAC5C,MAAM,CAAC,aAAa,GAAG,GAAG,EAAE;YACxB,IAAI,CAAC,aAAa,EAAE,CAAC;QACzB,CAAC,CAAC;QAEF,kEAAkE;QAClE,IAAI,CAAC,eAAe,EAAE,CAAC;IAC3B,CAAC;IAEO,gBAAgB;QACpB,IAAI;YACA,IAAI,CAAC,CAAC,MAAM,CAAC,UAAU,EAAE;gBACrB,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC1C;SACJ;QAAC,OAAO,CAAC,EAAE;YACR,2DAA2D;YAC3D,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;SACpB;IACL,CAAC;;iHA3GQ,mBAAmB;qGAAnB,mBAAmB,wdCnChC,0EAEA;4FDiCa,mBAAmB;kBAL/B,SAAS;+BAEI,gBAAgB;8BAK1B,OAAO;sBADN,KAAK;gBAIN,MAAM;sBADL,MAAM;gBAIP,WAAW;sBADV,MAAM;gBAIP,cAAc;sBADb,MAAM;gBAIP,SAAS;sBADR,MAAM;gBAIP,kBAAkB;sBADjB,SAAS;uBAAC,oBAAoB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBAIjD,wBAAwB;sBADvB,SAAS;uBAAC,0BAA0B,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    ViewChild\n} from '@angular/core';\n\nexport const RECAPTCHA_URL =\n    'https://www.recaptcha.net/recaptcha/api.js?onload=reCaptchaLoad&render=explicit';\nexport const SCRIPT_ID = 'prestations-ng-captcha-script';\n\ndeclare global {\n    interface Window {\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        grecaptcha: any;\n        reCaptchaLoad: () => void;\n    }\n}\n\n/**\n * Integration with Google Recaptcha.\n *\n * All the quirks and weirdness of this integration should be done in this\n * component.\n *\n */\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: 'app-grecaptcha',\n    templateUrl: './grecaptcha.component.html'\n})\nexport class GrecaptchaComponent implements OnInit, OnDestroy {\n    @Input()\n    siteKey: string;\n\n    @Output()\n    loaded = new EventEmitter<boolean>();\n\n    @Output()\n    scriptError = new EventEmitter<boolean>();\n\n    @Output()\n    tokenOnSuccess = new EventEmitter<string | null>();\n\n    @Output()\n    destroyed = new EventEmitter<boolean>();\n\n    @ViewChild('recaptchaContainer', { static: true })\n    recaptchaContainer: ElementRef;\n\n    @ViewChild('recaptchaScriptContainer', { static: true })\n    recaptchaScriptContainer: ElementRef;\n\n    private widgetId: number;\n\n    ngOnInit(): void {\n        // If the component is in the component tree, load the captcha regardless.\n        this.loadCaptcha();\n    }\n\n    ngOnDestroy(): void {\n        this.destroyed.emit(true);\n        this.removeScript();\n    }\n\n    reset(): void {\n        this.safeCaptchaReset();\n\n        this.tokenOnSuccess.emit(null);\n    }\n\n    private renderCaptcha(): void {\n        if (!this.siteKey) {\n            throw new Error('Missing siteKey on the component');\n        }\n\n        if (!window.grecaptcha) {\n            throw new Error('The captcha has not been loaded properly');\n        }\n\n        const config = {\n            sitekey: this.siteKey,\n            callback: (token: string) => this.tokenOnSuccess.emit(token),\n            'expired-callback': () => this.reset()\n        };\n\n        this.widgetId = window.grecaptcha.render(\n            this.recaptchaContainer.nativeElement,\n            config\n        );\n\n        this.loaded.emit(true);\n    }\n\n    private appendScriptTag(): void {\n        const script = document.createElement('script');\n        script.id = SCRIPT_ID;\n        script.src = RECAPTCHA_URL;\n        script.async = true;\n        script.defer = true;\n        script.onerror = () => this.scriptError.emit(true);\n        this.recaptchaScriptContainer.nativeElement.appendChild(script);\n    }\n\n    private removeScript(): void {\n        this.scriptError.emit(false);\n\n        window.reCaptchaLoad = null;\n    }\n\n    private loadCaptcha(): void {\n        this.loaded.emit(false);\n\n        // In case the script is already loaded.\n        const existingTag = document.getElementById(SCRIPT_ID);\n        if (existingTag) {\n            this.renderCaptcha();\n            return;\n        }\n\n        // This will be called if the captcha loads.\n        window.reCaptchaLoad = () => {\n            this.renderCaptcha();\n        };\n\n        // Add the script with the `reCaptchaLoad` function as a callback.\n        this.appendScriptTag();\n    }\n\n    private safeCaptchaReset(): void {\n        try {\n            if (!!window.grecaptcha) {\n                window.grecaptcha.reset(this.widgetId);\n            }\n        } catch (e) {\n            // In case the reset of the Google Captcha throws an error.\n            console.error(e);\n        }\n    }\n}\n","<div #recaptchaScriptContainer></div>\n<div #recaptchaContainer></div>\n"]}
110
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"grecaptcha.component.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/sdk-recaptcha/grecaptcha/grecaptcha.component.ts","../../../../../projects/prestations-ng/src/sdk-recaptcha/grecaptcha/grecaptcha.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    ViewChild\n} from '@angular/core';\n\nimport { RecaptchaService } from '../recaptcha.service';\n\nexport const SCRIPT_ID = 'prestations-ng-captcha-script';\n\ndeclare global {\n    interface Window {\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        grecaptcha: any;\n        reCaptchaLoad: () => void;\n    }\n}\n\n/**\n * Integration with Google Recaptcha.\n *\n * All the quirks and weirdness of this integration should be done in this\n * component.\n *\n */\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: 'app-grecaptcha',\n    templateUrl: './grecaptcha.component.html'\n})\nexport class GrecaptchaComponent implements OnInit, OnDestroy {\n    @Input()\n    siteKey: string;\n\n    @Output()\n    loaded = new EventEmitter<boolean>();\n\n    @Output()\n    scriptError = new EventEmitter<boolean>();\n\n    @Output()\n    tokenOnSuccess = new EventEmitter<string | null>();\n\n    @Output()\n    destroyed = new EventEmitter<boolean>();\n\n    @ViewChild('recaptchaContainer', { static: true })\n    recaptchaContainer: ElementRef;\n\n    @ViewChild('recaptchaScriptContainer', { static: true })\n    recaptchaScriptContainer: ElementRef;\n\n    private widgetId: number;\n\n    constructor(private recaptchaService: RecaptchaService) {}\n\n    ngOnInit(): void {\n        // If the component is in the component tree, load the captcha regardless.\n        this.loadCaptcha();\n    }\n\n    ngOnDestroy(): void {\n        this.destroyed.emit(true);\n        this.removeScript();\n    }\n\n    reset(): void {\n        this.safeCaptchaReset();\n\n        this.tokenOnSuccess.emit(null);\n    }\n\n    private renderCaptcha(): void {\n        if (!this.siteKey) {\n            throw new Error('Missing siteKey on the component');\n        }\n\n        if (!window.grecaptcha) {\n            throw new Error('The captcha has not been loaded properly');\n        }\n\n        const config = {\n            sitekey: this.siteKey,\n            theme: 'light',\n            callback: (token: string) => this.tokenOnSuccess.emit(token),\n            'expired-callback': () => this.reset()\n        };\n\n        this.widgetId = window.grecaptcha.render(\n            this.recaptchaContainer.nativeElement,\n            config\n        );\n\n        this.loaded.emit(true);\n    }\n\n    private appendScriptTag(): void {\n        const script = document.createElement('script');\n        script.id = SCRIPT_ID;\n        script.src = this.recaptchaService.getCaptchaScriptUrl();\n        script.async = true;\n        script.defer = true;\n        script.onerror = () => this.scriptError.emit(true);\n        this.recaptchaScriptContainer.nativeElement.appendChild(script);\n    }\n\n    private removeScript(): void {\n        this.scriptError.emit(false);\n\n        window.reCaptchaLoad = null;\n    }\n\n    private loadCaptcha(): void {\n        this.loaded.emit(false);\n\n        // In case the script is already loaded.\n        const existingTag = document.getElementById(SCRIPT_ID);\n        if (existingTag) {\n            this.renderCaptcha();\n            return;\n        }\n\n        // This will be called if the captcha loads.\n        window.reCaptchaLoad = () => {\n            this.renderCaptcha();\n        };\n\n        // Add the script with the `reCaptchaLoad` function as a callback.\n        this.appendScriptTag();\n    }\n\n    private safeCaptchaReset(): void {\n        try {\n            if (!!window.grecaptcha) {\n                window.grecaptcha.reset(this.widgetId);\n            }\n        } catch (e) {\n            // In case the reset of the Google Captcha throws an error.\n            console.error(e);\n        }\n    }\n}\n","<div #recaptchaScriptContainer></div>\n<div #recaptchaContainer></div>\n"]}
@@ -1,5 +1,5 @@
1
1
  import { HttpParams } from '@angular/common/http';
2
- import { Injectable } from '@angular/core';
2
+ import { Inject, Injectable, InjectionToken } from '@angular/core';
3
3
  import { BehaviorSubject, combineLatest, EMPTY, merge, of } from 'rxjs';
4
4
  import { catchError, debounceTime, map, tap } from 'rxjs/operators';
5
5
  import * as i0 from "@angular/core";
@@ -13,8 +13,12 @@ import * as i7 from "../sdk-appinfo/application-info.service";
13
13
  import * as i8 from "../sdk-redirect/iam-expired-interceptor.service";
14
14
  export const RECAPTCHA_API_URL = 'api/recaptcha';
15
15
  export const CAPTCHA_ERROR_NAME = 'recaptcha';
16
+ export const CAPTCHA_URL_TOKEN = new InjectionToken('captchaScriptUrl', {
17
+ providedIn: 'root',
18
+ factory: () => 'https://www.recaptcha.net/recaptcha/api.js?onload=reCaptchaLoad&render=explicit'
19
+ });
16
20
  export class RecaptchaService {
17
- constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService, iamExpiredInterceptorService) {
21
+ constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService, iamExpiredInterceptorService, captchaScriptUrl) {
18
22
  this.http = http;
19
23
  this.sessionInfo = sessionInfo;
20
24
  this.route = route;
@@ -23,6 +27,7 @@ export class RecaptchaService {
23
27
  this.gesdemEventService = gesdemEventService;
24
28
  this.applicationInfoService = applicationInfoService;
25
29
  this.iamExpiredInterceptorService = iamExpiredInterceptorService;
30
+ this.captchaScriptUrl = captchaScriptUrl;
26
31
  this.errorsSubject = new BehaviorSubject([]);
27
32
  this.errors = this.initErrorObservable(); // has to be first since it is used in initShouldDisplayObservable !
28
33
  this.shouldDisplay = this.initShouldDisplayObservable();
@@ -86,6 +91,9 @@ export class RecaptchaService {
86
91
  getToken() {
87
92
  return this.token;
88
93
  }
94
+ getCaptchaScriptUrl() {
95
+ return this.captchaScriptUrl;
96
+ }
89
97
  initErrorObservable() {
90
98
  const matchingGesdemErrors = merge(
91
99
  // Ensures that there's an initial value
@@ -170,12 +178,15 @@ export class RecaptchaService {
170
178
  return captcha.publickey || null;
171
179
  }
172
180
  }
173
- RecaptchaService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, deps: [{ token: i1.HttpClient }, { token: i2.SessionInfo }, { token: i3.ActivatedRoute }, { token: i4.ValidationHandlerService }, { token: i5.GesdemHandlerService }, { token: i6.GesdemEventService }, { token: i7.ApplicationInfoService }, { token: i8.IamExpiredInterceptorService }], target: i0.ɵɵFactoryTarget.Injectable });
181
+ RecaptchaService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, deps: [{ token: i1.HttpClient }, { token: i2.SessionInfo }, { token: i3.ActivatedRoute }, { token: i4.ValidationHandlerService }, { token: i5.GesdemHandlerService }, { token: i6.GesdemEventService }, { token: i7.ApplicationInfoService }, { token: i8.IamExpiredInterceptorService }, { token: CAPTCHA_URL_TOKEN }], target: i0.ɵɵFactoryTarget.Injectable });
174
182
  RecaptchaService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, providedIn: 'root' });
175
183
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, decorators: [{
176
184
  type: Injectable,
177
185
  args: [{
178
186
  providedIn: 'root'
179
187
  }]
180
- }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.SessionInfo }, { type: i3.ActivatedRoute }, { type: i4.ValidationHandlerService }, { type: i5.GesdemHandlerService }, { type: i6.GesdemEventService }, { type: i7.ApplicationInfoService }, { type: i8.IamExpiredInterceptorService }]; } });
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recaptcha.service.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/sdk-recaptcha/recaptcha.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpParams } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { ActivatedRoute } from '@angular/router';\nimport {\n    BehaviorSubject,\n    combineLatest,\n    EMPTY,\n    merge,\n    Observable,\n    of\n} from 'rxjs';\nimport { catchError, debounceTime, map, tap } from 'rxjs/operators';\n\nimport { FormError } from '../form-error';\nimport { GesdemEventService } from '../gesdem/gesdem-event.service';\nimport { GesdemHandlerService } from '../gesdem/gesdem-handler.service';\nimport { ApplicationInfo } from '../sdk-appinfo/application-info';\nimport { ApplicationInfoService } from '../sdk-appinfo/application-info.service';\nimport { IamExpiredInterceptorService } from '../sdk-redirect/iam-expired-interceptor.service';\nimport { SessionInfo } from '../sdk-session-info/session-info.service';\nimport { ValidationHandlerService } from '../validation/validation-handler.service';\n\nexport const RECAPTCHA_API_URL = 'api/recaptcha';\nexport const CAPTCHA_ERROR_NAME = 'recaptcha';\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class RecaptchaService {\n    private readonly errors: Observable<FormError[]>;\n    private readonly errorsSubject = new BehaviorSubject<FormError[]>([]);\n    private readonly shouldDisplay: Observable<boolean>;\n    private readonly publicKey: Observable<string>;\n    private token: string;\n\n    constructor(\n        private http: HttpClient,\n        private sessionInfo: SessionInfo,\n        private route: ActivatedRoute,\n        private validationHandlerService: ValidationHandlerService,\n        private gesdemService: GesdemHandlerService,\n        private gesdemEventService: GesdemEventService,\n        private applicationInfoService: ApplicationInfoService,\n        private iamExpiredInterceptorService: IamExpiredInterceptorService\n    ) {\n        this.errors = this.initErrorObservable(); // has to be first since it is used in initShouldDisplayObservable !\n        this.shouldDisplay = this.initShouldDisplayObservable();\n        this.publicKey = this.initPublicKeyObservable();\n    }\n\n    onSuccess(token: string): void {\n        // To be accessed by the gesdem service and to avoid to have a circular\n        // dependency between the gesdemHandlerService and the captcha service.\n        this.gesdemService.setCaptchaToken(token);\n\n        this.token = token;\n\n        if (!!this.token) {\n            this.clearErrors();\n            this.http\n                .post(RECAPTCHA_API_URL, { token: this.token })\n                .pipe(\n                    catchError((e: unknown) => {\n                        console.error(e);\n                        // 404, timeout or network issue. Could happen.\n                        this.addError(\n                            'Une erreur est survenue lors de la validation du captcha.'\n                        );\n                        return EMPTY;\n                    })\n                )\n                .subscribe(\n                    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                    (data: any) => {\n                        const success = !!data.success;\n                        if (success) {\n                            // Clear any potential errors from the server. We can do\n                            // that because this should be the only error in the set as\n                            // the captcha must be valid to have at least one form error.\n                            this.validationHandlerService.updateErrors([]);\n                        } else {\n                            // If this is thrown, that's pretty bad because it means\n                            // that Google couldn't verify the captcha key. This is\n                            // either an attack or a misconfiguration.\n                            this.addError('Votre captcha est invalide');\n                        }\n                    }\n                );\n        }\n    }\n\n    addError(message: string): void {\n        const existingErrors = this.errorsSubject.getValue();\n        const newError = {\n            name: CAPTCHA_ERROR_NAME,\n            code: CAPTCHA_ERROR_NAME,\n            message\n        };\n\n        this.errorsSubject.next([newError, ...existingErrors]);\n    }\n\n    clearErrors(): void {\n        this.errorsSubject.next([]);\n    }\n\n    getErrors(): Observable<FormError[]> {\n        return this.errors;\n    }\n\n    getShouldDisplay(): Observable<boolean> {\n        return this.shouldDisplay;\n    }\n\n    getPublicKey(): Observable<string> {\n        return this.publicKey;\n    }\n\n    getToken(): string {\n        return this.token;\n    }\n\n    private initErrorObservable(): Observable<FormError[]> {\n        const matchingGesdemErrors = merge(\n            // Ensures that there's an initial value\n            of([]),\n            // Read from the gesdem service all errors.\n            this.validationHandlerService.validationErrorsSubject\n        ).pipe(\n            // Errors from GesDem, or nothing\n            map(errors => errors.filter(ex => CAPTCHA_ERROR_NAME === ex.code))\n        );\n\n        // Bind errors to the component, either from the backend of from this component\n        return combineLatest([\n            // Errors from gesdem matching the name of the captcha.\n            matchingGesdemErrors,\n            // Manual errors\n            this.errorsSubject\n        ]).pipe(\n            // Merge the two errors sets\n            map(([gesdemErrors, localErrors]) => [\n                ...gesdemErrors,\n                ...localErrors\n            ])\n        );\n    }\n\n    private initShouldDisplayObservable(): Observable<boolean> {\n        const isConnectedObservable = this.sessionInfo.data.pipe(\n            map(data => !this.sessionInfo.neverConnected && !!data)\n        );\n\n        const hasBypassEnabledObservable = this.checkBypassProperty().pipe(\n            map(bypass => !!bypass.value)\n        );\n\n        const hasReferenceObservable = merge(\n            // To have an initial value. Can be null if there is no form.\n            of(this.gesdemService.lastResponse),\n            // Read the form from GesDem\n            this.gesdemEventService.formInitializationObservable()\n        ).pipe(\n            map(\n                form =>\n                    !!form &&\n                    !!this.gesdemService.lastResponse &&\n                    !!this.gesdemService.lastResponse.meta &&\n                    !!this.gesdemService.lastResponse.meta.reference\n            )\n        );\n\n        return combineLatest([\n            isConnectedObservable,\n            hasBypassEnabledObservable,\n            hasReferenceObservable,\n            this.errors\n        ]).pipe(\n            // To work around rapidly changing observables, such as the form\n            // when the page loads.\n            debounceTime(0),\n            tap(([isConnected, , , captchaErrors]) => {\n                const hasCaptchaError = !!captchaErrors?.length;\n                if (isConnected && hasCaptchaError) {\n                    // this should not happen, the session may have timed out\n                    this.iamExpiredInterceptorService.setIamSessionExpiredManually();\n                }\n            }),\n            map(\n                ([\n                    isConnected,\n                    hasBypassEnabled,\n                    hasReference,\n                    captchaErrors\n                ]) => {\n                    const hasCaptchaError = !!captchaErrors?.length;\n                    if (hasCaptchaError) {\n                        // always display the component if it has an error, user could has logged out in another window\n                        return true;\n                    }\n                    // Do not display the captcha if the user is connected, the application\n                    // already has a reference or if the bypass is enabled.\n                    return !isConnected && !hasBypassEnabled && !hasReference;\n                }\n            )\n        );\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    private checkBypassProperty(): Observable<any> {\n        const params = !!this.route.snapshot.queryParams.reCaptchaByPassUUID\n            ? new HttpParams().set(\n                  'reCaptchaByPassUUID',\n                  this.route.snapshot.queryParams.reCaptchaByPassUUID\n              )\n            : null;\n\n        return this.http.get(`${RECAPTCHA_API_URL}/bypass`, { params });\n    }\n\n    private initPublicKeyObservable(): Observable<string> {\n        return this.applicationInfoService.data.pipe(\n            map(applicationInfo => this.getPublicKeyOrNull(applicationInfo))\n        );\n    }\n\n    private getPublicKeyOrNull(\n        applicationInfo: ApplicationInfo\n    ): string | null {\n        // Replace this whole shenanigan with applicationInfo?.configuration?.captcha?.publickey when\n        // Typescript supports it.\n        if (!applicationInfo) {\n            return null;\n        }\n\n        const { configuration } = applicationInfo;\n\n        if (!configuration) {\n            return null;\n        }\n        const { captcha } = configuration;\n\n        if (!captcha) {\n            return null;\n        }\n\n        return captcha.publickey || null;\n    }\n}\n"]}
188
+ }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.SessionInfo }, { type: i3.ActivatedRoute }, { type: i4.ValidationHandlerService }, { type: i5.GesdemHandlerService }, { type: i6.GesdemEventService }, { type: i7.ApplicationInfoService }, { type: i8.IamExpiredInterceptorService }, { type: undefined, decorators: [{
189
+ type: Inject,
190
+ args: [CAPTCHA_URL_TOKEN]
191
+ }] }]; } });
192
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recaptcha.service.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/sdk-recaptcha/recaptcha.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpParams } from '@angular/common/http';\nimport { Inject, Injectable, InjectionToken } from '@angular/core';\nimport { ActivatedRoute } from '@angular/router';\nimport {\n    BehaviorSubject,\n    combineLatest,\n    EMPTY,\n    merge,\n    Observable,\n    of\n} from 'rxjs';\nimport { catchError, debounceTime, map, tap } from 'rxjs/operators';\n\nimport { FormError } from '../form-error';\nimport { GesdemEventService } from '../gesdem/gesdem-event.service';\nimport { GesdemHandlerService } from '../gesdem/gesdem-handler.service';\nimport { ApplicationInfo } from '../sdk-appinfo/application-info';\nimport { ApplicationInfoService } from '../sdk-appinfo/application-info.service';\nimport { IamExpiredInterceptorService } from '../sdk-redirect/iam-expired-interceptor.service';\nimport { SessionInfo } from '../sdk-session-info/session-info.service';\nimport { ValidationHandlerService } from '../validation/validation-handler.service';\n\nexport const RECAPTCHA_API_URL = 'api/recaptcha';\nexport const CAPTCHA_ERROR_NAME = 'recaptcha';\n\nexport const CAPTCHA_URL_TOKEN = new InjectionToken<string>(\n    'captchaScriptUrl',\n    {\n        providedIn: 'root',\n        factory: () =>\n            'https://www.recaptcha.net/recaptcha/api.js?onload=reCaptchaLoad&render=explicit'\n    }\n);\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class RecaptchaService {\n    private readonly errors: Observable<FormError[]>;\n    private readonly errorsSubject = new BehaviorSubject<FormError[]>([]);\n    private readonly shouldDisplay: Observable<boolean>;\n    private readonly publicKey: Observable<string>;\n    private token: string;\n\n    constructor(\n        private http: HttpClient,\n        private sessionInfo: SessionInfo,\n        private route: ActivatedRoute,\n        private validationHandlerService: ValidationHandlerService,\n        private gesdemService: GesdemHandlerService,\n        private gesdemEventService: GesdemEventService,\n        private applicationInfoService: ApplicationInfoService,\n        private iamExpiredInterceptorService: IamExpiredInterceptorService,\n        @Inject(CAPTCHA_URL_TOKEN) private captchaScriptUrl: string\n    ) {\n        this.errors = this.initErrorObservable(); // has to be first since it is used in initShouldDisplayObservable !\n        this.shouldDisplay = this.initShouldDisplayObservable();\n        this.publicKey = this.initPublicKeyObservable();\n    }\n\n    onSuccess(token: string): void {\n        // To be accessed by the gesdem service and to avoid to have a circular\n        // dependency between the gesdemHandlerService and the captcha service.\n        this.gesdemService.setCaptchaToken(token);\n\n        this.token = token;\n\n        if (!!this.token) {\n            this.clearErrors();\n            this.http\n                .post(RECAPTCHA_API_URL, { token: this.token })\n                .pipe(\n                    catchError((e: unknown) => {\n                        console.error(e);\n                        // 404, timeout or network issue. Could happen.\n                        this.addError(\n                            'Une erreur est survenue lors de la validation du captcha.'\n                        );\n                        return EMPTY;\n                    })\n                )\n                .subscribe(\n                    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                    (data: any) => {\n                        const success = !!data.success;\n                        if (success) {\n                            // Clear any potential errors from the server. We can do\n                            // that because this should be the only error in the set as\n                            // the captcha must be valid to have at least one form error.\n                            this.validationHandlerService.updateErrors([]);\n                        } else {\n                            // If this is thrown, that's pretty bad because it means\n                            // that Google couldn't verify the captcha key. This is\n                            // either an attack or a misconfiguration.\n                            this.addError('Votre captcha est invalide');\n                        }\n                    }\n                );\n        }\n    }\n\n    addError(message: string): void {\n        const existingErrors = this.errorsSubject.getValue();\n        const newError = {\n            name: CAPTCHA_ERROR_NAME,\n            code: CAPTCHA_ERROR_NAME,\n            message\n        };\n\n        this.errorsSubject.next([newError, ...existingErrors]);\n    }\n\n    clearErrors(): void {\n        this.errorsSubject.next([]);\n    }\n\n    getErrors(): Observable<FormError[]> {\n        return this.errors;\n    }\n\n    getShouldDisplay(): Observable<boolean> {\n        return this.shouldDisplay;\n    }\n\n    getPublicKey(): Observable<string> {\n        return this.publicKey;\n    }\n\n    getToken(): string {\n        return this.token;\n    }\n\n    getCaptchaScriptUrl(): string {\n        return this.captchaScriptUrl;\n    }\n\n    private initErrorObservable(): Observable<FormError[]> {\n        const matchingGesdemErrors = merge(\n            // Ensures that there's an initial value\n            of([]),\n            // Read from the gesdem service all errors.\n            this.validationHandlerService.validationErrorsSubject\n        ).pipe(\n            // Errors from GesDem, or nothing\n            map(errors => errors.filter(ex => CAPTCHA_ERROR_NAME === ex.code))\n        );\n\n        // Bind errors to the component, either from the backend of from this component\n        return combineLatest([\n            // Errors from gesdem matching the name of the captcha.\n            matchingGesdemErrors,\n            // Manual errors\n            this.errorsSubject\n        ]).pipe(\n            // Merge the two errors sets\n            map(([gesdemErrors, localErrors]) => [\n                ...gesdemErrors,\n                ...localErrors\n            ])\n        );\n    }\n\n    private initShouldDisplayObservable(): Observable<boolean> {\n        const isConnectedObservable = this.sessionInfo.data.pipe(\n            map(data => !this.sessionInfo.neverConnected && !!data)\n        );\n\n        const hasBypassEnabledObservable = this.checkBypassProperty().pipe(\n            map(bypass => !!bypass.value)\n        );\n\n        const hasReferenceObservable = merge(\n            // To have an initial value. Can be null if there is no form.\n            of(this.gesdemService.lastResponse),\n            // Read the form from GesDem\n            this.gesdemEventService.formInitializationObservable()\n        ).pipe(\n            map(\n                form =>\n                    !!form &&\n                    !!this.gesdemService.lastResponse &&\n                    !!this.gesdemService.lastResponse.meta &&\n                    !!this.gesdemService.lastResponse.meta.reference\n            )\n        );\n\n        return combineLatest([\n            isConnectedObservable,\n            hasBypassEnabledObservable,\n            hasReferenceObservable,\n            this.errors\n        ]).pipe(\n            // To work around rapidly changing observables, such as the form\n            // when the page loads.\n            debounceTime(0),\n            tap(([isConnected, , , captchaErrors]) => {\n                const hasCaptchaError = !!captchaErrors?.length;\n                if (isConnected && hasCaptchaError) {\n                    // this should not happen, the session may have timed out\n                    this.iamExpiredInterceptorService.setIamSessionExpiredManually();\n                }\n            }),\n            map(\n                ([\n                    isConnected,\n                    hasBypassEnabled,\n                    hasReference,\n                    captchaErrors\n                ]) => {\n                    const hasCaptchaError = !!captchaErrors?.length;\n                    if (hasCaptchaError) {\n                        // always display the component if it has an error, user could has logged out in another window\n                        return true;\n                    }\n                    // Do not display the captcha if the user is connected, the application\n                    // already has a reference or if the bypass is enabled.\n                    return !isConnected && !hasBypassEnabled && !hasReference;\n                }\n            )\n        );\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    private checkBypassProperty(): Observable<any> {\n        const params = !!this.route.snapshot.queryParams.reCaptchaByPassUUID\n            ? new HttpParams().set(\n                  'reCaptchaByPassUUID',\n                  this.route.snapshot.queryParams.reCaptchaByPassUUID\n              )\n            : null;\n\n        return this.http.get(`${RECAPTCHA_API_URL}/bypass`, { params });\n    }\n\n    private initPublicKeyObservable(): Observable<string> {\n        return this.applicationInfoService.data.pipe(\n            map(applicationInfo => this.getPublicKeyOrNull(applicationInfo))\n        );\n    }\n\n    private getPublicKeyOrNull(\n        applicationInfo: ApplicationInfo\n    ): string | null {\n        // Replace this whole shenanigan with applicationInfo?.configuration?.captcha?.publickey when\n        // Typescript supports it.\n        if (!applicationInfo) {\n            return null;\n        }\n\n        const { configuration } = applicationInfo;\n\n        if (!configuration) {\n            return null;\n        }\n        const { captcha } = configuration;\n\n        if (!captcha) {\n            return null;\n        }\n\n        return captcha.publickey || null;\n    }\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  import { Component, ViewChild } from '@angular/core';
2
2
  import { debounceTime } from 'rxjs/operators';
3
- import { GrecaptchaComponent, RECAPTCHA_URL } from './grecaptcha/grecaptcha.component';
3
+ import { GrecaptchaComponent } from './grecaptcha/grecaptcha.component';
4
4
  import * as i0 from "@angular/core";
5
5
  import * as i1 from "./recaptcha.service";
6
6
  import * as i2 from "@angular/common";
@@ -10,7 +10,7 @@ export class SdkRecaptchaComponent {
10
10
  this.recaptchaService = recaptchaService;
11
11
  this.cdr = cdr;
12
12
  this.hasLoadingError = false;
13
- const url = new URL(RECAPTCHA_URL);
13
+ const url = new URL(this.recaptchaService.getCaptchaScriptUrl());
14
14
  this.recaptchaUrlForUserError = `${url.protocol}//${url.hostname}`;
15
15
  }
16
16
  ngOnInit() {
@@ -68,4 +68,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
68
68
  type: ViewChild,
69
69
  args: [GrecaptchaComponent]
70
70
  }] } });
71
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sdk-recaptcha.component.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/sdk-recaptcha/sdk-recaptcha.component.ts","../../../../projects/prestations-ng/src/sdk-recaptcha/sdk-recaptcha.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEH,SAAS,EAGT,SAAS,EACZ,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAG9C,OAAO,EACH,mBAAmB,EACnB,aAAa,EAChB,MAAM,mCAAmC,CAAC;;;;;AAQ3C,MAAM,OAAO,qBAAqB;IAc9B,YACY,gBAAkC,EAClC,GAAsB;QADtB,qBAAgB,GAAhB,gBAAgB,CAAkB;QAClC,QAAG,GAAH,GAAG,CAAmB;QAVlC,oBAAe,GAAG,KAAK,CAAC;QAYpB,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,aAAa,CAAC,CAAC;QACnC,IAAI,CAAC,wBAAwB,GAAG,GAAG,GAAG,CAAC,QAAQ,KAAK,GAAG,CAAC,QAAQ,EAAE,CAAC;IACvE,CAAC;IAED,QAAQ;QACJ,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC;QACpC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC,gBAAgB,EAAE,CAAC;QAC9D,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC;QAChD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC,YAAY,EAAE,CAAC;QAEtD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM;aAChC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;YACxB,0DAA0D;aACzD,SAAS,CAAC,MAAM,CAAC,EAAE;YAChB,8DAA8D;YAC9D,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBACxB,IAAI,CAAC,iBAAiB,EAAE,CAAC;aAC5B;YACD,yDAAyD;YACzD,cAAc;YACd,IAAI,MAAM,CAAC,MAAM,IAAI,IAAI,CAAC,UAAU,EAAE;gBAClC,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;aAC3B;QACL,CAAC,CAAC,CAAC;QAEP,IAAI,CAAC,iBAAiB,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC5D,CAAC;IAED,WAAW;QACP,IAAI,IAAI,CAAC,kBAAkB,EAAE;YACzB,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,CAAC;SACzC;QAED,qCAAqC;QACrC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;IAClC,CAAC;IAED,WAAW,CAAC,QAAiB;QACzB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QAEzB,6CAA6C;QAC7C,IAAI,IAAI,CAAC,iBAAiB,EAAE;YACxB,IAAI,CAAC,iBAAiB,EAAE,CAAC;SAC5B;IACL,CAAC;IAED,cAAc,CAAC,WAAoB;QAC/B,IAAI,CAAC,eAAe,GAAG,WAAW,CAAC;QAEnC,6CAA6C;QAC7C,IAAI,IAAI,CAAC,iBAAiB,EAAE;YACxB,IAAI,CAAC,iBAAiB,EAAE,CAAC;SAC5B;IACL,CAAC;IAED,cAAc,CAAC,KAAa;QACxB,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IAC3C,CAAC;;mHA3EQ,qBAAqB;uGAArB,qBAAqB,2FACnB,mBAAmB,gDCvBlC,+oDAyCA;4FDnBa,qBAAqB;kBALjC,SAAS;+BAEI,SAAS;uIAKnB,UAAU;sBADT,SAAS;uBAAC,mBAAmB","sourcesContent":["import {\n    ChangeDetectorRef,\n    Component,\n    OnDestroy,\n    OnInit,\n    ViewChild\n} from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { debounceTime } from 'rxjs/operators';\n\nimport { FormError } from '../form-error';\nimport {\n    GrecaptchaComponent,\n    RECAPTCHA_URL\n} from './grecaptcha/grecaptcha.component';\nimport { RecaptchaService } from './recaptcha.service';\n\n@Component({\n    // eslint-disable-next-line  @angular-eslint/component-selector\n    selector: 'captcha',\n    templateUrl: './sdk-recaptcha.component.html'\n})\nexport class SdkRecaptchaComponent implements OnInit, OnDestroy {\n    @ViewChild(GrecaptchaComponent)\n    grecaptcha: GrecaptchaComponent;\n\n    shouldDisplay: Observable<boolean>;\n    isLoaded: boolean;\n    hasLoadingError = false;\n    errors: Observable<FormError[]>;\n    publicKey: Observable<string>;\n    readonly recaptchaUrlForUserError: string;\n\n    private errorsSubscription: Subscription;\n    private forceDetectChange: () => void;\n\n    constructor(\n        private recaptchaService: RecaptchaService,\n        private cdr: ChangeDetectorRef\n    ) {\n        const url = new URL(RECAPTCHA_URL);\n        this.recaptchaUrlForUserError = `${url.protocol}//${url.hostname}`;\n    }\n\n    ngOnInit(): void {\n        this.recaptchaService.clearErrors();\n        this.shouldDisplay = this.recaptchaService.getShouldDisplay();\n        this.errors = this.recaptchaService.getErrors();\n        this.publicKey = this.recaptchaService.getPublicKey();\n\n        this.errorsSubscription = this.errors\n            .pipe(debounceTime(500))\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(errors => {\n                // Hack because the captcha confuses Angular detection change.\n                if (this.forceDetectChange) {\n                    this.forceDetectChange();\n                }\n                // In case of errors, reset the captcha for a better user\n                // experience.\n                if (errors.length && this.grecaptcha) {\n                    this.grecaptcha.reset();\n                }\n            });\n\n        this.forceDetectChange = () => this.cdr.detectChanges();\n    }\n\n    ngOnDestroy(): void {\n        if (this.errorsSubscription) {\n            this.errorsSubscription.unsubscribe();\n        }\n\n        // Avoid cdr on destroyed components.\n        this.forceDetectChange = null;\n    }\n\n    setIsLoaded(isLoaded: boolean): void {\n        this.isLoaded = isLoaded;\n\n        // Angular doesn't detect the change properly\n        if (this.forceDetectChange) {\n            this.forceDetectChange();\n        }\n    }\n\n    setScriptError(scriptError: boolean): void {\n        this.hasLoadingError = scriptError;\n\n        // Angular doesn't detect the change properly\n        if (this.forceDetectChange) {\n            this.forceDetectChange();\n        }\n    }\n\n    tokenOnSuccess(token: string): void {\n        this.recaptchaService.onSuccess(token);\n    }\n}\n","<!-- Only display the content if the captcha should be displayed (user not connected for instance) -->\n<div id=\"foehn_recaptchaContainer\" *ngIf=\"shouldDisplay | async\">\n    <!-- Manual error handling because errors can come from multiple sources -->\n    <ng-container *ngIf=\"errors | async as err\">\n        <div\n            class=\"form-control-feedback text-danger\"\n            *ngIf=\"err && err.length\"\n        >\n            <p *ngFor=\"let error of err\">{{ error.message }}</p>\n        </div>\n    </ng-container>\n\n    <!-- While the captcha is being loaded -->\n    <div class=\"alert alert-info\" *ngIf=\"!isLoaded && !hasLoadingError\">\n        Chargement du captcha\n    </div>\n\n    <!-- In case google.com cannot be reached for instance -->\n    <div class=\"alert alert-danger\" *ngIf=\"hasLoadingError\">\n        <div class=\"mt-0 alert-heading h4\">\n            Une erreur est survenue lors du chargement du captcha.\n        </div>\n        <p>\n            Merci de vérifier que votre navigateur peut accéder à\n            <a [href]=\"recaptchaUrlForUserError\" target=\"_blank\">\n                {{ recaptchaUrlForUserError }}\n            </a>\n            .\n        </p>\n    </div>\n\n    <ng-container *ngIf=\"publicKey | async as key\">\n        <!-- The captcha itself is in a different container not to confused Angular change detection -->\n        <app-grecaptcha\n            [siteKey]=\"key\"\n            (loaded)=\"setIsLoaded($event)\"\n            (scriptError)=\"setScriptError($event)\"\n            (tokenOnSuccess)=\"tokenOnSuccess($event)\"\n        ></app-grecaptcha>\n    </ng-container>\n</div>\n"]}
71
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sdk-recaptcha.component.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/sdk-recaptcha/sdk-recaptcha.component.ts","../../../../projects/prestations-ng/src/sdk-recaptcha/sdk-recaptcha.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEH,SAAS,EAGT,SAAS,EACZ,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAG9C,OAAO,EAAE,mBAAmB,EAAE,MAAM,mCAAmC,CAAC;;;;;AAQxE,MAAM,OAAO,qBAAqB;IAc9B,YACY,gBAAkC,EAClC,GAAsB;QADtB,qBAAgB,GAAhB,gBAAgB,CAAkB;QAClC,QAAG,GAAH,GAAG,CAAmB;QAVlC,oBAAe,GAAG,KAAK,CAAC;QAYpB,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC,mBAAmB,EAAE,CAAC,CAAC;QACjE,IAAI,CAAC,wBAAwB,GAAG,GAAG,GAAG,CAAC,QAAQ,KAAK,GAAG,CAAC,QAAQ,EAAE,CAAC;IACvE,CAAC;IAED,QAAQ;QACJ,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC;QACpC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC,gBAAgB,EAAE,CAAC;QAC9D,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC;QAChD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC,YAAY,EAAE,CAAC;QAEtD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM;aAChC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;YACxB,0DAA0D;aACzD,SAAS,CAAC,MAAM,CAAC,EAAE;YAChB,8DAA8D;YAC9D,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBACxB,IAAI,CAAC,iBAAiB,EAAE,CAAC;aAC5B;YACD,yDAAyD;YACzD,cAAc;YACd,IAAI,MAAM,CAAC,MAAM,IAAI,IAAI,CAAC,UAAU,EAAE;gBAClC,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;aAC3B;QACL,CAAC,CAAC,CAAC;QAEP,IAAI,CAAC,iBAAiB,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC5D,CAAC;IAED,WAAW;QACP,IAAI,IAAI,CAAC,kBAAkB,EAAE;YACzB,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,CAAC;SACzC;QAED,qCAAqC;QACrC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;IAClC,CAAC;IAED,WAAW,CAAC,QAAiB;QACzB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QAEzB,6CAA6C;QAC7C,IAAI,IAAI,CAAC,iBAAiB,EAAE;YACxB,IAAI,CAAC,iBAAiB,EAAE,CAAC;SAC5B;IACL,CAAC;IAED,cAAc,CAAC,WAAoB;QAC/B,IAAI,CAAC,eAAe,GAAG,WAAW,CAAC;QAEnC,6CAA6C;QAC7C,IAAI,IAAI,CAAC,iBAAiB,EAAE;YACxB,IAAI,CAAC,iBAAiB,EAAE,CAAC;SAC5B;IACL,CAAC;IAED,cAAc,CAAC,KAAa;QACxB,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IAC3C,CAAC;;mHA3EQ,qBAAqB;uGAArB,qBAAqB,2FACnB,mBAAmB,gDCpBlC,+oDAyCA;4FDtBa,qBAAqB;kBALjC,SAAS;+BAEI,SAAS;uIAKnB,UAAU;sBADT,SAAS;uBAAC,mBAAmB","sourcesContent":["import {\n    ChangeDetectorRef,\n    Component,\n    OnDestroy,\n    OnInit,\n    ViewChild\n} from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { debounceTime } from 'rxjs/operators';\n\nimport { FormError } from '../form-error';\nimport { GrecaptchaComponent } from './grecaptcha/grecaptcha.component';\nimport { RecaptchaService } from './recaptcha.service';\n\n@Component({\n    // eslint-disable-next-line  @angular-eslint/component-selector\n    selector: 'captcha',\n    templateUrl: './sdk-recaptcha.component.html'\n})\nexport class SdkRecaptchaComponent implements OnInit, OnDestroy {\n    @ViewChild(GrecaptchaComponent)\n    grecaptcha: GrecaptchaComponent;\n\n    shouldDisplay: Observable<boolean>;\n    isLoaded: boolean;\n    hasLoadingError = false;\n    errors: Observable<FormError[]>;\n    publicKey: Observable<string>;\n    readonly recaptchaUrlForUserError: string;\n\n    private errorsSubscription: Subscription;\n    private forceDetectChange: () => void;\n\n    constructor(\n        private recaptchaService: RecaptchaService,\n        private cdr: ChangeDetectorRef\n    ) {\n        const url = new URL(this.recaptchaService.getCaptchaScriptUrl());\n        this.recaptchaUrlForUserError = `${url.protocol}//${url.hostname}`;\n    }\n\n    ngOnInit(): void {\n        this.recaptchaService.clearErrors();\n        this.shouldDisplay = this.recaptchaService.getShouldDisplay();\n        this.errors = this.recaptchaService.getErrors();\n        this.publicKey = this.recaptchaService.getPublicKey();\n\n        this.errorsSubscription = this.errors\n            .pipe(debounceTime(500))\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(errors => {\n                // Hack because the captcha confuses Angular detection change.\n                if (this.forceDetectChange) {\n                    this.forceDetectChange();\n                }\n                // In case of errors, reset the captcha for a better user\n                // experience.\n                if (errors.length && this.grecaptcha) {\n                    this.grecaptcha.reset();\n                }\n            });\n\n        this.forceDetectChange = () => this.cdr.detectChanges();\n    }\n\n    ngOnDestroy(): void {\n        if (this.errorsSubscription) {\n            this.errorsSubscription.unsubscribe();\n        }\n\n        // Avoid cdr on destroyed components.\n        this.forceDetectChange = null;\n    }\n\n    setIsLoaded(isLoaded: boolean): void {\n        this.isLoaded = isLoaded;\n\n        // Angular doesn't detect the change properly\n        if (this.forceDetectChange) {\n            this.forceDetectChange();\n        }\n    }\n\n    setScriptError(scriptError: boolean): void {\n        this.hasLoadingError = scriptError;\n\n        // Angular doesn't detect the change properly\n        if (this.forceDetectChange) {\n            this.forceDetectChange();\n        }\n    }\n\n    tokenOnSuccess(token: string): void {\n        this.recaptchaService.onSuccess(token);\n    }\n}\n","<!-- Only display the content if the captcha should be displayed (user not connected for instance) -->\n<div id=\"foehn_recaptchaContainer\" *ngIf=\"shouldDisplay | async\">\n    <!-- Manual error handling because errors can come from multiple sources -->\n    <ng-container *ngIf=\"errors | async as err\">\n        <div\n            class=\"form-control-feedback text-danger\"\n            *ngIf=\"err && err.length\"\n        >\n            <p *ngFor=\"let error of err\">{{ error.message }}</p>\n        </div>\n    </ng-container>\n\n    <!-- While the captcha is being loaded -->\n    <div class=\"alert alert-info\" *ngIf=\"!isLoaded && !hasLoadingError\">\n        Chargement du captcha\n    </div>\n\n    <!-- In case google.com cannot be reached for instance -->\n    <div class=\"alert alert-danger\" *ngIf=\"hasLoadingError\">\n        <div class=\"mt-0 alert-heading h4\">\n            Une erreur est survenue lors du chargement du captcha.\n        </div>\n        <p>\n            Merci de vérifier que votre navigateur peut accéder à\n            <a [href]=\"recaptchaUrlForUserError\" target=\"_blank\">\n                {{ recaptchaUrlForUserError }}\n            </a>\n            .\n        </p>\n    </div>\n\n    <ng-container *ngIf=\"publicKey | async as key\">\n        <!-- The captcha itself is in a different container not to confused Angular change detection -->\n        <app-grecaptcha\n            [siteKey]=\"key\"\n            (loaded)=\"setIsLoaded($event)\"\n            (scriptError)=\"setScriptError($event)\"\n            (tokenOnSuccess)=\"tokenOnSuccess($event)\"\n        ></app-grecaptcha>\n    </ng-container>\n</div>\n"]}
@@ -1,5 +1,5 @@
1
1
  import * as i0 from '@angular/core';
2
- import { Injectable, Optional, Inject, EventEmitter, Directive, Input, ViewChildren, ViewChild, forwardRef, Output, HostBinding, Pipe, Component, ContentChildren, HostListener, NgModule, TemplateRef, ContentChild } from '@angular/core';
2
+ import { Injectable, Optional, Inject, EventEmitter, Directive, Input, ViewChildren, ViewChild, forwardRef, Output, HostBinding, Pipe, Component, ContentChildren, HostListener, NgModule, TemplateRef, ContentChild, InjectionToken } from '@angular/core';
3
3
  import * as i1$1 from '@angular/router';
4
4
  import { ActivatedRoute, NavigationStart, RouterModule, NavigationEnd, RouterLink } from '@angular/router';
5
5
  import { of, BehaviorSubject, combineLatest, Subject, throwError, tap as tap$1, concat, toArray, EMPTY, merge, debounceTime as debounceTime$1, map as map$1 } from 'rxjs';
@@ -12370,8 +12370,12 @@ const HTTP_LOADER_FILTERED_URL = [
12370
12370
 
12371
12371
  const RECAPTCHA_API_URL = 'api/recaptcha';
12372
12372
  const CAPTCHA_ERROR_NAME = 'recaptcha';
12373
+ const CAPTCHA_URL_TOKEN = new InjectionToken('captchaScriptUrl', {
12374
+ providedIn: 'root',
12375
+ factory: () => 'https://www.recaptcha.net/recaptcha/api.js?onload=reCaptchaLoad&render=explicit'
12376
+ });
12373
12377
  class RecaptchaService {
12374
- constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService, iamExpiredInterceptorService) {
12378
+ constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService, iamExpiredInterceptorService, captchaScriptUrl) {
12375
12379
  this.http = http;
12376
12380
  this.sessionInfo = sessionInfo;
12377
12381
  this.route = route;
@@ -12380,6 +12384,7 @@ class RecaptchaService {
12380
12384
  this.gesdemEventService = gesdemEventService;
12381
12385
  this.applicationInfoService = applicationInfoService;
12382
12386
  this.iamExpiredInterceptorService = iamExpiredInterceptorService;
12387
+ this.captchaScriptUrl = captchaScriptUrl;
12383
12388
  this.errorsSubject = new BehaviorSubject([]);
12384
12389
  this.errors = this.initErrorObservable(); // has to be first since it is used in initShouldDisplayObservable !
12385
12390
  this.shouldDisplay = this.initShouldDisplayObservable();
@@ -12443,6 +12448,9 @@ class RecaptchaService {
12443
12448
  getToken() {
12444
12449
  return this.token;
12445
12450
  }
12451
+ getCaptchaScriptUrl() {
12452
+ return this.captchaScriptUrl;
12453
+ }
12446
12454
  initErrorObservable() {
12447
12455
  const matchingGesdemErrors = merge(
12448
12456
  // Ensures that there's an initial value
@@ -12527,16 +12535,20 @@ class RecaptchaService {
12527
12535
  return captcha.publickey || null;
12528
12536
  }
12529
12537
  }
12530
- RecaptchaService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, deps: [{ token: i1.HttpClient }, { token: SessionInfo }, { token: i1$1.ActivatedRoute }, { token: ValidationHandlerService }, { token: GesdemHandlerService }, { token: GesdemEventService }, { token: ApplicationInfoService }, { token: IamExpiredInterceptorService }], target: i0.ɵɵFactoryTarget.Injectable });
12538
+ RecaptchaService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, deps: [{ token: i1.HttpClient }, { token: SessionInfo }, { token: i1$1.ActivatedRoute }, { token: ValidationHandlerService }, { token: GesdemHandlerService }, { token: GesdemEventService }, { token: ApplicationInfoService }, { token: IamExpiredInterceptorService }, { token: CAPTCHA_URL_TOKEN }], target: i0.ɵɵFactoryTarget.Injectable });
12531
12539
  RecaptchaService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, providedIn: 'root' });
12532
12540
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RecaptchaService, decorators: [{
12533
12541
  type: Injectable,
12534
12542
  args: [{
12535
12543
  providedIn: 'root'
12536
12544
  }]
12537
- }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: SessionInfo }, { type: i1$1.ActivatedRoute }, { type: ValidationHandlerService }, { type: GesdemHandlerService }, { type: GesdemEventService }, { type: ApplicationInfoService }, { type: IamExpiredInterceptorService }]; } });
12545
+ }], ctorParameters: function () {
12546
+ return [{ type: i1.HttpClient }, { type: SessionInfo }, { type: i1$1.ActivatedRoute }, { type: ValidationHandlerService }, { type: GesdemHandlerService }, { type: GesdemEventService }, { type: ApplicationInfoService }, { type: IamExpiredInterceptorService }, { type: undefined, decorators: [{
12547
+ type: Inject,
12548
+ args: [CAPTCHA_URL_TOKEN]
12549
+ }] }];
12550
+ } });
12538
12551
 
12539
- const RECAPTCHA_URL = 'https://www.recaptcha.net/recaptcha/api.js?onload=reCaptchaLoad&render=explicit';
12540
12552
  const SCRIPT_ID = 'prestations-ng-captcha-script';
12541
12553
  /**
12542
12554
  * Integration with Google Recaptcha.
@@ -12546,7 +12558,8 @@ const SCRIPT_ID = 'prestations-ng-captcha-script';
12546
12558
  *
12547
12559
  */
12548
12560
  class GrecaptchaComponent {
12549
- constructor() {
12561
+ constructor(recaptchaService) {
12562
+ this.recaptchaService = recaptchaService;
12550
12563
  this.loaded = new EventEmitter();
12551
12564
  this.scriptError = new EventEmitter();
12552
12565
  this.tokenOnSuccess = new EventEmitter();
@@ -12573,6 +12586,7 @@ class GrecaptchaComponent {
12573
12586
  }
12574
12587
  const config = {
12575
12588
  sitekey: this.siteKey,
12589
+ theme: 'light',
12576
12590
  callback: (token) => this.tokenOnSuccess.emit(token),
12577
12591
  'expired-callback': () => this.reset()
12578
12592
  };
@@ -12582,7 +12596,7 @@ class GrecaptchaComponent {
12582
12596
  appendScriptTag() {
12583
12597
  const script = document.createElement('script');
12584
12598
  script.id = SCRIPT_ID;
12585
- script.src = RECAPTCHA_URL;
12599
+ script.src = this.recaptchaService.getCaptchaScriptUrl();
12586
12600
  script.async = true;
12587
12601
  script.defer = true;
12588
12602
  script.onerror = () => this.scriptError.emit(true);
@@ -12619,12 +12633,12 @@ class GrecaptchaComponent {
12619
12633
  }
12620
12634
  }
12621
12635
  }
12622
- GrecaptchaComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GrecaptchaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
12636
+ GrecaptchaComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GrecaptchaComponent, deps: [{ token: RecaptchaService }], target: i0.ɵɵFactoryTarget.Component });
12623
12637
  GrecaptchaComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: GrecaptchaComponent, selector: "app-grecaptcha", inputs: { siteKey: "siteKey" }, outputs: { loaded: "loaded", scriptError: "scriptError", tokenOnSuccess: "tokenOnSuccess", destroyed: "destroyed" }, viewQueries: [{ propertyName: "recaptchaContainer", first: true, predicate: ["recaptchaContainer"], descendants: true, static: true }, { propertyName: "recaptchaScriptContainer", first: true, predicate: ["recaptchaScriptContainer"], descendants: true, static: true }], ngImport: i0, template: "<div #recaptchaScriptContainer></div>\n<div #recaptchaContainer></div>\n" });
12624
12638
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GrecaptchaComponent, decorators: [{
12625
12639
  type: Component,
12626
12640
  args: [{ selector: 'app-grecaptcha', template: "<div #recaptchaScriptContainer></div>\n<div #recaptchaContainer></div>\n" }]
12627
- }], propDecorators: { siteKey: [{
12641
+ }], ctorParameters: function () { return [{ type: RecaptchaService }]; }, propDecorators: { siteKey: [{
12628
12642
  type: Input
12629
12643
  }], loaded: [{
12630
12644
  type: Output
@@ -12647,7 +12661,7 @@ class SdkRecaptchaComponent {
12647
12661
  this.recaptchaService = recaptchaService;
12648
12662
  this.cdr = cdr;
12649
12663
  this.hasLoadingError = false;
12650
- const url = new URL(RECAPTCHA_URL);
12664
+ const url = new URL(this.recaptchaService.getCaptchaScriptUrl());
12651
12665
  this.recaptchaUrlForUserError = `${url.protocol}//${url.hostname}`;
12652
12666
  }
12653
12667
  ngOnInit() {
@@ -14021,6 +14035,8 @@ class SdkEpaymentComponent {
14021
14035
  this.platformFailureMessageDicoKey = 'epayment-form.errors.default';
14022
14036
  this.failureModalDisplayed = false;
14023
14037
  this.redirectModalDisplayed = false;
14038
+ // Make it easier for testing purpose
14039
+ this.window = window;
14024
14040
  this.failureMessageDicoKeyByCode = new Map([
14025
14041
  ['back', 'epayment-form.errors.back'],
14026
14042
  ['cancel', 'epayment-form.errors.cancel'],
@@ -14044,7 +14060,7 @@ class SdkEpaymentComponent {
14044
14060
  // eslint-disable-next-line rxjs-angular/prefer-async-pipe
14045
14061
  .subscribe(redirectUrl => {
14046
14062
  this.redirectModalDisplayed = true;
14047
- window.location.href = redirectUrl;
14063
+ this.window.location.href = redirectUrl;
14048
14064
  });
14049
14065
  }
14050
14066
  handleFailure(failureCode) {
@@ -15302,5 +15318,5 @@ class DropdownMenuItem {
15302
15318
  * Generated bundle index. Do not edit.
15303
15319
  */
15304
15320
 
15305
- export { APP_INFO_API_URL, AbstractFoehnUploaderComponent, AbstractListDetailPageComponent, AbstractMenuPageComponent, AbstractPageComponent, AbstractPageFromMenuComponent, ActionStatut, Address, AddressTypeLight, ApplicationInfo, ApplicationInfoService, BAD_PARAMS_HELP_TEXT, BoDocumentError, BoDocumentsWithErrors, BoMultiUploadService, Breadcrumb, BreadcrumbEventService, BreadcrumbItem, CAPTCHA_ERROR_NAME, CURRENCY_REGEXP, Calendar, Canton, Captcha, ComponentError, Configuration, Country, CurrencyHelper, CurrentWeek, DECIMALS_SEPARATOR, DEFAULT_INTERNATIONAL_AND_NO_SWISS, DEFAULT_INTERNATIONAL_AND_NO_SWISS_MOBILE, DEFAULT_INTERNATIONAL_AND_NO_SWISS_PHONE, DEFAULT_INTERNATIONAL_HELP_TEXT, DEFAULT_PREFIX, DEFAULT_SWISS_HELP_TEXT, DEFAULT_SWISS_MOBILE_PHONE_HELP_TEXT, DEFAULT_SWISS_PHONE_HELP_TEXT, DICTIONARY_BASE_URL, DateHelper, DatePickerHelper, DatePickerNavigationHelper, DayMonth, DaySlots, DemandeExpirationService, DisplayCurrencyPipe, DisplayDatePipe, District, Document, DocumentError, DocumentReference, DocumentReferenceWithFile, DocumentsWithErrors, DropdownMenuGroup, DropdownMenuItem, EPaymentService, EtapeInfo, FORM_SUPPORT_CYBER_TITLE_FALLBACK, FocusedDay, FoehnAbbrComponent, FoehnAddressModule, FoehnAgendaComponent, FoehnAgendaModule, FoehnAgendaNavigationComponent, FoehnAgendaTimeslotPanelComponent, FoehnAutocompleteComponent, FoehnAutocompleteModule, FoehnBoMultiUploadComponent, FoehnBoMultiUploadModule, FoehnBooleanCheckboxComponent, FoehnBooleanModule, FoehnBooleanRadioComponent, FoehnBreadcrumbComponent, FoehnBreadcrumbModule, FoehnCheckableGroupComponent, FoehnCheckablesModule, FoehnCheckboxComponent, FoehnConfirmModalComponent, FoehnConfirmModalContent, FoehnConfirmModalModule, FoehnConfirmModalService, FoehnDateComponent, FoehnDatePickerButtonComponent, FoehnDatePickerButtonModule, FoehnDatePickerComponent, FoehnDatePickerModule, FoehnDateTimeComponent, FoehnDecisionElectroniqueComponent, FoehnDecisionElectroniqueModule, FoehnDisplayAddressComponent, FoehnDropdownMenuComponent, FoehnDropdownMenuModule, FoehnErrorPillComponent, FoehnFormComponent, FoehnFormModule, FoehnHeaderComponent, FoehnHeaderModule, FoehnHelpModalComponent, FoehnHelpModalModule, FoehnIconCalendarComponent, FoehnIconCheckComponent, FoehnIconCheckSquareOComponent, FoehnIconChevronDownComponent, FoehnIconChevronLeftComponent, FoehnIconChevronRightComponent, FoehnIconChevronUpComponent, FoehnIconClockComponent, FoehnIconCommentDotsComponent, FoehnIconEditComponent, FoehnIconExternalLinkAltComponent, FoehnIconFilePdfComponent, FoehnIconInfoCircleComponent, FoehnIconLockComponent, FoehnIconMapMarkerComponent, FoehnIconMinusCircleComponent, FoehnIconPencilComponent, FoehnIconPlusCircleComponent, FoehnIconPlusSquareComponent, FoehnIconSearchComponent, FoehnIconTimesComponent, FoehnIconTrashAltComponent, FoehnIconUnlockAltComponent, FoehnIconUserComponent, FoehnIconsModule, FoehnInputAddressComponent, FoehnInputComponent, FoehnInputEmailComponent, FoehnInputForeignLocalityComponent, FoehnInputForeignStreetComponent, FoehnInputHiddenComponent, FoehnInputModule, FoehnInputNav13Component, FoehnInputNav13Module, FoehnInputNumberComponent, FoehnInputPasswordComponent, FoehnInputPhoneComponent, FoehnInputPrefixedTextComponent, FoehnInputStringComponent, FoehnInputTextComponent, FoehnInputTextareaComponent, FoehnListComponent, FoehnListItem, FoehnListModule, FoehnListSummaryComponent, FoehnMenuItemComponent, FoehnMenuItemTransmitComponent, FoehnMenuPrestationModule, FoehnMiscModule, FoehnModalComponent, FoehnModalModule, FoehnMultiUploadComponent, FoehnMultiUploadModule, FoehnMultiselectAutocompleteComponent, FoehnMultiselectAutocompleteModule, FoehnNavigationComponent, FoehnNavigationModule, FoehnNavigationService, FoehnNotFoundModule, FoehnNotfoundComponent, FoehnPageComponent, FoehnPageCounterComponent, FoehnPageExpirationTimerComponent, FoehnPageModalComponent, FoehnPageModule, FoehnPageService, FoehnPictureUploadComponent, FoehnPictureUploadModule, FoehnRadioComponent, FoehnRecapSectionComponent, FoehnRecapSectionModule, FoehnRemainingAlertsSummaryComponent, FoehnRemainingAlertsSummaryModule, FoehnSelectComponent, FoehnSimpleNavigationComponent, FoehnSkipLinkComponent, FoehnStatusProgressBarComponent, FoehnStatusProgressBarModule, FoehnTableColumnConfiguration, FoehnTableComponent, FoehnTableModule, FoehnTablePageChangeEvent, FoehnTimeComponent, FoehnTransmitWaitingModalComponent, FoehnTransmitWaitingModalService, FoehnUploadProgressBarComponent, FoehnUploadProgressBarModule, FoehnUserConnectedAsComponent, FoehnUserConnectedAsModule, FoehnValidationAlertSummaryComponent, FoehnValidationAlertSummaryModule, FoehnValidationAlertsComponent, FoehnValidationAlertsModule, FooterLink, FormMetadata, FormPostResponse, FormSelectOption, FormatIdePipe, FormatterModule, GESDEM_MAX_DATA_LENGTH, GesdemActionRecoveryLoginComponent, GesdemActionRecoveryModule, GesdemActionRecoveryRegistrationComponent, GesdemConfirmationComponent, GesdemConfirmationModule, GesdemErrorComponent, GesdemErrorModule, GesdemEventService, GesdemHandlerService, GesdemStatutUtils, GrowlBrokerService, GrowlMessage, GrowlType, HTTP_LOADER_FILTERED_URL, I18nForm, IbanFormatterDirective, IdeFormatterDirective, Locality, MonthYear, MultiUploadService, Municipality, NDCFormatterDirective, NavigationDirection, NumberCurrencyFormatterDirective, ObjectHelper, PORTAIL_BASE_URL_INT, PageChangeEvent, PageUploadLimitService, PaginationWeek, PendingFiles, PendingUploadService, PipeModule, PlaceOfOrigin, Portail, PostalLocality, Preferences, PrestationsNgCoreModule, RECAPTCHA_API_URL, RecaptchaService, RedirectComponent, RegisterNgModelService, SESSION_INFO_API_URL, SWISS_ISO_ID, SdkDictionaryModule, SdkDictionaryPipe, SdkDictionaryService, SdkEpaymentComponent, SdkEpaymentModule, SdkRecaptchaComponent, SdkRecaptchaModule, SdkRedirectModule, SdkStatisticsService, SelectedSlot, ServiceLocator, Session, SessionInfo, SessionInfoData, SessionInfoWithApplicationService, Street, StreetNumber, THOUSANDS_SEPARATOR, TRANSMIT_WAITING_MODAL_DELAY_IN_MILLISECONDS, TableSort, UploadProgress, UploadProgressService, UploaderHelper, ValidationHandlerService, formatDecimalCurrency, formatNonDecimalCurrency, formatNumberAsGiven, gesdemLoaderGuard, getSafeReference, replaceAll };
15321
+ export { APP_INFO_API_URL, AbstractFoehnUploaderComponent, AbstractListDetailPageComponent, AbstractMenuPageComponent, AbstractPageComponent, AbstractPageFromMenuComponent, ActionStatut, Address, AddressTypeLight, ApplicationInfo, ApplicationInfoService, BAD_PARAMS_HELP_TEXT, BoDocumentError, BoDocumentsWithErrors, BoMultiUploadService, Breadcrumb, BreadcrumbEventService, BreadcrumbItem, CAPTCHA_ERROR_NAME, CAPTCHA_URL_TOKEN, CURRENCY_REGEXP, Calendar, Canton, Captcha, ComponentError, Configuration, Country, CurrencyHelper, CurrentWeek, DECIMALS_SEPARATOR, DEFAULT_INTERNATIONAL_AND_NO_SWISS, DEFAULT_INTERNATIONAL_AND_NO_SWISS_MOBILE, DEFAULT_INTERNATIONAL_AND_NO_SWISS_PHONE, DEFAULT_INTERNATIONAL_HELP_TEXT, DEFAULT_PREFIX, DEFAULT_SWISS_HELP_TEXT, DEFAULT_SWISS_MOBILE_PHONE_HELP_TEXT, DEFAULT_SWISS_PHONE_HELP_TEXT, DICTIONARY_BASE_URL, DateHelper, DatePickerHelper, DatePickerNavigationHelper, DayMonth, DaySlots, DemandeExpirationService, DisplayCurrencyPipe, DisplayDatePipe, District, Document, DocumentError, DocumentReference, DocumentReferenceWithFile, DocumentsWithErrors, DropdownMenuGroup, DropdownMenuItem, EPaymentService, EtapeInfo, FORM_SUPPORT_CYBER_TITLE_FALLBACK, FocusedDay, FoehnAbbrComponent, FoehnAddressModule, FoehnAgendaComponent, FoehnAgendaModule, FoehnAgendaNavigationComponent, FoehnAgendaTimeslotPanelComponent, FoehnAutocompleteComponent, FoehnAutocompleteModule, FoehnBoMultiUploadComponent, FoehnBoMultiUploadModule, FoehnBooleanCheckboxComponent, FoehnBooleanModule, FoehnBooleanRadioComponent, FoehnBreadcrumbComponent, FoehnBreadcrumbModule, FoehnCheckableGroupComponent, FoehnCheckablesModule, FoehnCheckboxComponent, FoehnConfirmModalComponent, FoehnConfirmModalContent, FoehnConfirmModalModule, FoehnConfirmModalService, FoehnDateComponent, FoehnDatePickerButtonComponent, FoehnDatePickerButtonModule, FoehnDatePickerComponent, FoehnDatePickerModule, FoehnDateTimeComponent, FoehnDecisionElectroniqueComponent, FoehnDecisionElectroniqueModule, FoehnDisplayAddressComponent, FoehnDropdownMenuComponent, FoehnDropdownMenuModule, FoehnErrorPillComponent, FoehnFormComponent, FoehnFormModule, FoehnHeaderComponent, FoehnHeaderModule, FoehnHelpModalComponent, FoehnHelpModalModule, FoehnIconCalendarComponent, FoehnIconCheckComponent, FoehnIconCheckSquareOComponent, FoehnIconChevronDownComponent, FoehnIconChevronLeftComponent, FoehnIconChevronRightComponent, FoehnIconChevronUpComponent, FoehnIconClockComponent, FoehnIconCommentDotsComponent, FoehnIconEditComponent, FoehnIconExternalLinkAltComponent, FoehnIconFilePdfComponent, FoehnIconInfoCircleComponent, FoehnIconLockComponent, FoehnIconMapMarkerComponent, FoehnIconMinusCircleComponent, FoehnIconPencilComponent, FoehnIconPlusCircleComponent, FoehnIconPlusSquareComponent, FoehnIconSearchComponent, FoehnIconTimesComponent, FoehnIconTrashAltComponent, FoehnIconUnlockAltComponent, FoehnIconUserComponent, FoehnIconsModule, FoehnInputAddressComponent, FoehnInputComponent, FoehnInputEmailComponent, FoehnInputForeignLocalityComponent, FoehnInputForeignStreetComponent, FoehnInputHiddenComponent, FoehnInputModule, FoehnInputNav13Component, FoehnInputNav13Module, FoehnInputNumberComponent, FoehnInputPasswordComponent, FoehnInputPhoneComponent, FoehnInputPrefixedTextComponent, FoehnInputStringComponent, FoehnInputTextComponent, FoehnInputTextareaComponent, FoehnListComponent, FoehnListItem, FoehnListModule, FoehnListSummaryComponent, FoehnMenuItemComponent, FoehnMenuItemTransmitComponent, FoehnMenuPrestationModule, FoehnMiscModule, FoehnModalComponent, FoehnModalModule, FoehnMultiUploadComponent, FoehnMultiUploadModule, FoehnMultiselectAutocompleteComponent, FoehnMultiselectAutocompleteModule, FoehnNavigationComponent, FoehnNavigationModule, FoehnNavigationService, FoehnNotFoundModule, FoehnNotfoundComponent, FoehnPageComponent, FoehnPageCounterComponent, FoehnPageExpirationTimerComponent, FoehnPageModalComponent, FoehnPageModule, FoehnPageService, FoehnPictureUploadComponent, FoehnPictureUploadModule, FoehnRadioComponent, FoehnRecapSectionComponent, FoehnRecapSectionModule, FoehnRemainingAlertsSummaryComponent, FoehnRemainingAlertsSummaryModule, FoehnSelectComponent, FoehnSimpleNavigationComponent, FoehnSkipLinkComponent, FoehnStatusProgressBarComponent, FoehnStatusProgressBarModule, FoehnTableColumnConfiguration, FoehnTableComponent, FoehnTableModule, FoehnTablePageChangeEvent, FoehnTimeComponent, FoehnTransmitWaitingModalComponent, FoehnTransmitWaitingModalService, FoehnUploadProgressBarComponent, FoehnUploadProgressBarModule, FoehnUserConnectedAsComponent, FoehnUserConnectedAsModule, FoehnValidationAlertSummaryComponent, FoehnValidationAlertSummaryModule, FoehnValidationAlertsComponent, FoehnValidationAlertsModule, FooterLink, FormMetadata, FormPostResponse, FormSelectOption, FormatIdePipe, FormatterModule, GESDEM_MAX_DATA_LENGTH, GesdemActionRecoveryLoginComponent, GesdemActionRecoveryModule, GesdemActionRecoveryRegistrationComponent, GesdemConfirmationComponent, GesdemConfirmationModule, GesdemErrorComponent, GesdemErrorModule, GesdemEventService, GesdemHandlerService, GesdemStatutUtils, GrowlBrokerService, GrowlMessage, GrowlType, HTTP_LOADER_FILTERED_URL, I18nForm, IbanFormatterDirective, IdeFormatterDirective, Locality, MonthYear, MultiUploadService, Municipality, NDCFormatterDirective, NavigationDirection, NumberCurrencyFormatterDirective, ObjectHelper, PORTAIL_BASE_URL_INT, PageChangeEvent, PageUploadLimitService, PaginationWeek, PendingFiles, PendingUploadService, PipeModule, PlaceOfOrigin, Portail, PostalLocality, Preferences, PrestationsNgCoreModule, RECAPTCHA_API_URL, RecaptchaService, RedirectComponent, RegisterNgModelService, SESSION_INFO_API_URL, SWISS_ISO_ID, SdkDictionaryModule, SdkDictionaryPipe, SdkDictionaryService, SdkEpaymentComponent, SdkEpaymentModule, SdkRecaptchaComponent, SdkRecaptchaModule, SdkRedirectModule, SdkStatisticsService, SelectedSlot, ServiceLocator, Session, SessionInfo, SessionInfoData, SessionInfoWithApplicationService, Street, StreetNumber, THOUSANDS_SEPARATOR, TRANSMIT_WAITING_MODAL_DELAY_IN_MILLISECONDS, TableSort, UploadProgress, UploadProgressService, UploaderHelper, ValidationHandlerService, formatDecimalCurrency, formatNonDecimalCurrency, formatNumberAsGiven, gesdemLoaderGuard, getSafeReference, replaceAll };
15306
15322
  //# sourceMappingURL=dsivd-prestations-ng.mjs.map