@dsivd/prestations-ng 17.8.2-beta.2 → 17.9.0-beta.1
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 +2 -0
- package/UPGRADING_V17.md +12 -8
- package/UPGRADING_V18.md +21 -0
- package/dsivd-prestations-ng-17.9.0-beta.1.tgz +0 -0
- package/esm2022/sdk-appinfo/application-info.mjs +1 -1
- package/esm2022/sdk-recaptcha/grecaptcha/grecaptcha.component.mjs +11 -8
- package/esm2022/sdk-recaptcha/recaptcha.service.mjs +9 -13
- package/esm2022/sdk-recaptcha/sdk-recaptcha.component.mjs +10 -6
- package/fesm2022/dsivd-prestations-ng.mjs +27 -24
- package/fesm2022/dsivd-prestations-ng.mjs.map +1 -1
- package/package.json +1 -1
- package/sdk-appinfo/application-info.d.ts +1 -0
- package/sdk-recaptcha/recaptcha.service.d.ts +4 -5
- package/sdk-recaptcha/sdk-recaptcha.component.d.ts +1 -1
- package/dsivd-prestations-ng-17.8.2-beta.2.tgz +0 -0
package/CHANGELOG.md
CHANGED
package/UPGRADING_V17.md
CHANGED
|
@@ -251,16 +251,20 @@ If you have captcha cloudflare setup already, you need to remove `CAPTCHA_URL_TO
|
|
|
251
251
|
})
|
|
252
252
|
```
|
|
253
253
|
|
|
254
|
-
|
|
254
|
+
#### Cloudflare turnstile verify delegated to Cybercache
|
|
255
255
|
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
256
|
+
Verification of the captcha is now made by Cybercache and not by the application itself (through `prestations-be`). You should remove all properties like `recaptcha.XXX`.
|
|
257
|
+
All these properties are now stored in Cybercache's configuration.
|
|
258
|
+
|
|
259
|
+
Therefore, you can remove properties related to this configuration :
|
|
260
|
+
|
|
261
|
+
- `recaptcha.byPassUUID`
|
|
262
|
+
- `recaptcha.proxyUrl`
|
|
263
|
+
- `recaptcha.proxyPort`
|
|
264
|
+
- `recaptcha.secretkey`
|
|
265
|
+
- `recaptcha.publickey`
|
|
262
266
|
|
|
263
|
-
|
|
267
|
+
Cookie storage is kept as before, therefore you can keep cookie related properties that starts with `recaptcha.cookie.YYY` in your properties.
|
|
264
268
|
|
|
265
269
|
### gesdem-confirmation / foehn-feedback-notification
|
|
266
270
|
|
package/UPGRADING_V18.md
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
# PRESTAKIT : Upgrading from v17 to v18
|
|
2
|
+
|
|
3
|
+
For the commands in this guide, it is supposed that your are in the `front` folder of your project.
|
|
4
|
+
|
|
5
|
+
## PRESTATIONS-BE
|
|
6
|
+
|
|
7
|
+
From here on, we are back in the root folder of your project
|
|
8
|
+
|
|
9
|
+
### Captcha configuration
|
|
10
|
+
|
|
11
|
+
- As stated in the migration guide from v16 to v17, Cybercache is responsible of the captcha turnstile verify call.
|
|
12
|
+
- Therefore, you can remove properties related to this configuration :
|
|
13
|
+
- `recaptcha.byPassUUID`
|
|
14
|
+
- `recaptcha.proxyUrl`
|
|
15
|
+
- `recaptcha.proxyPort`
|
|
16
|
+
- `recaptcha.secretkey`
|
|
17
|
+
- `recaptcha.publickey`
|
|
18
|
+
|
|
19
|
+
Cookie storage is kept as before, therefore you can keep cookie related properties that starts with `recaptcha.cookie.YYY` in your properties.
|
|
20
|
+
|
|
21
|
+
See [Skeleton](https://dsigit.etat-de-vaud.ch/outils/git/projects/CYBSDK/repos/skeleton/pull-requests/708/overview) for a working example
|
|
Binary file
|
|
@@ -19,4 +19,4 @@ export class Captcha {
|
|
|
19
19
|
}
|
|
20
20
|
export class Portail {
|
|
21
21
|
}
|
|
22
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwbGljYXRpb24taW5mby5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3ByZXN0YXRpb25zLW5nL3NyYy9zZGstYXBwaW5mby9hcHBsaWNhdGlvbi1pbmZvLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLHNDQUFzQztBQUN0QyxNQUFNLE9BQU8sZUFBZTtJQUE1QjtRQVFJLGtCQUFhLEdBQWtCLElBQUksYUFBYSxFQUFFLENBQUM7SUFFdkQsQ0FBQztDQUFBO0FBRUQsTUFBTSxPQUFPLFNBQVM7Q0FhckI7QUFFRCxNQUFNLE9BQU8sYUFBYTtJQUExQjtRQUVJLFlBQU8sR0FBWSxJQUFJLE9BQU8sRUFBRSxDQUFDO0lBR3JDLENBQUM7Q0FBQTtBQUVELE1BQU0sT0FBTyxPQUFPO0NBR25CO0FBRUQsTUFBTSxPQUFPLFFBQVE7Q0FLcEI7QUFFRCxNQUFNLE9BQU8sT0FBTztDQUduQjtBQUVELE1BQU0sT0FBTyxPQUFPO0NBT25CIiwic291cmNlc0NvbnRlbnQiOlsiLy8gZXNsaW50LWRpc2FibGUgbWF4LWNsYXNzZXMtcGVyLWZpbGVcbmV4cG9ydCBjbGFzcyBBcHBsaWNhdGlvbkluZm8ge1xuICAgIHZlcnNpb246IHN0cmluZztcbiAgICBidWlsZHRpbWU6IHN0cmluZztcbiAgICBjb250ZXh0UGF0aDogc3RyaW5nO1xuICAgIGVudmlyb25tZW50OiBzdHJpbmc7XG4gICAgZW52aXJvbm1lbnRMYWJlbDogc3RyaW5nO1xuICAgIGFwcFByb3BlcnRpZXM6IHsgW2tleTogc3RyaW5nXTogc3RyaW5nIH07XG4gICAgZXRhcGVJbmZvczogeyBba2V5OiBzdHJpbmddOiBFdGFwZUluZm8gfTtcbiAgICBjb25maWd1cmF0aW9uOiBDb25maWd1cmF0aW9uID0gbmV3IENvbmZpZ3VyYXRpb24oKTtcbiAgICBtb2NrZWRTZXJ2aWNlczogc3RyaW5nW107XG59XG5cbmV4cG9ydCBjbGFzcyBFdGFwZUluZm8ge1xuICAgIHRpdHJlRXRhcGU6IHN0cmluZztcbiAgICBpZE1ldGllckV0YXBlOiBzdHJpbmc7XG4gICAgcmVwcmlzZVBvc3NpYmxlOiBib29sZWFuO1xuICAgIHRpdHJlUHJlc3RhdGlvbjogc3RyaW5nO1xuICAgIGlkTWV0aWVyUHJlc3RhdGlvbjogc3RyaW5nO1xuICAgIHB1YmxpY0NpYmxlOiBzdHJpbmdbXTtcbiAgICBkZWNpc2lvbkVsZWN0cm9uaXF1ZUF2YWlsYWJsZTogYm9vbGVhbjtcbiAgICBkZXBsb3llZUN5YmVyOiBib29sZWFuO1xuICAgIGxpZW5BY2Nlczogc3RyaW5nO1xuICAgIGRyYWZ0RGVtUmV0ZW50aW9uRGF5c1B1YmxpYzogbnVtYmVyO1xuICAgIGRyYWZ0RGVtUmV0ZW50aW9uRGF5c0Nvbm5lY3RlZDogbnVtYmVyO1xuICAgIHVzZXJGZWVkYmFja1JlcXVlc3RlZD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjbGFzcyBDb25maWd1cmF0aW9uIHtcbiAgICBjYXB0Y2hhPzogQ2FwdGNoYTtcbiAgICBwb3J0YWlsOiBQb3J0YWlsID0gbmV3IFBvcnRhaWwoKTtcbiAgICBkb2N1bWVudD86IERvY3VtZW50O1xuICAgIHNlc3Npb24/OiBTZXNzaW9uO1xufVxuXG5leHBvcnQgY2xhc3MgU2Vzc2lvbiB7XG4gICAgc2Vzc2lvblRva2VuTWVzc2FnZURlbGF5SW5NaW51dGVzOiBudW1iZXI7XG4gICAgaWFtTWVzc2FnZURlbGF5SW5NaW51dGVzOiBudW1iZXI7XG59XG5cbmV4cG9ydCBjbGFzcyBEb2N1bWVudCB7XG4gICAgZmlsZU1heFNpemVJbkJ5dGVzQnlGb3JtS2V5OiB7IFtrZXk6IHN0cmluZ106IG51bWJlciB9O1xuICAgIG1heEZpbGVOYW1lTGVuZ3RoOiBzdHJpbmc7XG4gICAgZmlsZUV4dGVuc2lvbnM6IHN0cmluZztcbiAgICBpbGxlZ2FsQ2hhcmFjdGVyczogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgQ2FwdGNoYSB7XG4gICAgcHVibGlja2V5Pzogc3RyaW5nO1xuICAgIGNhcHRjaGFVcmxUb2tlbj86IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIFBvcnRhaWwge1xuICAgIGJhc2VWZENoVXJsPzogc3RyaW5nO1xuICAgIGN5YmVyTG9naW5Vcmw/OiBzdHJpbmc7XG4gICAgc3VwcG9ydEZvcm1Vcmw/OiBzdHJpbmc7XG4gICAgZXNwYWNlU2VjdXJpc2VVcmw/OiBzdHJpbmc7XG4gICAgcG9ydGFpbFRpdGxlc0J5UHJlc3RhQ29kZT86IHsgW2tleTogc3RyaW5nXTogc3RyaW5nIH07XG4gICAgYmFzZVVybD86IHN0cmluZztcbn1cbiJdfQ==
|
|
@@ -46,13 +46,16 @@ export class GrecaptchaComponent {
|
|
|
46
46
|
this.loaded.emit(true);
|
|
47
47
|
}
|
|
48
48
|
appendScriptTag() {
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
49
|
+
// eslint-disable-next-line rxjs-angular/prefer-async-pipe
|
|
50
|
+
this.recaptchaService.getCaptchaScriptUrl().subscribe(url => {
|
|
51
|
+
const script = document.createElement('script');
|
|
52
|
+
script.id = SCRIPT_ID;
|
|
53
|
+
script.src = url;
|
|
54
|
+
script.async = true;
|
|
55
|
+
script.defer = true;
|
|
56
|
+
script.onerror = () => this.scriptError.emit(true);
|
|
57
|
+
this.recaptchaScriptContainer.nativeElement.appendChild(script);
|
|
58
|
+
});
|
|
56
59
|
}
|
|
57
60
|
removeScript() {
|
|
58
61
|
this.scriptError.emit(false);
|
|
@@ -107,4 +110,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
107
110
|
type: ViewChild,
|
|
108
111
|
args: ['recaptchaScriptContainer', { static: true }]
|
|
109
112
|
}] } });
|
|
110
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
113
|
+
//# 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;;;AAIvB,MAAM,CAAC,MAAM,SAAS,GAAG,+BAA+B,CAAC;AAUzD;;;;;;GAMG;AAMH,MAAM,OAAO,mBAAmB;IAwB5B,YAAoB,gBAAkC;QAAlC,qBAAgB,GAAhB,gBAAgB,CAAkB;QAnBtD,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;IAUiB,CAAC;IAE1D,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,CAAC;YAChB,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,CAAC;QACxD,CAAC;QAED,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;YACrB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;QAChE,CAAC;QAED,MAAM,MAAM,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,KAAK,EAAE,OAAO;YACd,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,0DAA0D;QAC1D,IAAI,CAAC,gBAAgB,CAAC,mBAAmB,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACxD,MAAM,MAAM,GAAG,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;YAChD,MAAM,CAAC,EAAE,GAAG,SAAS,CAAC;YACtB,MAAM,CAAC,GAAG,GAAG,GAAG,CAAC;YACjB,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC;YACpB,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC;YACpB,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACnD,IAAI,CAAC,wBAAwB,CAAC,aAAa,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QACpE,CAAC,CAAC,CAAC;IACP,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,CAAC;YACd,IAAI,CAAC,aAAa,EAAE,CAAC;YACrB,OAAO;QACX,CAAC;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,CAAC;YACD,IAAI,CAAC,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;gBACtB,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC3C,CAAC;QACL,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACT,2DAA2D;YAC3D,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACrB,CAAC;IACL,CAAC;+GAjHQ,mBAAmB;mGAAnB,mBAAmB,wdCnChC,0EAEA;;4FDiCa,mBAAmB;kBAL/B,SAAS;+BAEI,gBAAgB;qFAK1B,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\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        // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n        this.recaptchaService.getCaptchaScriptUrl().subscribe(url => {\n            const script = document.createElement('script');\n            script.id = SCRIPT_ID;\n            script.src = 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\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 {
|
|
2
|
+
import { Injectable } from '@angular/core';
|
|
3
3
|
import { BehaviorSubject, combineLatest, EMPTY, merge, of } from 'rxjs';
|
|
4
4
|
import { catchError, debounceTime, map } from 'rxjs/operators';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
@@ -12,12 +12,8 @@ import * as i6 from "../gesdem/gesdem-event.service";
|
|
|
12
12
|
import * as i7 from "../sdk-appinfo/application-info.service";
|
|
13
13
|
export const RECAPTCHA_API_URL = 'api/recaptcha';
|
|
14
14
|
export const CAPTCHA_ERROR_NAME = 'recaptcha';
|
|
15
|
-
export const CAPTCHA_URL_TOKEN = new InjectionToken('captchaScriptUrl', {
|
|
16
|
-
providedIn: 'root',
|
|
17
|
-
factory: () => 'https://challenges.cloudflare.com/turnstile/v0/api.js?onload=reCaptchaLoad&render=explicit&compat=recaptcha'
|
|
18
|
-
});
|
|
19
15
|
export class RecaptchaService {
|
|
20
|
-
constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService
|
|
16
|
+
constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService) {
|
|
21
17
|
this.http = http;
|
|
22
18
|
this.sessionInfo = sessionInfo;
|
|
23
19
|
this.route = route;
|
|
@@ -25,11 +21,11 @@ export class RecaptchaService {
|
|
|
25
21
|
this.gesdemService = gesdemService;
|
|
26
22
|
this.gesdemEventService = gesdemEventService;
|
|
27
23
|
this.applicationInfoService = applicationInfoService;
|
|
28
|
-
this.captchaScriptUrl = captchaScriptUrl;
|
|
29
24
|
this.errorsSubject = new BehaviorSubject([]);
|
|
30
25
|
this.errors = this.initErrorObservable(); // has to be first since it is used in initShouldDisplayObservable !
|
|
31
26
|
this.shouldDisplay = this.initShouldDisplayObservable();
|
|
32
27
|
this.publicKey = this.initPublicKeyObservable();
|
|
28
|
+
this.captchaScriptUrl = this.initCaptchaScriptUrlObservable();
|
|
33
29
|
}
|
|
34
30
|
onSuccess(token) {
|
|
35
31
|
// To be accessed by the gesdem service and to avoid to have a circular
|
|
@@ -153,6 +149,9 @@ export class RecaptchaService {
|
|
|
153
149
|
initPublicKeyObservable() {
|
|
154
150
|
return this.applicationInfoService.data.pipe(map(applicationInfo => this.getPublicKeyOrNull(applicationInfo)));
|
|
155
151
|
}
|
|
152
|
+
initCaptchaScriptUrlObservable() {
|
|
153
|
+
return this.applicationInfoService.data.pipe(map(applicationInfo => applicationInfo?.configuration?.captcha?.captchaUrlToken));
|
|
154
|
+
}
|
|
156
155
|
getPublicKeyOrNull(applicationInfo) {
|
|
157
156
|
// Replace this whole shenanigan with applicationInfo?.configuration?.captcha?.publickey when
|
|
158
157
|
// Typescript supports it.
|
|
@@ -169,7 +168,7 @@ export class RecaptchaService {
|
|
|
169
168
|
}
|
|
170
169
|
return captcha.publickey || null;
|
|
171
170
|
}
|
|
172
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", 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 }
|
|
171
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", 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 }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
173
172
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecaptchaService, providedIn: 'root' }); }
|
|
174
173
|
}
|
|
175
174
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecaptchaService, decorators: [{
|
|
@@ -177,8 +176,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
177
176
|
args: [{
|
|
178
177
|
providedIn: 'root'
|
|
179
178
|
}]
|
|
180
|
-
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.SessionInfo }, { type: i3.ActivatedRoute }, { type: i4.ValidationHandlerService }, { type: i5.GesdemHandlerService }, { type: i6.GesdemEventService }, { type: i7.ApplicationInfoService }
|
|
181
|
-
type: Inject,
|
|
182
|
-
args: [CAPTCHA_URL_TOKEN]
|
|
183
|
-
}] }] });
|
|
184
|
-
//# 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 } 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 { 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://challenges.cloudflare.com/turnstile/v0/api.js?onload=reCaptchaLoad&render=explicit&compat=recaptcha'\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        @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            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"]}
|
|
179
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.SessionInfo }, { type: i3.ActivatedRoute }, { type: i4.ValidationHandlerService }, { type: i5.GesdemHandlerService }, { type: i6.GesdemEventService }, { type: i7.ApplicationInfoService }] });
|
|
180
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recaptcha.service.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/sdk-recaptcha/recaptcha.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAc,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAC9D,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAE3C,OAAO,EACH,eAAe,EACf,aAAa,EACb,KAAK,EACL,KAAK,EAEL,EAAE,EACL,MAAM,MAAM,CAAC;AACd,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;;;;;;;;;AAU/D,MAAM,CAAC,MAAM,iBAAiB,GAAG,eAAe,CAAC;AACjD,MAAM,CAAC,MAAM,kBAAkB,GAAG,WAAW,CAAC;AAK9C,MAAM,OAAO,gBAAgB;IAQzB,YACY,IAAgB,EAChB,WAAwB,EACxB,KAAqB,EACrB,wBAAkD,EAClD,aAAmC,EACnC,kBAAsC,EACtC,sBAA8C;QAN9C,SAAI,GAAJ,IAAI,CAAY;QAChB,gBAAW,GAAX,WAAW,CAAa;QACxB,UAAK,GAAL,KAAK,CAAgB;QACrB,6BAAwB,GAAxB,wBAAwB,CAA0B;QAClD,kBAAa,GAAb,aAAa,CAAsB;QACnC,uBAAkB,GAAlB,kBAAkB,CAAoB;QACtC,2BAAsB,GAAtB,sBAAsB,CAAwB;QAbzC,kBAAa,GAAG,IAAI,eAAe,CAAc,EAAE,CAAC,CAAC;QAelE,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC,CAAC,oEAAoE;QAC9G,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,2BAA2B,EAAE,CAAC;QACxD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAChD,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,8BAA8B,EAAE,CAAC;IAClE,CAAC;IAED,SAAS,CAAC,KAAa;QACnB,uEAAuE;QACvE,uEAAuE;QACvE,IAAI,CAAC,aAAa,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;QAE1C,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QAEnB,IAAI,CAAC,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,IAAI,CAAC,IAAI;iBACJ,IAAI,CAAC,iBAAiB,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC;iBAC9C,IAAI,CACD,UAAU,CAAC,CAAC,CAAU,EAAE,EAAE;gBACtB,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBACjB,+CAA+C;gBAC/C,IAAI,CAAC,QAAQ,CACT,2DAA2D,CAC9D,CAAC;gBACF,OAAO,KAAK,CAAC;YACjB,CAAC,CAAC,CACL;iBACA,SAAS;YACN,8DAA8D;YAC9D,CAAC,IAAS,EAAE,EAAE;gBACV,MAAM,OAAO,GAAG,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC;gBAC/B,IAAI,OAAO,EAAE,CAAC;oBACV,wDAAwD;oBACxD,2DAA2D;oBAC3D,6DAA6D;oBAC7D,IAAI,CAAC,wBAAwB,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;gBACnD,CAAC;qBAAM,CAAC;oBACJ,wDAAwD;oBACxD,uDAAuD;oBACvD,0CAA0C;oBAC1C,IAAI,CAAC,QAAQ,CAAC,4BAA4B,CAAC,CAAC;gBAChD,CAAC;YACL,CAAC,CACJ,CAAC;QACV,CAAC;IACL,CAAC;IAED,QAAQ,CAAC,OAAe;QACpB,MAAM,cAAc,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,EAAE,CAAC;QACrD,MAAM,QAAQ,GAAG;YACb,IAAI,EAAE,kBAAkB;YACxB,IAAI,EAAE,kBAAkB;YACxB,OAAO;SACV,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,GAAG,cAAc,CAAC,CAAC,CAAC;IAC3D,CAAC;IAED,WAAW;QACP,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAChC,CAAC;IAED,SAAS;QACL,OAAO,IAAI,CAAC,MAAM,CAAC;IACvB,CAAC;IAED,gBAAgB;QACZ,OAAO,IAAI,CAAC,aAAa,CAAC;IAC9B,CAAC;IAED,YAAY;QACR,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IAED,QAAQ;QACJ,OAAO,IAAI,CAAC,KAAK,CAAC;IACtB,CAAC;IAED,mBAAmB;QACf,OAAO,IAAI,CAAC,gBAAgB,CAAC;IACjC,CAAC;IAEO,mBAAmB;QACvB,MAAM,oBAAoB,GAAG,KAAK;QAC9B,wCAAwC;QACxC,EAAE,CAAC,EAAE,CAAC;QACN,2CAA2C;QAC3C,IAAI,CAAC,wBAAwB,CAAC,uBAAuB,CACxD,CAAC,IAAI;QACF,iCAAiC;QACjC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,kBAAkB,KAAK,EAAE,CAAC,IAAI,CAAC,CAAC,CACrE,CAAC;QAEF,+EAA+E;QAC/E,OAAO,aAAa,CAAC;YACjB,uDAAuD;YACvD,oBAAoB;YACpB,gBAAgB;YAChB,IAAI,CAAC,aAAa;SACrB,CAAC,CAAC,IAAI;QACH,4BAA4B;QAC5B,GAAG,CAAC,CAAC,CAAC,YAAY,EAAE,WAAW,CAAC,EAAE,EAAE,CAAC;YACjC,GAAG,YAAY;YACf,GAAG,WAAW;SACjB,CAAC,CACL,CAAC;IACN,CAAC;IAEO,2BAA2B;QAC/B,MAAM,qBAAqB,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CACpD,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,cAAc,IAAI,CAAC,CAAC,IAAI,CAAC,CAC1D,CAAC;QAEF,MAAM,0BAA0B,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC,IAAI,CAC9D,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAChC,CAAC;QAEF,MAAM,sBAAsB,GAAG,KAAK;QAChC,6DAA6D;QAC7D,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC;QACnC,4BAA4B;QAC5B,IAAI,CAAC,kBAAkB,CAAC,4BAA4B,EAAE,CACzD,CAAC,IAAI,CACF,GAAG,CACC,IAAI,CAAC,EAAE,CACH,CAAC,CAAC,IAAI;YACN,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY;YACjC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI;YACtC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CACvD,CACJ,CAAC;QAEF,OAAO,aAAa,CAAC;YACjB,qBAAqB;YACrB,0BAA0B;YAC1B,sBAAsB;YACtB,IAAI,CAAC,MAAM;SACd,CAAC,CAAC,IAAI;QACH,gEAAgE;QAChE,uBAAuB;QACvB,YAAY,CAAC,CAAC,CAAC,EACf,GAAG,CACC,CAAC,CACG,WAAW,EACX,gBAAgB,EAChB,YAAY,EACZ,aAAa,CAChB,EAAE,EAAE;YACD,MAAM,eAAe,GAAG,CAAC,CAAC,aAAa,EAAE,MAAM,CAAC;YAChD,IAAI,eAAe,EAAE,CAAC;gBAClB,+FAA+F;gBAC/F,OAAO,IAAI,CAAC;YAChB,CAAC;YACD,uEAAuE;YACvE,uDAAuD;YACvD,OAAO,CAAC,WAAW,IAAI,CAAC,gBAAgB,IAAI,CAAC,YAAY,CAAC;QAC9D,CAAC,CACJ,CACJ,CAAC;IACN,CAAC;IAED,8DAA8D;IACtD,mBAAmB;QACvB,MAAM,MAAM,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,WAAW,CAAC,mBAAmB;YAChE,CAAC,CAAC,IAAI,UAAU,EAAE,CAAC,GAAG,CAChB,qBAAqB,EACrB,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,WAAW,CAAC,mBAAmB,CACtD;YACH,CAAC,CAAC,IAAI,CAAC;QAEX,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,iBAAiB,SAAS,EAAE,EAAE,MAAM,EAAE,CAAC,CAAC;IACpE,CAAC;IAEO,uBAAuB;QAC3B,OAAO,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,IAAI,CACxC,GAAG,CAAC,eAAe,CAAC,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,eAAe,CAAC,CAAC,CACnE,CAAC;IACN,CAAC;IAEO,8BAA8B;QAClC,OAAO,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,IAAI,CACxC,GAAG,CACC,eAAe,CAAC,EAAE,CACd,eAAe,EAAE,aAAa,EAAE,OAAO,EAAE,eAAe,CAC/D,CACJ,CAAC;IACN,CAAC;IAEO,kBAAkB,CACtB,eAAgC;QAEhC,6FAA6F;QAC7F,0BAA0B;QAC1B,IAAI,CAAC,eAAe,EAAE,CAAC;YACnB,OAAO,IAAI,CAAC;QAChB,CAAC;QAED,MAAM,EAAE,aAAa,EAAE,GAAG,eAAe,CAAC;QAE1C,IAAI,CAAC,aAAa,EAAE,CAAC;YACjB,OAAO,IAAI,CAAC;QAChB,CAAC;QACD,MAAM,EAAE,OAAO,EAAE,GAAG,aAAa,CAAC;QAElC,IAAI,CAAC,OAAO,EAAE,CAAC;YACX,OAAO,IAAI,CAAC;QAChB,CAAC;QAED,OAAO,OAAO,CAAC,SAAS,IAAI,IAAI,CAAC;IACrC,CAAC;+GAlOQ,gBAAgB;mHAAhB,gBAAgB,cAFb,MAAM;;4FAET,gBAAgB;kBAH5B,UAAU;mBAAC;oBACR,UAAU,EAAE,MAAM;iBACrB","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 } 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 { 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 readonly captchaScriptUrl: 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    ) {\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        this.captchaScriptUrl = this.initCaptchaScriptUrlObservable();\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(): Observable<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            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 initCaptchaScriptUrlObservable(): Observable<string> {\n        return this.applicationInfoService.data.pipe(\n            map(\n                applicationInfo =>\n                    applicationInfo?.configuration?.captcha?.captchaUrlToken\n            )\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,5 +1,5 @@
|
|
|
1
1
|
import { Component, ViewChild } from '@angular/core';
|
|
2
|
-
import { debounceTime } from 'rxjs/operators';
|
|
2
|
+
import { debounceTime, map } from 'rxjs/operators';
|
|
3
3
|
import { GrecaptchaComponent } from './grecaptcha/grecaptcha.component';
|
|
4
4
|
import * as i0 from "@angular/core";
|
|
5
5
|
import * as i1 from "./recaptcha.service";
|
|
@@ -10,8 +10,12 @@ export class SdkRecaptchaComponent {
|
|
|
10
10
|
this.recaptchaService = recaptchaService;
|
|
11
11
|
this.cdr = cdr;
|
|
12
12
|
this.hasLoadingError = false;
|
|
13
|
-
|
|
14
|
-
|
|
13
|
+
this.recaptchaUrlForUserError = this.recaptchaService
|
|
14
|
+
.getCaptchaScriptUrl()
|
|
15
|
+
.pipe(map(captchaScriptUrl => {
|
|
16
|
+
const url = new URL(captchaScriptUrl);
|
|
17
|
+
return `${url.protocol}//${url.hostname}`;
|
|
18
|
+
}));
|
|
15
19
|
}
|
|
16
20
|
ngOnInit() {
|
|
17
21
|
this.recaptchaService.clearErrors();
|
|
@@ -59,13 +63,13 @@ export class SdkRecaptchaComponent {
|
|
|
59
63
|
this.recaptchaService.onSuccess(token);
|
|
60
64
|
}
|
|
61
65
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SdkRecaptchaComponent, deps: [{ token: i1.RecaptchaService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
62
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SdkRecaptchaComponent, selector: "captcha", viewQueries: [{ propertyName: "grecaptcha", first: true, predicate: GrecaptchaComponent, descendants: true }], ngImport: i0, template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a [href]=\"
|
|
66
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SdkRecaptchaComponent, selector: "captcha", viewQueries: [{ propertyName: "grecaptcha", first: true, predicate: GrecaptchaComponent, descendants: true }], ngImport: i0, template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a\n *ngIf=\"\n recaptchaUrlForUserError | async as recaptchaUrlUserError\n \"\n [href]=\"recaptchaUrlUserError\"\n target=\"_blank\"\n >\n {{ recaptchaUrlUserError }}\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", dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.GrecaptchaComponent, selector: "app-grecaptcha", inputs: ["siteKey"], outputs: ["loaded", "scriptError", "tokenOnSuccess", "destroyed"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }] }); }
|
|
63
67
|
}
|
|
64
68
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SdkRecaptchaComponent, decorators: [{
|
|
65
69
|
type: Component,
|
|
66
|
-
args: [{ selector: 'captcha', template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a [href]=\"
|
|
70
|
+
args: [{ selector: 'captcha', template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a\n *ngIf=\"\n recaptchaUrlForUserError | async as recaptchaUrlUserError\n \"\n [href]=\"recaptchaUrlUserError\"\n target=\"_blank\"\n >\n {{ recaptchaUrlUserError }}\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" }]
|
|
67
71
|
}], ctorParameters: () => [{ type: i1.RecaptchaService }, { type: i0.ChangeDetectorRef }], propDecorators: { grecaptcha: [{
|
|
68
72
|
type: ViewChild,
|
|
69
73
|
args: [GrecaptchaComponent]
|
|
70
74
|
}] } });
|
|
71
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
75
|
+
//# 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,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAGnD,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,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC,gBAAgB;aAChD,mBAAmB,EAAE;aACrB,IAAI,CACD,GAAG,CAAC,gBAAgB,CAAC,EAAE;YACnB,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,gBAAgB,CAAC,CAAC;YACtC,OAAO,GAAG,GAAG,CAAC,QAAQ,KAAK,GAAG,CAAC,QAAQ,EAAE,CAAC;QAC9C,CAAC,CAAC,CACL,CAAC;IACV,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,CAAC;gBACzB,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC7B,CAAC;YACD,yDAAyD;YACzD,cAAc;YACd,IAAI,MAAM,CAAC,MAAM,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACnC,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;YAC5B,CAAC;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,CAAC;YAC1B,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,CAAC;QAC1C,CAAC;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,CAAC;YACzB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC7B,CAAC;IACL,CAAC;IAED,cAAc,CAAC,WAAoB;QAC/B,IAAI,CAAC,eAAe,GAAG,WAAW,CAAC;QAEnC,6CAA6C;QAC7C,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC7B,CAAC;IACL,CAAC;IAED,cAAc,CAAC,KAAa;QACxB,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IAC3C,CAAC;+GAjFQ,qBAAqB;mGAArB,qBAAqB,2FACnB,mBAAmB,gDCpBlC,szDA+CA;;4FD5Ba,qBAAqB;kBALjC,SAAS;+BAEI,SAAS;qHAKnB,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, map } 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    recaptchaUrlForUserError: Observable<string>;\n\n    private errorsSubscription: Subscription;\n    private forceDetectChange: () => void;\n\n    constructor(\n        private recaptchaService: RecaptchaService,\n        private cdr: ChangeDetectorRef\n    ) {\n        this.recaptchaUrlForUserError = this.recaptchaService\n            .getCaptchaScriptUrl()\n            .pipe(\n                map(captchaScriptUrl => {\n                    const url = new URL(captchaScriptUrl);\n                    return `${url.protocol}//${url.hostname}`;\n                })\n            );\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\n                *ngIf=\"\n                    recaptchaUrlForUserError | async as recaptchaUrlUserError\n                \"\n                [href]=\"recaptchaUrlUserError\"\n                target=\"_blank\"\n            >\n                {{ recaptchaUrlUserError }}\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 { Optional, Inject, Injectable, EventEmitter, Input, HostBinding, Output, forwardRef, ViewChildren, ViewChild, Directive, Pipe, Component, ContentChildren, HostListener,
|
|
2
|
+
import { Optional, Inject, Injectable, EventEmitter, Input, HostBinding, Output, forwardRef, ViewChildren, ViewChild, Directive, Pipe, Component, ContentChildren, HostListener, NgModule, TemplateRef, ContentChild } 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, Subject, BehaviorSubject, combineLatest, throwError, switchMap as switchMap$1, forkJoin, tap as tap$1, concat, toArray, EMPTY, startWith, filter as filter$1, merge, withLatestFrom, debounceTime as debounceTime$1, map as map$1 } from 'rxjs';
|
|
@@ -4315,12 +4315,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
4315
4315
|
|
|
4316
4316
|
const RECAPTCHA_API_URL = 'api/recaptcha';
|
|
4317
4317
|
const CAPTCHA_ERROR_NAME = 'recaptcha';
|
|
4318
|
-
const CAPTCHA_URL_TOKEN = new InjectionToken('captchaScriptUrl', {
|
|
4319
|
-
providedIn: 'root',
|
|
4320
|
-
factory: () => 'https://challenges.cloudflare.com/turnstile/v0/api.js?onload=reCaptchaLoad&render=explicit&compat=recaptcha'
|
|
4321
|
-
});
|
|
4322
4318
|
class RecaptchaService {
|
|
4323
|
-
constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService
|
|
4319
|
+
constructor(http, sessionInfo, route, validationHandlerService, gesdemService, gesdemEventService, applicationInfoService) {
|
|
4324
4320
|
this.http = http;
|
|
4325
4321
|
this.sessionInfo = sessionInfo;
|
|
4326
4322
|
this.route = route;
|
|
@@ -4328,11 +4324,11 @@ class RecaptchaService {
|
|
|
4328
4324
|
this.gesdemService = gesdemService;
|
|
4329
4325
|
this.gesdemEventService = gesdemEventService;
|
|
4330
4326
|
this.applicationInfoService = applicationInfoService;
|
|
4331
|
-
this.captchaScriptUrl = captchaScriptUrl;
|
|
4332
4327
|
this.errorsSubject = new BehaviorSubject([]);
|
|
4333
4328
|
this.errors = this.initErrorObservable(); // has to be first since it is used in initShouldDisplayObservable !
|
|
4334
4329
|
this.shouldDisplay = this.initShouldDisplayObservable();
|
|
4335
4330
|
this.publicKey = this.initPublicKeyObservable();
|
|
4331
|
+
this.captchaScriptUrl = this.initCaptchaScriptUrlObservable();
|
|
4336
4332
|
}
|
|
4337
4333
|
onSuccess(token) {
|
|
4338
4334
|
// To be accessed by the gesdem service and to avoid to have a circular
|
|
@@ -4456,6 +4452,9 @@ class RecaptchaService {
|
|
|
4456
4452
|
initPublicKeyObservable() {
|
|
4457
4453
|
return this.applicationInfoService.data.pipe(map(applicationInfo => this.getPublicKeyOrNull(applicationInfo)));
|
|
4458
4454
|
}
|
|
4455
|
+
initCaptchaScriptUrlObservable() {
|
|
4456
|
+
return this.applicationInfoService.data.pipe(map(applicationInfo => applicationInfo?.configuration?.captcha?.captchaUrlToken));
|
|
4457
|
+
}
|
|
4459
4458
|
getPublicKeyOrNull(applicationInfo) {
|
|
4460
4459
|
// Replace this whole shenanigan with applicationInfo?.configuration?.captcha?.publickey when
|
|
4461
4460
|
// Typescript supports it.
|
|
@@ -4472,7 +4471,7 @@ class RecaptchaService {
|
|
|
4472
4471
|
}
|
|
4473
4472
|
return captcha.publickey || null;
|
|
4474
4473
|
}
|
|
4475
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecaptchaService, deps: [{ token: i1.HttpClient }, { token: SessionInfo }, { token: i1$1.ActivatedRoute }, { token: ValidationHandlerService }, { token: GesdemHandlerService }, { token: GesdemEventService }, { token: ApplicationInfoService }
|
|
4474
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecaptchaService, deps: [{ token: i1.HttpClient }, { token: SessionInfo }, { token: i1$1.ActivatedRoute }, { token: ValidationHandlerService }, { token: GesdemHandlerService }, { token: GesdemEventService }, { token: ApplicationInfoService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
4476
4475
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecaptchaService, providedIn: 'root' }); }
|
|
4477
4476
|
}
|
|
4478
4477
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RecaptchaService, decorators: [{
|
|
@@ -4480,10 +4479,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
4480
4479
|
args: [{
|
|
4481
4480
|
providedIn: 'root'
|
|
4482
4481
|
}]
|
|
4483
|
-
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: SessionInfo }, { type: i1$1.ActivatedRoute }, { type: ValidationHandlerService }, { type: GesdemHandlerService }, { type: GesdemEventService }, { type: ApplicationInfoService }
|
|
4484
|
-
type: Inject,
|
|
4485
|
-
args: [CAPTCHA_URL_TOKEN]
|
|
4486
|
-
}] }] });
|
|
4482
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: SessionInfo }, { type: i1$1.ActivatedRoute }, { type: ValidationHandlerService }, { type: GesdemHandlerService }, { type: GesdemEventService }, { type: ApplicationInfoService }] });
|
|
4487
4483
|
|
|
4488
4484
|
const SCRIPT_ID = 'prestations-ng-captcha-script';
|
|
4489
4485
|
/**
|
|
@@ -4530,13 +4526,16 @@ class GrecaptchaComponent {
|
|
|
4530
4526
|
this.loaded.emit(true);
|
|
4531
4527
|
}
|
|
4532
4528
|
appendScriptTag() {
|
|
4533
|
-
|
|
4534
|
-
|
|
4535
|
-
|
|
4536
|
-
|
|
4537
|
-
|
|
4538
|
-
|
|
4539
|
-
|
|
4529
|
+
// eslint-disable-next-line rxjs-angular/prefer-async-pipe
|
|
4530
|
+
this.recaptchaService.getCaptchaScriptUrl().subscribe(url => {
|
|
4531
|
+
const script = document.createElement('script');
|
|
4532
|
+
script.id = SCRIPT_ID;
|
|
4533
|
+
script.src = url;
|
|
4534
|
+
script.async = true;
|
|
4535
|
+
script.defer = true;
|
|
4536
|
+
script.onerror = () => this.scriptError.emit(true);
|
|
4537
|
+
this.recaptchaScriptContainer.nativeElement.appendChild(script);
|
|
4538
|
+
});
|
|
4540
4539
|
}
|
|
4541
4540
|
removeScript() {
|
|
4542
4541
|
this.scriptError.emit(false);
|
|
@@ -4597,8 +4596,12 @@ class SdkRecaptchaComponent {
|
|
|
4597
4596
|
this.recaptchaService = recaptchaService;
|
|
4598
4597
|
this.cdr = cdr;
|
|
4599
4598
|
this.hasLoadingError = false;
|
|
4600
|
-
|
|
4601
|
-
|
|
4599
|
+
this.recaptchaUrlForUserError = this.recaptchaService
|
|
4600
|
+
.getCaptchaScriptUrl()
|
|
4601
|
+
.pipe(map(captchaScriptUrl => {
|
|
4602
|
+
const url = new URL(captchaScriptUrl);
|
|
4603
|
+
return `${url.protocol}//${url.hostname}`;
|
|
4604
|
+
}));
|
|
4602
4605
|
}
|
|
4603
4606
|
ngOnInit() {
|
|
4604
4607
|
this.recaptchaService.clearErrors();
|
|
@@ -4646,11 +4649,11 @@ class SdkRecaptchaComponent {
|
|
|
4646
4649
|
this.recaptchaService.onSuccess(token);
|
|
4647
4650
|
}
|
|
4648
4651
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SdkRecaptchaComponent, deps: [{ token: RecaptchaService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
4649
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SdkRecaptchaComponent, selector: "captcha", viewQueries: [{ propertyName: "grecaptcha", first: true, predicate: GrecaptchaComponent, descendants: true }], ngImport: i0, template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a [href]=\"
|
|
4652
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SdkRecaptchaComponent, selector: "captcha", viewQueries: [{ propertyName: "grecaptcha", first: true, predicate: GrecaptchaComponent, descendants: true }], ngImport: i0, template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a\n *ngIf=\"\n recaptchaUrlForUserError | async as recaptchaUrlUserError\n \"\n [href]=\"recaptchaUrlUserError\"\n target=\"_blank\"\n >\n {{ recaptchaUrlUserError }}\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", dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: GrecaptchaComponent, selector: "app-grecaptcha", inputs: ["siteKey"], outputs: ["loaded", "scriptError", "tokenOnSuccess", "destroyed"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }] }); }
|
|
4650
4653
|
}
|
|
4651
4654
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SdkRecaptchaComponent, decorators: [{
|
|
4652
4655
|
type: Component,
|
|
4653
|
-
args: [{ selector: 'captcha', template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a [href]=\"
|
|
4656
|
+
args: [{ selector: 'captcha', template: "<!-- 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\u00E9rifier que votre navigateur peut acc\u00E9der \u00E0\n <a\n *ngIf=\"\n recaptchaUrlForUserError | async as recaptchaUrlUserError\n \"\n [href]=\"recaptchaUrlUserError\"\n target=\"_blank\"\n >\n {{ recaptchaUrlUserError }}\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" }]
|
|
4654
4657
|
}], ctorParameters: () => [{ type: RecaptchaService }, { type: i0.ChangeDetectorRef }], propDecorators: { grecaptcha: [{
|
|
4655
4658
|
type: ViewChild,
|
|
4656
4659
|
args: [GrecaptchaComponent]
|
|
@@ -16023,5 +16026,5 @@ class DropdownMenuItem {
|
|
|
16023
16026
|
* Generated bundle index. Do not edit.
|
|
16024
16027
|
*/
|
|
16025
16028
|
|
|
16026
|
-
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,
|
|
16029
|
+
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, DemandeHelper, DisplayCurrencyPipe, DisplayDatePipe, District, Document, DocumentError, DocumentReference, DocumentReferenceWithFile, DocumentsWithErrors, DropdownMenuGroup, DropdownMenuItem, EPaymentService, EtapeInfo, FEEDBACK_USAGERS_LOCAL_STORAGE_PREFIX, 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, FoehnFeedbackNotificationComponent, FoehnFeedbackNotificationModule, 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, IntervalHelper, Locality, MonthYear, MultiUploadService, Municipality, NDCFormatterDirective, NavigationDirection, NumberCurrencyFormatterDirective, ObjectHelper, PORTAIL_BASE_URL_INT, PORTAIL_BASE_URL_PROD, PORTAIL_BASE_URL_VAL, PageChangeEvent, PageUploadLimitService, PaginationWeek, PendingFiles, PendingPaymentComponent, PendingUploadService, PipeModule, PlaceOfOrigin, Portail, PostalLocality, Preferences, PrestationsNgCoreModule, RECAPTCHA_API_URL, RecaptchaService, RedirectComponent, RegisterNgModelService, RowElement, SESSION_INFO_API_URL, SWISS_ISO_ID, SdkDictionaryModule, SdkDictionaryPipe, SdkDictionaryService, SdkEpaymentComponent, SdkEpaymentModule, SdkEvent, SdkEventType, SdkEventsLoggerService, 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, replaceAll };
|
|
16027
16030
|
//# sourceMappingURL=dsivd-prestations-ng.mjs.map
|