@dsivd/prestations-ng 16.1.0-beta.2 → 16.1.0-beta.3

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.
Files changed (24) hide show
  1. package/CHANGELOG.md +32 -2
  2. package/dsivd-prestations-ng-v16.1.0-beta.3.tgz +0 -0
  3. package/esm2020/foehn-upload/foehn-bo-multi-upload/bo-multi-upload.service.mjs +64 -23
  4. package/esm2020/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.mjs +9 -9
  5. package/esm2020/foehn-upload/foehn-multi-upload/multi-upload.service.mjs +65 -26
  6. package/esm2020/foehn-upload/foehn-upload-progress-bar/foehn-upload-progress-bar.component.mjs +14 -7
  7. package/esm2020/foehn-upload/foehn-upload-progress-bar/upload-progress.service.mjs +8 -4
  8. package/esm2020/foehn-upload/foehn-upload-progress-bar/upload-progress.type.mjs +5 -1
  9. package/esm2020/gesdem-action-recovery/gesdem-action-recovery-registration/gesdem-action-recovery-registration.component.mjs +14 -7
  10. package/esm2020/gesdem-action-recovery/gesdem-action-recovery.module.mjs +12 -4
  11. package/esm2020/sdk-dictionary/default-dictionary.mjs +20 -1
  12. package/fesm2015/dsivd-prestations-ng.mjs +202 -76
  13. package/fesm2015/dsivd-prestations-ng.mjs.map +1 -1
  14. package/fesm2020/dsivd-prestations-ng.mjs +200 -76
  15. package/fesm2020/dsivd-prestations-ng.mjs.map +1 -1
  16. package/foehn-upload/foehn-bo-multi-upload/bo-multi-upload.service.d.ts +3 -0
  17. package/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.d.ts +1 -1
  18. package/foehn-upload/foehn-multi-upload/multi-upload.service.d.ts +3 -0
  19. package/foehn-upload/foehn-upload-progress-bar/foehn-upload-progress-bar.component.d.ts +3 -0
  20. package/foehn-upload/foehn-upload-progress-bar/upload-progress.service.d.ts +5 -5
  21. package/foehn-upload/foehn-upload-progress-bar/upload-progress.type.d.ts +2 -0
  22. package/gesdem-action-recovery/gesdem-action-recovery.module.d.ts +3 -1
  23. package/package.json +1 -1
  24. package/dsivd-prestations-ng-v16.1.0-beta.2.tgz +0 -0
package/CHANGELOG.md CHANGED
@@ -39,12 +39,42 @@ A change is considered **breaking** if you have to change your code or update yo
39
39
  - added field `cantonCode: string`
40
40
 
41
41
  - [foehn-menu-item-transmit.component.ts](projects/prestations-ng/src/foehn-menu-prestation/foehn-menu-items/foehn-menu-item-transmit/foehn-menu-item-transmit.component.ts)
42
- - added `@Ouput() clickedWithRemainingErrors = new EventEmitter<number>();` to be able to react to a click one the button when the form still has errors
42
+
43
+ - added `@Ouput() clickedWithRemainingErrors = new EventEmitter<number>();` to be able to react to a click on the button when the form still has errors
44
+
45
+ - [default-dictionary.ts](projects/prestations-ng/src/sdk-dictionary/default-dictionary.ts)
46
+ - added keys:
47
+ - `foehn-uploader.files-saved-success-message` (default message: `{successfulDocumentsCount} fichier(s) sauvegardé(s) avec succès`)
48
+ - `foehn-uploader.transmission-error-message` (default message: `Une erreur est survenue lors de la transmission de vos documents`)
49
+ - `foehn-uploader.files-deleted-success-message` (default message: `Suppression du fichier {filename} réussie`)
50
+ - `foehn-uploader.delete-error-message` (default message: `Une erreur est survenue lors de la suppression de votre document`)
51
+ - `foehn-upload-progress-bar.group.upload` (default message: `Lot {packageIndex}/{nbPackages}: Enregistrement en cours...`)
52
+ - `foehn-upload-progress-bar.group.analysis` (default message: `Lot {packageIndex}/{nbPackages}: Analyse en cours... Merci de patienter.`)
53
+ - `foehn-upload-progress-bar.group.info` (default message: `Le téléchargement se fait en plusieurs lots en fonction du nombre de fichiers chargés`)
43
54
 
44
55
  ### Fixed
45
56
 
46
57
  - [uploader.helper.ts](projects/prestations-ng/src/foehn-upload/uploader.helper.ts)
47
- - handle `applicationInfo.configuration.document.fileMaxSizeInBytesByFormKey` correctly if the key is a regex
58
+ - handle `applicationInfo.configuration.document.fileMaxSizeInBytesByFormKey` correctly if the key is a regex
59
+
60
+ ### Updated
61
+
62
+ - [multi-upload.service.ts](projects/prestations-ng/src/foehn-upload/foehn-multi-upload/multi-upload.service.ts)
63
+ - [bo-multi-upload.service.ts](projects/prestations-ng/src/foehn-upload/foehn-bo-multi-upload/bo-multi-upload.service.ts)
64
+
65
+ - multiple document uploads are now done by packages of 3 files maximum
66
+
67
+ - [upload-progress.service.ts](projects/prestations-ng/src/foehn-upload/foehn-upload-progress-bar/upload-progress.service.ts)
68
+
69
+ - Function `manageUploadEventFilter` has two more parameters `packageIndex: number` (default: 0) and `nbPackages: number` (default: 1)
70
+
71
+ - [foehn-upload-progress-bar.component.html](projects/prestations-ng/src/foehn-upload/foehn-upload-progress-bar/foehn-upload-progress-bar.component.html)
72
+
73
+ - Upload and analysis messages have been updated to take into account packages upload
74
+
75
+ - [gesdem-action-recovery-registration.component.ts](projects/prestations-ng/src/gesdem-action-recovery/gesdem-action-recovery-registration/gesdem-action-recovery-registration.component.ts)
76
+ - reload form from the backend after a successfull registration
77
+ - added default labels in [default-dictionary.ts](projects/prestations-ng/src/sdk-dictionary/default-dictionary.ts)
48
78
 
49
79
  ## [16.0.7]
50
80
 
@@ -1,8 +1,9 @@
1
1
  import { Injectable } from '@angular/core';
2
- import { throwError } from 'rxjs';
2
+ import { concat, throwError, toArray } from 'rxjs';
3
3
  import { catchError, filter, map, tap } from 'rxjs/operators';
4
4
  import { GrowlType } from '../../foehn-growl/growl-types';
5
5
  import { UploaderHelper } from '../uploader.helper';
6
+ import { BoDocumentsWithErrors } from './bo-multi-upload.type';
6
7
  import * as i0 from "@angular/core";
7
8
  import * as i1 from "@angular/common/http";
8
9
  import * as i2 from "../../foehn-growl/growl-broker.service";
@@ -18,46 +19,86 @@ export class BoMultiUploadService {
18
19
  this.uploadProgressService = uploadProgressService;
19
20
  // A way to have a unique Id per file
20
21
  this.globalSequence = 0;
22
+ this.NB_CONCURRENT_FILES_TO_UPLOAD = 3;
23
+ this.mergeDocumentsWithErrors = (arrayOfDocumentsWithErrors) => {
24
+ const mergedResponse = new BoDocumentsWithErrors();
25
+ arrayOfDocumentsWithErrors.forEach(item => {
26
+ if (!!item.documents?.length) {
27
+ if (!mergedResponse.documents) {
28
+ mergedResponse.documents = [];
29
+ }
30
+ mergedResponse.documents = mergedResponse.documents.concat(item.documents);
31
+ }
32
+ if (!!item.errors?.length) {
33
+ if (!mergedResponse.errors) {
34
+ mergedResponse.errors = [];
35
+ }
36
+ mergedResponse.errors = mergedResponse.errors.concat(item.errors);
37
+ }
38
+ });
39
+ return mergedResponse;
40
+ };
21
41
  this.uploaderHelper = new UploaderHelper(dictionaryService, applicationInfoService);
22
42
  }
23
43
  uploadDocuments(url, formKey, label, files, key, isMultiple, language, shouldDisplayFileSavedConfirmation) {
24
- const documents = files.map(file => this.uploaderHelper.mapToDocumentReference(file, label, key, isMultiple, new Date().getTime(), this.globalSequence++));
25
- const formData = this.uploaderHelper.mapToFormData(documents, formKey, language);
44
+ const filesToGroup = [...files];
45
+ const groupsOfFiles = [];
46
+ while (!!filesToGroup.length) {
47
+ groupsOfFiles.push(filesToGroup.splice(0, this.NB_CONCURRENT_FILES_TO_UPLOAD));
48
+ }
49
+ const arrayOfDocumentsWithErrorsObservable = groupsOfFiles.map((groupOfFile, index) => {
50
+ const documents = groupOfFile.map(file => this.uploaderHelper.mapToDocumentReference(file, label, key, isMultiple, new Date().getTime(), this.globalSequence++));
51
+ const formData = this.uploaderHelper.mapToFormData(documents, formKey, language);
52
+ return this.getDocumentsWithErrorsObservable(url, formData, shouldDisplayFileSavedConfirmation, index, groupsOfFiles.length);
53
+ });
54
+ if (arrayOfDocumentsWithErrorsObservable.length === 1) {
55
+ return arrayOfDocumentsWithErrorsObservable[0];
56
+ }
57
+ // concat() operator sequentially emits all values from the given Observable and then proceed to the next one.
58
+ return concat(...arrayOfDocumentsWithErrorsObservable).pipe(
59
+ // toArray() operator collects all the values emitted by the source observable into an array and sends it to the observer
60
+ toArray(), map(this.mergeDocumentsWithErrors.bind(this)));
61
+ }
62
+ deleteDocument(baseUrl, document) {
63
+ const url = `${baseUrl}/${document.reference}`;
64
+ return this.httpClient.delete(url).pipe(
65
+ // Reflect the document once the update is done to ease chaining observables.
66
+ map(() => document), tap(() => {
67
+ const message = this.dictionaryService.getKeySync('foehn-uploader.files-deleted-success-message', { filename: document.filename });
68
+ this.growlService.addWithType(GrowlType.SUCCESS, message);
69
+ }), catchError((e) => {
70
+ const message = this.dictionaryService.getKeySync('foehn-uploader.delete-error-message');
71
+ this.growlService.addWithType(GrowlType.DANGER, message);
72
+ return throwError(() => e);
73
+ }));
74
+ }
75
+ getDownloadUrl(baseUrl, document) {
76
+ return `${baseUrl}/${document.reference}`;
77
+ }
78
+ getDocumentsWithErrorsObservable(url, formData, shouldDisplayFileSavedConfirmation, packageIndex = 0, nbPackages = 1) {
26
79
  return this.httpClient
27
80
  .post(url, formData, {
28
81
  reportProgress: true,
29
82
  observe: 'events',
30
83
  responseType: 'json'
31
84
  })
32
- .pipe(filter((e) => this.uploadProgressService.manageUploadEventFilter(e)), map((e) => e.body), tap(result => {
85
+ .pipe(filter((e) => this.uploadProgressService.manageUploadEventFilter(e, packageIndex, nbPackages)), map((e) => e.body), tap(result => {
33
86
  const successfulDocumentsCount = result.documents && result.documents.length;
34
87
  if (successfulDocumentsCount &&
35
88
  shouldDisplayFileSavedConfirmation) {
36
- const message = `${successfulDocumentsCount} fichier(s) sauvegardé(s) avec succès`;
89
+ const message = this.dictionaryService.getKeySync('foehn-uploader.files-saved-success-message', {
90
+ successfulDocumentsCount: successfulDocumentsCount.toString()
91
+ });
37
92
  this.growlService.addWithType(GrowlType.SUCCESS, message);
38
93
  }
39
94
  }), catchError((e) => {
40
- const message = 'Une erreur est survenue lors de la transmission de vos documents';
95
+ this.uploadProgressService.analysisProgress.next(false);
96
+ this.uploadProgressService.showProgress.next(false);
97
+ const message = this.dictionaryService.getKeySync('foehn-uploader.transmission-error-message');
41
98
  this.growlService.addWithType(GrowlType.DANGER, message);
42
99
  return throwError(() => e);
43
100
  }));
44
101
  }
45
- deleteDocument(baseUrl, document) {
46
- const url = `${baseUrl}/${document.reference}`;
47
- return this.httpClient.delete(url).pipe(
48
- // Reflect the document once the update is done to ease chaining observables.
49
- map(() => document), tap(() => {
50
- const message = `Suppression du fichier ${document.filename} réussie`;
51
- this.growlService.addWithType(GrowlType.SUCCESS, message);
52
- }), catchError((e) => {
53
- const message = 'Une erreur est survenue lors de la suppression de votre document';
54
- this.growlService.addWithType(GrowlType.DANGER, message);
55
- return throwError(() => e);
56
- }));
57
- }
58
- getDownloadUrl(baseUrl, document) {
59
- return `${baseUrl}/${document.reference}`;
60
- }
61
102
  }
62
103
  BoMultiUploadService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: BoMultiUploadService, deps: [{ token: i1.HttpClient }, { token: i2.GrowlBrokerService }, { token: i3.ApplicationInfoService }, { token: i4.SdkDictionaryService }, { token: i5.UploadProgressService }], target: i0.ɵɵFactoryTarget.Injectable });
63
104
  BoMultiUploadService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: BoMultiUploadService, providedIn: 'root' });
@@ -67,4 +108,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
67
108
  providedIn: 'root'
68
109
  }]
69
110
  }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.GrowlBrokerService }, { type: i3.ApplicationInfoService }, { type: i4.SdkDictionaryService }, { type: i5.UploadProgressService }]; } });
70
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bo-multi-upload.service.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-bo-multi-upload/bo-multi-upload.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpEvent, HttpResponse } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Observable, throwError } from 'rxjs';\nimport { catchError, filter, map, tap } from 'rxjs/operators';\n\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { GrowlType } from '../../foehn-growl/growl-types';\nimport { ApplicationInfoService } from '../../sdk-appinfo/application-info.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { DocumentReference } from '../document-reference';\nimport { UploadProgressService } from '../foehn-upload-progress-bar/upload-progress.service';\nimport { UploaderHelper } from '../uploader.helper';\nimport { BoDocumentsWithErrors } from './bo-multi-upload.type';\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class BoMultiUploadService {\n    // A way to have a unique Id per file\n    private globalSequence = 0;\n    private uploaderHelper: UploaderHelper;\n\n    constructor(\n        private httpClient: HttpClient,\n        private growlService: GrowlBrokerService,\n        private applicationInfoService: ApplicationInfoService,\n        private dictionaryService: SdkDictionaryService,\n        private uploadProgressService: UploadProgressService\n    ) {\n        this.uploaderHelper = new UploaderHelper(\n            dictionaryService,\n            applicationInfoService\n        );\n    }\n\n    uploadDocuments(\n        url: string,\n        formKey: string,\n        label: string,\n        files: File[],\n        key: string,\n        isMultiple: boolean,\n        language: string,\n        shouldDisplayFileSavedConfirmation: boolean\n    ): Observable<BoDocumentsWithErrors> {\n        const documents = files.map(file =>\n            this.uploaderHelper.mapToDocumentReference(\n                file,\n                label,\n                key,\n                isMultiple,\n                new Date().getTime(),\n                this.globalSequence++\n            )\n        );\n        const formData = this.uploaderHelper.mapToFormData(\n            documents,\n            formKey,\n            language\n        );\n        return this.httpClient\n            .post<BoDocumentsWithErrors>(url, formData, {\n                reportProgress: true,\n                observe: 'events',\n                responseType: 'json'\n            })\n            .pipe(\n                filter((e: HttpEvent<BoDocumentsWithErrors>) =>\n                    this.uploadProgressService.manageUploadEventFilter(e)\n                ),\n                map(\n                    (e: HttpEvent<BoDocumentsWithErrors>) =>\n                        (e as HttpResponse<BoDocumentsWithErrors>).body\n                ),\n                tap(result => {\n                    const successfulDocumentsCount =\n                        result.documents && result.documents.length;\n                    if (\n                        successfulDocumentsCount &&\n                        shouldDisplayFileSavedConfirmation\n                    ) {\n                        const message = `${successfulDocumentsCount} fichier(s) sauvegardé(s) avec succès`;\n                        this.growlService.addWithType(\n                            GrowlType.SUCCESS,\n                            message\n                        );\n                    }\n                }),\n                catchError((e: unknown) => {\n                    const message =\n                        'Une erreur est survenue lors de la transmission de vos documents';\n                    this.growlService.addWithType(GrowlType.DANGER, message);\n                    return throwError(() => e);\n                })\n            );\n    }\n\n    deleteDocument(\n        baseUrl: string,\n        document: DocumentReference\n    ): Observable<DocumentReference> {\n        const url = `${baseUrl}/${document.reference}`;\n\n        return this.httpClient.delete(url).pipe(\n            // Reflect the document once the update is done to ease chaining observables.\n            map(() => document),\n            tap(() => {\n                const message = `Suppression du fichier ${document.filename} réussie`;\n                this.growlService.addWithType(GrowlType.SUCCESS, message);\n            }),\n            catchError((e: unknown) => {\n                const message =\n                    'Une erreur est survenue lors de la suppression de votre document';\n                this.growlService.addWithType(GrowlType.DANGER, message);\n                return throwError(() => e);\n            })\n        );\n    }\n\n    getDownloadUrl(baseUrl: string, document: DocumentReference): string {\n        return `${baseUrl}/${document.reference}`;\n    }\n}\n"]}
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bo-multi-upload.service.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-bo-multi-upload/bo-multi-upload.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpEvent, HttpResponse } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { concat, Observable, throwError, toArray } from 'rxjs';\nimport { catchError, filter, map, tap } from 'rxjs/operators';\n\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { GrowlType } from '../../foehn-growl/growl-types';\nimport { ApplicationInfoService } from '../../sdk-appinfo/application-info.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { DocumentReference } from '../document-reference';\nimport { UploadProgressService } from '../foehn-upload-progress-bar/upload-progress.service';\nimport { UploaderHelper } from '../uploader.helper';\nimport { BoDocumentsWithErrors } from './bo-multi-upload.type';\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class BoMultiUploadService {\n    // A way to have a unique Id per file\n    private globalSequence = 0;\n    private uploaderHelper: UploaderHelper;\n    private readonly NB_CONCURRENT_FILES_TO_UPLOAD = 3;\n\n    constructor(\n        private httpClient: HttpClient,\n        private growlService: GrowlBrokerService,\n        private applicationInfoService: ApplicationInfoService,\n        private dictionaryService: SdkDictionaryService,\n        private uploadProgressService: UploadProgressService\n    ) {\n        this.uploaderHelper = new UploaderHelper(\n            dictionaryService,\n            applicationInfoService\n        );\n    }\n\n    uploadDocuments(\n        url: string,\n        formKey: string,\n        label: string,\n        files: File[],\n        key: string,\n        isMultiple: boolean,\n        language: string,\n        shouldDisplayFileSavedConfirmation: boolean\n    ): Observable<BoDocumentsWithErrors> {\n        const filesToGroup = [...files];\n        const groupsOfFiles: File[][] = [];\n\n        while (!!filesToGroup.length) {\n            groupsOfFiles.push(\n                filesToGroup.splice(0, this.NB_CONCURRENT_FILES_TO_UPLOAD)\n            );\n        }\n\n        const arrayOfDocumentsWithErrorsObservable: Observable<BoDocumentsWithErrors>[] = groupsOfFiles.map(\n            (groupOfFile, index) => {\n                const documents = groupOfFile.map(file =>\n                    this.uploaderHelper.mapToDocumentReference(\n                        file,\n                        label,\n                        key,\n                        isMultiple,\n                        new Date().getTime(),\n                        this.globalSequence++\n                    )\n                );\n                const formData = this.uploaderHelper.mapToFormData(\n                    documents,\n                    formKey,\n                    language\n                );\n                return this.getDocumentsWithErrorsObservable(\n                    url,\n                    formData,\n                    shouldDisplayFileSavedConfirmation,\n                    index,\n                    groupsOfFiles.length\n                );\n            }\n        );\n\n        if (arrayOfDocumentsWithErrorsObservable.length === 1) {\n            return arrayOfDocumentsWithErrorsObservable[0];\n        }\n\n        //  concat() operator sequentially emits all values from the given Observable and then proceed to the next one.\n        return concat(...arrayOfDocumentsWithErrorsObservable).pipe(\n            //  toArray() operator collects all the values emitted by the source observable into an array and sends it to the observer\n            toArray(),\n            map(this.mergeDocumentsWithErrors.bind(this))\n        );\n    }\n\n    deleteDocument(\n        baseUrl: string,\n        document: DocumentReference\n    ): Observable<DocumentReference> {\n        const url = `${baseUrl}/${document.reference}`;\n\n        return this.httpClient.delete(url).pipe(\n            // Reflect the document once the update is done to ease chaining observables.\n            map(() => document),\n            tap(() => {\n                const message = this.dictionaryService.getKeySync(\n                    'foehn-uploader.files-deleted-success-message',\n                    { filename: document.filename }\n                );\n                this.growlService.addWithType(GrowlType.SUCCESS, message);\n            }),\n            catchError((e: unknown) => {\n                const message = this.dictionaryService.getKeySync(\n                    'foehn-uploader.delete-error-message'\n                );\n                this.growlService.addWithType(GrowlType.DANGER, message);\n                return throwError(() => e);\n            })\n        );\n    }\n\n    getDownloadUrl(baseUrl: string, document: DocumentReference): string {\n        return `${baseUrl}/${document.reference}`;\n    }\n\n    private getDocumentsWithErrorsObservable(\n        url: string,\n        formData: FormData,\n        shouldDisplayFileSavedConfirmation: boolean,\n        packageIndex: number = 0,\n        nbPackages: number = 1\n    ): Observable<BoDocumentsWithErrors> {\n        return this.httpClient\n            .post<BoDocumentsWithErrors>(url, formData, {\n                reportProgress: true,\n                observe: 'events',\n                responseType: 'json'\n            })\n            .pipe(\n                filter((e: HttpEvent<BoDocumentsWithErrors>) =>\n                    this.uploadProgressService.manageUploadEventFilter(\n                        e,\n                        packageIndex,\n                        nbPackages\n                    )\n                ),\n                map(\n                    (e: HttpEvent<BoDocumentsWithErrors>) =>\n                        (e as HttpResponse<BoDocumentsWithErrors>).body\n                ),\n                tap(result => {\n                    const successfulDocumentsCount =\n                        result.documents && result.documents.length;\n                    if (\n                        successfulDocumentsCount &&\n                        shouldDisplayFileSavedConfirmation\n                    ) {\n                        const message = this.dictionaryService.getKeySync(\n                            'foehn-uploader.files-saved-success-message',\n                            {\n                                successfulDocumentsCount: successfulDocumentsCount.toString()\n                            }\n                        );\n                        this.growlService.addWithType(\n                            GrowlType.SUCCESS,\n                            message\n                        );\n                    }\n                }),\n                catchError((e: unknown) => {\n                    this.uploadProgressService.analysisProgress.next(false);\n                    this.uploadProgressService.showProgress.next(false);\n\n                    const message = this.dictionaryService.getKeySync(\n                        'foehn-uploader.transmission-error-message'\n                    );\n                    this.growlService.addWithType(GrowlType.DANGER, message);\n                    return throwError(() => e);\n                })\n            );\n    }\n\n    private mergeDocumentsWithErrors = (\n        arrayOfDocumentsWithErrors: BoDocumentsWithErrors[]\n    ): BoDocumentsWithErrors => {\n        const mergedResponse: BoDocumentsWithErrors = new BoDocumentsWithErrors();\n        arrayOfDocumentsWithErrors.forEach(item => {\n            if (!!item.documents?.length) {\n                if (!mergedResponse.documents) {\n                    mergedResponse.documents = [];\n                }\n                mergedResponse.documents = mergedResponse.documents.concat(\n                    item.documents\n                );\n            }\n            if (!!item.errors?.length) {\n                if (!mergedResponse.errors) {\n                    mergedResponse.errors = [];\n                }\n                mergedResponse.errors = mergedResponse.errors.concat(\n                    item.errors\n                );\n            }\n        });\n        return mergedResponse;\n    };\n}\n"]}
@@ -112,6 +112,14 @@ export class FoehnBoMultiUploadComponent extends AbstractFoehnUploaderComponent
112
112
  this.model = [...this.model, ...(documents || [])];
113
113
  this.triggerUserInput(this.model);
114
114
  }
115
+ removeDocumentFromModel(document) {
116
+ const index = this.model.indexOf(document);
117
+ if (index > -1) {
118
+ this.model.splice(index, 1);
119
+ this.model = [...this.model];
120
+ this.triggerUserInput(this.model);
121
+ }
122
+ }
115
123
  displayErrorsFromServer(errors) {
116
124
  if (!errors || !errors.length) {
117
125
  // PRESTAKIT-309: Simulate user interaction with component
@@ -123,14 +131,6 @@ export class FoehnBoMultiUploadComponent extends AbstractFoehnUploaderComponent
123
131
  errors.forEach(error => this.uploaderHelper.addCustomErrorOrDefault(this.name, newErrors, `${error.errorCode}`, error.label, this.customErrors, this.overrideAcceptedExtensions, this.overrideMaxFileNameLength, this.overrideIllegalCharacters));
124
132
  this.refreshErrors([newErrors, true]);
125
133
  }
126
- removeDocumentFromModel(document) {
127
- const index = this.model.indexOf(document);
128
- if (index > -1) {
129
- this.model.splice(index, 1);
130
- this.model = [...this.model];
131
- this.triggerUserInput(this.model);
132
- }
133
- }
134
134
  manageSingleFile(files) {
135
135
  // Edge browser lets you drop files from dialog box when selecting a file.
136
136
  // We need to make sure we only keep last selected file from user
@@ -175,4 +175,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
175
175
  }], documentDeleted: [{
176
176
  type: Output
177
177
  }] } });
178
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"foehn-bo-multi-upload.component.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.ts","../../../../../projects/prestations-ng/src/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,SAAS,EACT,YAAY,EACZ,UAAU,EACV,KAAK,EAGL,MAAM,EACT,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,KAAK,EAAE,MAAM,MAAM,CAAC;AAC7B,OAAO,EACH,UAAU,EACV,MAAM,EACN,QAAQ,EACR,GAAG,EACH,YAAY,EACf,MAAM,gBAAgB,CAAC;AAIxB,OAAO,EAAE,mBAAmB,EAAE,MAAM,yCAAyC,CAAC;AAI9E,OAAO,EAAE,8BAA8B,EAAE,MAAM,sCAAsC,CAAC;AAEtF,OAAO,EAAE,qBAAqB,EAAE,MAAM,sDAAsD,CAAC;AAC7F,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;;;;;;;;;;;;;AAiBjE,MAAM,OAAO,2BACT,SAAQ,8BAAmD;IAmB3D,YACI,oBAA0C,EAChC,sBAA8C,EAC9C,mBAA6C,EAC7C,iBAAuC,EACvC,YAAgC;QAE1C,KAAK,CACD,sBAAsB,EACtB,mBAAmB,EACnB,iBAAiB,EACjB,YAAY,CACf,CAAC;QAVQ,2BAAsB,GAAtB,sBAAsB,CAAwB;QAC9C,wBAAmB,GAAnB,mBAAmB,CAA0B;QAC7C,sBAAiB,GAAjB,iBAAiB,CAAsB;QACvC,iBAAY,GAAZ,YAAY,CAAoB;QAZ9C,uCAAkC,GAAG,IAAI,CAAC;QAG1C,oBAAe,GAAG,IAAI,YAAY,EAAqB,CAAC;QAiBpD,IAAI,CAAC,oBAAoB,GAAG,oBAAoB,CAAC;IACrD,CAAC;IAED,QAAQ;QACJ,KAAK,CAAC,QAAQ,EAAE,CAAC;QAEjB,kBAAkB;QAClB,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,kBAAkB;aACjD,IAAI,CACD,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;QAC7D,sDAAsD;QACtD,YAAY,CAAC,GAAG,CAAC;QACjB,uCAAuC;QACvC,QAAQ,CACJ,KAAK,CAAC,EAAE,CACJ,IAAI,CAAC,oBAAoB,CAAC,eAAe,CACrC,IAAI,CAAC,SAAS,EACd,IAAI,CAAC,IAAI,EACT,IAAI,CAAC,KAAK,EACV,KAAK,EACL,IAAI,CAAC,GAAG,EACR,IAAI,CAAC,QAAQ,EACb,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,kCAAkC,CAC1C,EACL,CAAC,CACJ,EACD,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,MAAM,MAAM,GAAgB,EAAE,CAAC;YAC/B,8DAA8D;YAC9D,MAAM,EAAE,MAAM,EAAE,GAAQ,KAAK,CAAC;YAC9B,IAAI,CAAC,cAAc,CAAC,uBAAuB,CACvC,IAAI,CAAC,IAAI,EACT,MAAM,EACN,GAAG,MAAM,EAAE,EACX,EAAE,EACF,IAAI,CAAC,YAAY,EACjB,IAAI,CAAC,0BAA0B,EAC/B,IAAI,CAAC,yBAAyB,EAC9B,IAAI,CAAC,yBAAyB,CACjC,CAAC;YACF,IAAI,CAAC,aAAa,CAAC,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC,CAAC;YACnC,OAAO,KAAK,CAAC;QACjB,CAAC,CAAC,CACL;YACD,0DAA0D;aACzD,SAAS,CAAC,CAAC,EAAE,SAAS,EAAE,MAAM,EAAE,EAAE,EAAE;YACjC,IAAI,CAAC,uBAAuB,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;QACpD,CAAC,CAAC,CAAC;QAEP,wBAAwB;QACxB,IAAI,CAAC,0BAA0B,GAAG,IAAI,CAAC,qBAAqB;aACvD,IAAI,CACD,MAAM,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC;QAC9B,uDAAuD;QACvD,YAAY,CAAC,GAAG,CAAC;QACjB,uCAAuC;QACvC,QAAQ,CACJ,QAAQ,CAAC,EAAE,CACP,IAAI,CAAC,oBAAoB,CAAC,cAAc,CACpC,IAAI,CAAC,SAAS,EACd,QAAQ,CACX,EACL,CAAC,CACJ;QACD,kDAAkD;QAClD,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,EACpD,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,uBAAuB,CAAC,QAAQ,CAAC,CAAC,CAC1D;YACD,0DAA0D;aACzD,SAAS,EAAE,CAAC;IACrB,CAAC;IAED,eAAe;QACX,IAAI,IAAI,CAAC,QAAQ,EAAE;YACf,OAAO,KAAK,CAAC;SAChB;QAED,IAAI,IAAI,CAAC,QAAQ,EAAE;YACf,OAAO,IAAI,CAAC;SACf;QAED,OAAO,CAAC,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAChD,CAAC;IAED,YAAY,CAAC,KAAY;QACrB,8FAA8F;QAC9F,IAAI,CAAC,cAAc,EAAE,CAAC;QAEtB,MAAM,MAAM,GAAqB,KAAK,CAAC,MAA0B,CAAC;QAClE,IAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACxC,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,iCAAiC;IAC1D,CAAC;IAED,8DAA8D;IAC9D,MAAM,CAAC,KAAU;QACb,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;QAC1C,IAAI,IAAI,CAAC,WAAW,EAAE;YAClB,MAAM,QAAQ,GAAa,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAClE,IAAI,QAAQ,EAAE;gBACV,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;aACvC;SACJ;IACL,CAAC;IAEO,oBAAoB,CAAC,QAAkB;QAC3C,IAAI,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEjC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE;YACf,OAAO;SACV;QAED,0FAA0F;QAC1F,IAAI,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YACpC,KAAK,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;SACxC;QAED,2CAA2C;QAC3C,MAAM,WAAW,GAAG,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;QACnD,IAAI,WAAW,CAAC,MAAM,EAAE;YACpB,OAAO;SACV;QAED,4BAA4B;QAC5B,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE;YACnC,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;SACnB;QAED,0BAA0B;QAC1B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACxC,CAAC;IAEO,uBAAuB,CAC3B,SAA8B,EAC9B,MAAyB;QAEzB,IAAI,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;QACrC,IAAI,CAAC,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,EAAE,GAAG,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,CAAC;QACnD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC;IAEO,uBAAuB,CAAC,MAAyB;QACrD,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE;YAC3B,0DAA0D;YAC1D,0EAA0E;YAC1E,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,OAAO;SACV;QAED,MAAM,SAAS,GAAgB,EAAE,CAAC;QAClC,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CACnB,IAAI,CAAC,cAAc,CAAC,uBAAuB,CACvC,IAAI,CAAC,IAAI,EACT,SAAS,EACT,GAAG,KAAK,CAAC,SAAS,EAAE,EACpB,KAAK,CAAC,KAAK,EACX,IAAI,CAAC,YAAY,EACjB,IAAI,CAAC,0BAA0B,EAC/B,IAAI,CAAC,yBAAyB,EAC9B,IAAI,CAAC,yBAAyB,CACjC,CACJ,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC,CAAC;IAC1C,CAAC;IAEO,uBAAuB,CAAC,QAA2B;QACvD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAC3C,IAAI,KAAK,GAAG,CAAC,CAAC,EAAE;YACZ,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;YAC5B,IAAI,CAAC,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC;YAC7B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;SACrC;IACL,CAAC;IAEO,gBAAgB,CAAC,KAAa;QAClC,0EAA0E;QAC1E,iEAAiE;QACjE,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE;YACrC,MAAM,iBAAiB,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC;YAC1C,8DAA8D;YAC9D,iBAAiB,CAAC,OAAO,CAAC,CAAC,GAAQ,EAAE,EAAE,CACnC,IAAI,CAAC,wBAAwB,CAAC,GAAG,CAAC,CACrC,CAAC;SACL;QACD,8DAA8D;QAC9D,OAAO,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAC7B,CAAC;;wHA5NQ,2BAA2B;4GAA3B,2BAA2B,mQAVzB;QACP;YACI,OAAO,EAAE,mBAAmB;YAC5B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,2BAA2B,CAAC;YAC1D,KAAK,EAAE,IAAI;SACd;QACD,EAAE,OAAO,EAAE,oBAAoB,EAAE,QAAQ,EAAE,oBAAoB,EAAE;QACjE,EAAE,OAAO,EAAE,qBAAqB,EAAE,QAAQ,EAAE,qBAAqB,EAAE;KACtE,iDC1CL,+nIA+HA;2FDnFa,2BAA2B;kBAdvC,SAAS;+BACI,uBAAuB,aAGtB;wBACP;4BACI,OAAO,EAAE,mBAAmB;4BAC5B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,4BAA4B,CAAC;4BAC1D,KAAK,EAAE,IAAI;yBACd;wBACD,EAAE,OAAO,EAAE,oBAAoB,EAAE,QAAQ,EAAE,oBAAoB,EAAE;wBACjE,EAAE,OAAO,EAAE,qBAAqB,EAAE,QAAQ,EAAE,qBAAqB,EAAE;qBACtE;2PAMD,SAAS;sBADR,KAAK;gBAIN,SAAS;sBADR,KAAK;gBAIN,WAAW;sBADV,KAAK;gBAIN,kCAAkC;sBADjC,KAAK;gBAIN,eAAe;sBADd,MAAM","sourcesContent":["import {\n    Component,\n    EventEmitter,\n    forwardRef,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output\n} from '@angular/core';\nimport { EMPTY } from 'rxjs';\nimport {\n    catchError,\n    filter,\n    mergeMap,\n    tap,\n    throttleTime\n} from 'rxjs/operators';\n\nimport { FoehnConfirmModalService } from '../../foehn-confirm-modal/foehn-confirm-modal.service';\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { FoehnInputComponent } from '../../foehn-input/foehn-input.component';\nimport { FormError } from '../../form-error';\nimport { ApplicationInfoService } from '../../sdk-appinfo/application-info.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { AbstractFoehnUploaderComponent } from '../abstract-foehn-uploader.component';\nimport { DocumentReference } from '../document-reference';\nimport { UploadProgressService } from '../foehn-upload-progress-bar/upload-progress.service';\nimport { BoMultiUploadService } from './bo-multi-upload.service';\nimport { BoDocumentError } from './bo-multi-upload.type';\n\n@Component({\n    selector: 'foehn-bo-multi-upload',\n    templateUrl: './foehn-bo-multi-upload.component.html',\n    styleUrls: ['../foehn-upload.component.css'],\n    providers: [\n        {\n            provide: FoehnInputComponent,\n            useExisting: forwardRef(() => FoehnBoMultiUploadComponent),\n            multi: true\n        },\n        { provide: BoMultiUploadService, useClass: BoMultiUploadService },\n        { provide: UploadProgressService, useClass: UploadProgressService }\n    ]\n})\nexport class FoehnBoMultiUploadComponent\n    extends AbstractFoehnUploaderComponent<DocumentReference[]>\n    implements OnInit, OnDestroy {\n    @Input()\n    uploadUrl: string;\n\n    @Input()\n    deleteUrl: string;\n\n    @Input()\n    downloadUrl: string;\n\n    @Input()\n    shouldDisplayFileSavedConfirmation = true;\n\n    @Output()\n    documentDeleted = new EventEmitter<DocumentReference>();\n\n    boMultiUploadService: BoMultiUploadService;\n\n    constructor(\n        boMultiUploadService: BoMultiUploadService,\n        protected applicationInfoService: ApplicationInfoService,\n        protected confirmModalService: FoehnConfirmModalService,\n        protected dictionaryService: SdkDictionaryService,\n        protected growlService: GrowlBrokerService\n    ) {\n        super(\n            applicationInfoService,\n            confirmModalService,\n            dictionaryService,\n            growlService\n        );\n        this.boMultiUploadService = boMultiUploadService;\n    }\n\n    ngOnInit(): void {\n        super.ngOnInit();\n\n        // Add files logic\n        this.uploadFilesSubscription = this.uploadFilesSubject\n            .pipe(\n                filter(files => !!files && this.isMaxSelectionReached(files)),\n                // Only one call every 300ms to stop concurrent access\n                throttleTime(300),\n                // Only one concurrent call at the time\n                mergeMap(\n                    files =>\n                        this.boMultiUploadService.uploadDocuments(\n                            this.uploadUrl,\n                            this.name,\n                            this.label,\n                            files,\n                            this.key,\n                            this.multiple,\n                            this.currentLanguage,\n                            this.shouldDisplayFileSavedConfirmation\n                        ),\n                    1\n                ),\n                catchError((error: unknown) => {\n                    const errors: FormError[] = [];\n                    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                    const { status }: any = error;\n                    this.uploaderHelper.addCustomErrorOrDefault(\n                        this.name,\n                        errors,\n                        `${status}`,\n                        '',\n                        this.customErrors,\n                        this.overrideAcceptedExtensions,\n                        this.overrideMaxFileNameLength,\n                        this.overrideIllegalCharacters\n                    );\n                    this.refreshErrors([errors, true]);\n                    return EMPTY;\n                })\n            )\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(({ documents, errors }) => {\n                this.displayResultFromServer(documents, errors);\n            });\n\n        // Delete document logic\n        this.deleteDocumentSubscription = this.deleteDocumentSubject\n            .pipe(\n                filter(document => !!document),\n                // Only one call every 300 ms to stop concurrent access\n                throttleTime(300),\n                // Only one concurrent call at the time\n                mergeMap(\n                    document =>\n                        this.boMultiUploadService.deleteDocument(\n                            this.deleteUrl,\n                            document\n                        ),\n                    1\n                ),\n                // Side effect once the document has been deleted.\n                tap(document => this.documentDeleted.next(document)),\n                tap(document => this.removeDocumentFromModel(document))\n            )\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe();\n    }\n\n    canAddMoreFiles(): boolean {\n        if (this.readonly) {\n            return false;\n        }\n\n        if (this.multiple) {\n            return true;\n        }\n\n        return !(this.model && !!this.model.length);\n    }\n\n    onFileChange(event: Event): void {\n        // PRESTAKIT-309: Mark as pristine so errors can be shown directly to user when uploading file\n        this.markAsPristine();\n\n        const target: HTMLInputElement = event.target as HTMLInputElement;\n        this.populateFileItemList(target.files);\n        target.value = null; // reset hidden file input values\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    onDrop(event: any): void {\n        this.uploaderHelper.preventAndStop(event);\n        if (this.dragAndDrop) {\n            const fileList: FileList = this.uploaderHelper.getTransfer(event);\n            if (fileList) {\n                this.populateFileItemList(fileList);\n            }\n        }\n    }\n\n    private populateFileItemList(fileList: FileList): void {\n        let files = Array.from(fileList);\n\n        if (!files.length) {\n            return;\n        }\n\n        // When drag/drop and asked for single upload, we ensure we store only one file for upload\n        if (this.dragAndDrop && !this.multiple) {\n            files = this.manageSingleFile(files);\n        }\n\n        // Display local errors and stop processing\n        const filesErrors = this.displayLocalErrors(files);\n        if (filesErrors.length) {\n            return;\n        }\n\n        // Initialize model if empty\n        if (!this.model || !this.model.length) {\n            this.model = [];\n        }\n\n        // we send files to server\n        this.uploadFilesSubject.next(files);\n    }\n\n    private displayResultFromServer(\n        documents: DocumentReference[],\n        errors: BoDocumentError[]\n    ): void {\n        this.displayErrorsFromServer(errors);\n        this.model = [...this.model, ...(documents || [])];\n        this.triggerUserInput(this.model);\n    }\n\n    private displayErrorsFromServer(errors: BoDocumentError[]): void {\n        if (!errors || !errors.length) {\n            // PRESTAKIT-309: Simulate user interaction with component\n            // only when no error so isPristine is set to false for validation purpose\n            this.markAsDirty();\n            return;\n        }\n\n        const newErrors: FormError[] = [];\n        errors.forEach(error =>\n            this.uploaderHelper.addCustomErrorOrDefault(\n                this.name,\n                newErrors,\n                `${error.errorCode}`,\n                error.label,\n                this.customErrors,\n                this.overrideAcceptedExtensions,\n                this.overrideMaxFileNameLength,\n                this.overrideIllegalCharacters\n            )\n        );\n\n        this.refreshErrors([newErrors, true]);\n    }\n\n    private removeDocumentFromModel(document: DocumentReference): void {\n        const index = this.model.indexOf(document);\n        if (index > -1) {\n            this.model.splice(index, 1);\n            this.model = [...this.model];\n            this.triggerUserInput(this.model);\n        }\n    }\n\n    private manageSingleFile(files: File[]): File[] {\n        // Edge browser lets you drop files from dialog box when selecting a file.\n        // We need to make sure we only keep last selected file from user\n        if (this.model && this.model.length > 0) {\n            const currentModelFiles = [...this.model];\n            // eslint-disable-next-line @typescript-eslint/no-explicit-any\n            currentModelFiles.forEach((doc: any) =>\n                this.removeFileOnConfirmation(doc)\n            );\n        }\n        // Then we ensure we take only one file from current selection\n        return files.slice(0, 1);\n    }\n}\n","<div\n    class=\"form-group\"\n    [class.has-danger]=\"hasErrorsToDisplay()\"\n    [class.vd-form-group-danger]=\"hasErrorsToDisplay()\"\n    [class.drop-zone-highlight]=\"showDropZone\"\n    [attr.id]=\"buildId('Container')\"\n    tabindex=\"-1\"\n    (drop)=\"onDrop($event)\"\n    (dragenter)=\"onDragenter($event)\"\n    (dragleave)=\"onDragleave($event)\"\n    (dragover)=\"onDragover($event)\"\n    #fallBackTrigger\n>\n    <label\n        [attr.for]=\"buildChildId()\"\n        *ngIf=\"!!label\"\n        [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n    >\n        <span [innerHTML]=\"label\"></span>\n        <span\n            *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n            aria-hidden=\"true\"\n        >\n            {{ 'foehn-input.optional' | fromDictionary }}\n        </span>\n    </label>\n\n    <foehn-validation-alerts\n        [component]=\"this\"\n        [shouldErrorsBeLive]=\"hasLiveUploadErrors()\"\n    ></foehn-validation-alerts>\n\n    <small\n        *ngIf=\"helpText\"\n        [attr.id]=\"buildChildId() + 'Help'\"\n        class=\"form-text text-secondary\"\n        [innerHTML]=\"helpText\"\n    ></small>\n\n    <!-- PRESTAKIT-309: Fake input with NgModel to be registered into Form controls -->\n    <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n    <ng-content></ng-content>\n\n    <div\n        *ngFor=\"let document of model; trackBy: trackByDocument\"\n        class=\"file file-uploaded\"\n    >\n        <!-- prettier-ignore -->\n        <a [href]=\"boMultiUploadService.getDownloadUrl(downloadUrl, document)\">{{ document.filename }}</a>\n        <span class=\"ml-1 mr-1\">\n            ({{ toMegaOctets(document.fileSize) }}\n            <abbr\n                [title]=\"'foehn-uploader.abbr-megaoctet-title' | fromDictionary\"\n            >\n                {{ 'foehn-uploader.abbr-megaoctet' | fromDictionary }}\n            </abbr>\n            )\n        </span>\n        <button\n            *ngIf=\"!readonly\"\n            type=\"button\"\n            class=\"btn icon-button delete-uploaded\"\n            (click)=\"removeFile(document)\"\n        >\n            <foehn-icon-times\n                [title]=\"\n                    'foehn-uploader.delete-icon-title'\n                        | fromDictionary: { docName: document.filename }\n                \"\n            ></foehn-icon-times>\n        </button>\n    </div>\n\n    <small\n        class=\"form-text text-secondary uploaded-global-info\"\n        *ngIf=\"showGlobalInfos(model)\"\n        [innerHTML]=\"getGlobalInfos(model)\"\n    ></small>\n\n    <ng-container *ngIf=\"canAddMoreFiles()\">\n        <input\n            class=\"form-control-file actual-input\"\n            type=\"file\"\n            [name]=\"name || label\"\n            [multiple]=\"multiple\"\n            [attr.accept]=\"\n                overrideAcceptedExtensions\n                    ? overrideAcceptedExtensions\n                    : uploaderHelper.accept\n                    ? uploaderHelper.accept\n                    : null\n            \"\n            (change)=\"onFileChange($event)\"\n            #inputFile\n        />\n\n        <button\n            type=\"button\"\n            class=\"btn btn-primary\"\n            [ngClass]=\"{ 'sr-only': !showUploadButton }\"\n            [attr.id]=\"buildChildId()\"\n            [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n            [attr.aria-describedby]=\"buildId('ErrorsContainer')\"\n            (click)=\"inputFile.click()\"\n            #entryComponent\n        >\n            {{ multiple ? chooseButtonLabelMultiple : chooseButtonLabel }}\n        </button>\n\n        <div\n            *ngIf=\"dragAndDrop\"\n            aria-hidden=\"true\"\n            class=\"dropfile\"\n            [class.dropfile-border-transparent]=\"showDropZone\"\n            (click)=\"inputFile.click()\"\n        >\n            {{ multiple ? dropZoneLabelMultiple : dropZoneLabel }}\n        </div>\n    </ng-container>\n\n    <foehn-upload-progress-bar\n        [progressBarTriggerHtmlElement]=\"\n            inputElement?.nativeElement || fallBackTrigger\n        \"\n    ></foehn-upload-progress-bar>\n</div>\n"]}
178
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"foehn-bo-multi-upload.component.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.ts","../../../../../projects/prestations-ng/src/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    EventEmitter,\n    forwardRef,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output\n} from '@angular/core';\nimport { EMPTY } from 'rxjs';\nimport {\n    catchError,\n    filter,\n    mergeMap,\n    tap,\n    throttleTime\n} from 'rxjs/operators';\n\nimport { FoehnConfirmModalService } from '../../foehn-confirm-modal/foehn-confirm-modal.service';\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { FoehnInputComponent } from '../../foehn-input/foehn-input.component';\nimport { FormError } from '../../form-error';\nimport { ApplicationInfoService } from '../../sdk-appinfo/application-info.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { AbstractFoehnUploaderComponent } from '../abstract-foehn-uploader.component';\nimport { DocumentReference } from '../document-reference';\nimport { UploadProgressService } from '../foehn-upload-progress-bar/upload-progress.service';\nimport { BoMultiUploadService } from './bo-multi-upload.service';\nimport { BoDocumentError } from './bo-multi-upload.type';\n\n@Component({\n    selector: 'foehn-bo-multi-upload',\n    templateUrl: './foehn-bo-multi-upload.component.html',\n    styleUrls: ['../foehn-upload.component.css'],\n    providers: [\n        {\n            provide: FoehnInputComponent,\n            useExisting: forwardRef(() => FoehnBoMultiUploadComponent),\n            multi: true\n        },\n        { provide: BoMultiUploadService, useClass: BoMultiUploadService },\n        { provide: UploadProgressService, useClass: UploadProgressService }\n    ]\n})\nexport class FoehnBoMultiUploadComponent\n    extends AbstractFoehnUploaderComponent<DocumentReference[]>\n    implements OnInit, OnDestroy {\n    @Input()\n    uploadUrl: string;\n\n    @Input()\n    deleteUrl: string;\n\n    @Input()\n    downloadUrl: string;\n\n    @Input()\n    shouldDisplayFileSavedConfirmation = true;\n\n    @Output()\n    documentDeleted = new EventEmitter<DocumentReference>();\n\n    boMultiUploadService: BoMultiUploadService;\n\n    constructor(\n        boMultiUploadService: BoMultiUploadService,\n        protected applicationInfoService: ApplicationInfoService,\n        protected confirmModalService: FoehnConfirmModalService,\n        protected dictionaryService: SdkDictionaryService,\n        protected growlService: GrowlBrokerService\n    ) {\n        super(\n            applicationInfoService,\n            confirmModalService,\n            dictionaryService,\n            growlService\n        );\n        this.boMultiUploadService = boMultiUploadService;\n    }\n\n    ngOnInit(): void {\n        super.ngOnInit();\n\n        // Add files logic\n        this.uploadFilesSubscription = this.uploadFilesSubject\n            .pipe(\n                filter(files => !!files && this.isMaxSelectionReached(files)),\n                // Only one call every 300ms to stop concurrent access\n                throttleTime(300),\n                // Only one concurrent call at the time\n                mergeMap(\n                    files =>\n                        this.boMultiUploadService.uploadDocuments(\n                            this.uploadUrl,\n                            this.name,\n                            this.label,\n                            files,\n                            this.key,\n                            this.multiple,\n                            this.currentLanguage,\n                            this.shouldDisplayFileSavedConfirmation\n                        ),\n                    1\n                ),\n                catchError((error: unknown) => {\n                    const errors: FormError[] = [];\n                    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                    const { status }: any = error;\n                    this.uploaderHelper.addCustomErrorOrDefault(\n                        this.name,\n                        errors,\n                        `${status}`,\n                        '',\n                        this.customErrors,\n                        this.overrideAcceptedExtensions,\n                        this.overrideMaxFileNameLength,\n                        this.overrideIllegalCharacters\n                    );\n                    this.refreshErrors([errors, true]);\n                    return EMPTY;\n                })\n            )\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe(({ documents, errors }) => {\n                this.displayResultFromServer(documents, errors);\n            });\n\n        // Delete document logic\n        this.deleteDocumentSubscription = this.deleteDocumentSubject\n            .pipe(\n                filter(document => !!document),\n                // Only one call every 300 ms to stop concurrent access\n                throttleTime(300),\n                // Only one concurrent call at the time\n                mergeMap(\n                    document =>\n                        this.boMultiUploadService.deleteDocument(\n                            this.deleteUrl,\n                            document\n                        ),\n                    1\n                ),\n                // Side effect once the document has been deleted.\n                tap(document => this.documentDeleted.next(document)),\n                tap(document => this.removeDocumentFromModel(document))\n            )\n            // eslint-disable-next-line rxjs-angular/prefer-async-pipe\n            .subscribe();\n    }\n\n    canAddMoreFiles(): boolean {\n        if (this.readonly) {\n            return false;\n        }\n\n        if (this.multiple) {\n            return true;\n        }\n\n        return !(this.model && !!this.model.length);\n    }\n\n    onFileChange(event: Event): void {\n        // PRESTAKIT-309: Mark as pristine so errors can be shown directly to user when uploading file\n        this.markAsPristine();\n\n        const target: HTMLInputElement = event.target as HTMLInputElement;\n        this.populateFileItemList(target.files);\n        target.value = null; // reset hidden file input values\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    onDrop(event: any): void {\n        this.uploaderHelper.preventAndStop(event);\n        if (this.dragAndDrop) {\n            const fileList: FileList = this.uploaderHelper.getTransfer(event);\n            if (fileList) {\n                this.populateFileItemList(fileList);\n            }\n        }\n    }\n\n    private populateFileItemList(fileList: FileList): void {\n        let files = Array.from(fileList);\n\n        if (!files.length) {\n            return;\n        }\n\n        // When drag/drop and asked for single upload, we ensure we store only one file for upload\n        if (this.dragAndDrop && !this.multiple) {\n            files = this.manageSingleFile(files);\n        }\n\n        // Display local errors and stop processing\n        const filesErrors = this.displayLocalErrors(files);\n        if (filesErrors.length) {\n            return;\n        }\n\n        // Initialize model if empty\n        if (!this.model || !this.model.length) {\n            this.model = [];\n        }\n\n        // we send files to server\n        this.uploadFilesSubject.next(files);\n    }\n\n    private displayResultFromServer(\n        documents: DocumentReference[],\n        errors: BoDocumentError[]\n    ): void {\n        this.displayErrorsFromServer(errors);\n        this.model = [...this.model, ...(documents || [])];\n        this.triggerUserInput(this.model);\n    }\n\n    private removeDocumentFromModel(document: DocumentReference): void {\n        const index = this.model.indexOf(document);\n        if (index > -1) {\n            this.model.splice(index, 1);\n            this.model = [...this.model];\n            this.triggerUserInput(this.model);\n        }\n    }\n\n    private displayErrorsFromServer(errors: BoDocumentError[]): void {\n        if (!errors || !errors.length) {\n            // PRESTAKIT-309: Simulate user interaction with component\n            // only when no error so isPristine is set to false for validation purpose\n            this.markAsDirty();\n            return;\n        }\n\n        const newErrors: FormError[] = [];\n        errors.forEach(error =>\n            this.uploaderHelper.addCustomErrorOrDefault(\n                this.name,\n                newErrors,\n                `${error.errorCode}`,\n                error.label,\n                this.customErrors,\n                this.overrideAcceptedExtensions,\n                this.overrideMaxFileNameLength,\n                this.overrideIllegalCharacters\n            )\n        );\n\n        this.refreshErrors([newErrors, true]);\n    }\n\n    private manageSingleFile(files: File[]): File[] {\n        // Edge browser lets you drop files from dialog box when selecting a file.\n        // We need to make sure we only keep last selected file from user\n        if (this.model && this.model.length > 0) {\n            const currentModelFiles = [...this.model];\n            // eslint-disable-next-line @typescript-eslint/no-explicit-any\n            currentModelFiles.forEach((doc: any) =>\n                this.removeFileOnConfirmation(doc)\n            );\n        }\n        // Then we ensure we take only one file from current selection\n        return files.slice(0, 1);\n    }\n}\n","<div\n    class=\"form-group\"\n    [class.has-danger]=\"hasErrorsToDisplay()\"\n    [class.vd-form-group-danger]=\"hasErrorsToDisplay()\"\n    [class.drop-zone-highlight]=\"showDropZone\"\n    [attr.id]=\"buildId('Container')\"\n    tabindex=\"-1\"\n    (drop)=\"onDrop($event)\"\n    (dragenter)=\"onDragenter($event)\"\n    (dragleave)=\"onDragleave($event)\"\n    (dragover)=\"onDragover($event)\"\n    #fallBackTrigger\n>\n    <label\n        [attr.for]=\"buildChildId()\"\n        *ngIf=\"!!label\"\n        [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n    >\n        <span [innerHTML]=\"label\"></span>\n        <span\n            *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n            aria-hidden=\"true\"\n        >\n            {{ 'foehn-input.optional' | fromDictionary }}\n        </span>\n    </label>\n\n    <foehn-validation-alerts\n        [component]=\"this\"\n        [shouldErrorsBeLive]=\"hasLiveUploadErrors()\"\n    ></foehn-validation-alerts>\n\n    <small\n        *ngIf=\"helpText\"\n        [attr.id]=\"buildChildId() + 'Help'\"\n        class=\"form-text text-secondary\"\n        [innerHTML]=\"helpText\"\n    ></small>\n\n    <!-- PRESTAKIT-309: Fake input with NgModel to be registered into Form controls -->\n    <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n    <ng-content></ng-content>\n\n    <div\n        *ngFor=\"let document of model; trackBy: trackByDocument\"\n        class=\"file file-uploaded\"\n    >\n        <!-- prettier-ignore -->\n        <a [href]=\"boMultiUploadService.getDownloadUrl(downloadUrl, document)\">{{ document.filename }}</a>\n        <span class=\"ml-1 mr-1\">\n            ({{ toMegaOctets(document.fileSize) }}\n            <abbr\n                [title]=\"'foehn-uploader.abbr-megaoctet-title' | fromDictionary\"\n            >\n                {{ 'foehn-uploader.abbr-megaoctet' | fromDictionary }}\n            </abbr>\n            )\n        </span>\n        <button\n            *ngIf=\"!readonly\"\n            type=\"button\"\n            class=\"btn icon-button delete-uploaded\"\n            (click)=\"removeFile(document)\"\n        >\n            <foehn-icon-times\n                [title]=\"\n                    'foehn-uploader.delete-icon-title'\n                        | fromDictionary: { docName: document.filename }\n                \"\n            ></foehn-icon-times>\n        </button>\n    </div>\n\n    <small\n        class=\"form-text text-secondary uploaded-global-info\"\n        *ngIf=\"showGlobalInfos(model)\"\n        [innerHTML]=\"getGlobalInfos(model)\"\n    ></small>\n\n    <ng-container *ngIf=\"canAddMoreFiles()\">\n        <input\n            class=\"form-control-file actual-input\"\n            type=\"file\"\n            [name]=\"name || label\"\n            [multiple]=\"multiple\"\n            [attr.accept]=\"\n                overrideAcceptedExtensions\n                    ? overrideAcceptedExtensions\n                    : uploaderHelper.accept\n                    ? uploaderHelper.accept\n                    : null\n            \"\n            (change)=\"onFileChange($event)\"\n            #inputFile\n        />\n\n        <button\n            type=\"button\"\n            class=\"btn btn-primary\"\n            [ngClass]=\"{ 'sr-only': !showUploadButton }\"\n            [attr.id]=\"buildChildId()\"\n            [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n            [attr.aria-describedby]=\"buildId('ErrorsContainer')\"\n            (click)=\"inputFile.click()\"\n            #entryComponent\n        >\n            {{ multiple ? chooseButtonLabelMultiple : chooseButtonLabel }}\n        </button>\n\n        <div\n            *ngIf=\"dragAndDrop\"\n            aria-hidden=\"true\"\n            class=\"dropfile\"\n            [class.dropfile-border-transparent]=\"showDropZone\"\n            (click)=\"inputFile.click()\"\n        >\n            {{ multiple ? dropZoneLabelMultiple : dropZoneLabel }}\n        </div>\n    </ng-container>\n\n    <foehn-upload-progress-bar\n        [progressBarTriggerHtmlElement]=\"\n            inputElement?.nativeElement || fallBackTrigger\n        \"\n    ></foehn-upload-progress-bar>\n</div>\n"]}
@@ -1,8 +1,9 @@
1
1
  import { Injectable } from '@angular/core';
2
- import { EMPTY, throwError } from 'rxjs';
2
+ import { concat, EMPTY, throwError, toArray } from 'rxjs';
3
3
  import { catchError, filter, map, tap } from 'rxjs/operators';
4
4
  import { GrowlType } from '../../foehn-growl/growl-types';
5
5
  import { UploaderHelper } from '../uploader.helper';
6
+ import { DocumentsWithErrors } from './multi-upload.type';
6
7
  import * as i0 from "@angular/core";
7
8
  import * as i1 from "@angular/common/http";
8
9
  import * as i2 from "../../foehn-growl/growl-broker.service";
@@ -20,34 +21,48 @@ export class MultiUploadService {
20
21
  this.uploadProgressService = uploadProgressService;
21
22
  // A way to have a unique Id per file
22
23
  this.globalSequence = 0;
24
+ this.NB_CONCURRENT_FILES_TO_UPLOAD = 3;
25
+ this.mergeDocumentsWithErrors = (arrayOfDocumentsWithErrors) => {
26
+ const mergedResponse = new DocumentsWithErrors();
27
+ arrayOfDocumentsWithErrors.forEach(item => {
28
+ if (!!item.documents?.length) {
29
+ if (!mergedResponse.documents) {
30
+ mergedResponse.documents = [];
31
+ }
32
+ mergedResponse.documents = mergedResponse.documents.concat(item.documents);
33
+ }
34
+ if (!!item.errors?.length) {
35
+ if (!mergedResponse.errors) {
36
+ mergedResponse.errors = [];
37
+ }
38
+ mergedResponse.errors = mergedResponse.errors.concat(item.errors);
39
+ }
40
+ });
41
+ return mergedResponse;
42
+ };
23
43
  this.uploaderHelper = new UploaderHelper(dictionaryService, applicationInfoService);
24
44
  }
25
45
  uploadDocuments(baseUrl, formKey, label, files, key, isMultiple, language, shouldDisplayFileSavedConfirmation) {
26
46
  const reference = this.gesdemHandlerService.lastResponse.meta.reference;
27
47
  if (!!reference?.length) {
28
- const documents = files.map(file => this.uploaderHelper.mapToDocumentReference(file, label, key, isMultiple, new Date().getTime(), this.globalSequence++));
29
- const formData = this.uploaderHelper.mapToFormData(documents, formKey, language);
30
48
  const url = `${baseUrl}/upload/${reference}`;
31
- return this.httpClient
32
- .post(url, formData, {
33
- reportProgress: true,
34
- observe: 'events',
35
- responseType: 'json'
36
- })
37
- .pipe(filter((e) => this.uploadProgressService.manageUploadEventFilter(e)), map((e) => e.body), tap(result => {
38
- const successfulDocumentsCount = result.documents && result.documents.length;
39
- if (successfulDocumentsCount &&
40
- shouldDisplayFileSavedConfirmation) {
41
- const message = `${successfulDocumentsCount} fichier(s) sauvegardé(s) avec succès`;
42
- this.growlService.addWithType(GrowlType.SUCCESS, message);
43
- }
44
- }), catchError((e) => {
45
- this.uploadProgressService.analysisProgress.next(false);
46
- this.uploadProgressService.showProgress.next(false);
47
- const message = 'Une erreur est survenue lors de la transmission de vos documents';
48
- this.growlService.addWithType(GrowlType.DANGER, message);
49
- return throwError(() => e);
50
- }));
49
+ const filesToGroup = [...files];
50
+ const groupsOfFiles = [];
51
+ while (!!filesToGroup.length) {
52
+ groupsOfFiles.push(filesToGroup.splice(0, this.NB_CONCURRENT_FILES_TO_UPLOAD));
53
+ }
54
+ const arrayOfDocumentsWithErrorsObservable = groupsOfFiles.map((groupOfFile, index) => {
55
+ const documents = groupOfFile.map(file => this.uploaderHelper.mapToDocumentReference(file, label, key, isMultiple, new Date().getTime(), this.globalSequence++));
56
+ const formData = this.uploaderHelper.mapToFormData(documents, formKey, language);
57
+ return this.getDocumentsWithErrorsObservable(url, formData, shouldDisplayFileSavedConfirmation, index, groupsOfFiles.length);
58
+ });
59
+ if (arrayOfDocumentsWithErrorsObservable.length === 1) {
60
+ return arrayOfDocumentsWithErrorsObservable[0];
61
+ }
62
+ // concat() operator sequentially emits all values from the given Observable and then proceed to the next one.
63
+ return concat(...arrayOfDocumentsWithErrorsObservable).pipe(
64
+ // toArray() operator collects all the values emitted by the source observable into an array and sends it to the observer
65
+ toArray(), map(this.mergeDocumentsWithErrors.bind(this)));
51
66
  }
52
67
  return EMPTY;
53
68
  }
@@ -57,10 +72,10 @@ export class MultiUploadService {
57
72
  return this.httpClient.delete(url).pipe(
58
73
  // Reflect the document once the update is done to ease chaining observables.
59
74
  map(() => document), tap(() => {
60
- const message = `Suppression du fichier ${document.filename} réussie`;
75
+ const message = this.dictionaryService.getKeySync('foehn-uploader.files-deleted-success-message', { filename: document.filename });
61
76
  this.growlService.addWithType(GrowlType.SUCCESS, message);
62
77
  }), catchError((e) => {
63
- const message = 'Une erreur est survenue lors de la suppression de votre document';
78
+ const message = this.dictionaryService.getKeySync('foehn-uploader.delete-error-message');
64
79
  this.growlService.addWithType(GrowlType.DANGER, message);
65
80
  return throwError(() => e);
66
81
  }));
@@ -69,6 +84,30 @@ export class MultiUploadService {
69
84
  const reference = this.gesdemHandlerService.lastResponse.meta.reference;
70
85
  return `${baseUrl}/download/${reference}/doc/${document.reference}`;
71
86
  }
87
+ getDocumentsWithErrorsObservable(url, formData, shouldDisplayFileSavedConfirmation, packageIndex = 0, nbPackages = 1) {
88
+ return this.httpClient
89
+ .post(url, formData, {
90
+ reportProgress: true,
91
+ observe: 'events',
92
+ responseType: 'json'
93
+ })
94
+ .pipe(filter((e) => this.uploadProgressService.manageUploadEventFilter(e, packageIndex, nbPackages)), map((e) => e.body), tap(result => {
95
+ const successfulDocumentsCount = result.documents && result.documents.length;
96
+ if (successfulDocumentsCount &&
97
+ shouldDisplayFileSavedConfirmation) {
98
+ const message = this.dictionaryService.getKeySync('foehn-uploader.files-saved-success-message', {
99
+ successfulDocumentsCount: successfulDocumentsCount.toString()
100
+ });
101
+ this.growlService.addWithType(GrowlType.SUCCESS, message);
102
+ }
103
+ }), catchError((e) => {
104
+ this.uploadProgressService.analysisProgress.next(false);
105
+ this.uploadProgressService.showProgress.next(false);
106
+ const message = this.dictionaryService.getKeySync('foehn-uploader.transmission-error-message');
107
+ this.growlService.addWithType(GrowlType.DANGER, message);
108
+ return throwError(() => e);
109
+ }));
110
+ }
72
111
  }
73
112
  MultiUploadService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MultiUploadService, deps: [{ token: i1.HttpClient }, { token: i2.GrowlBrokerService }, { token: i3.GesdemHandlerService }, { token: i4.ApplicationInfoService }, { token: i5.SdkDictionaryService }, { token: i6.UploadProgressService }], target: i0.ɵɵFactoryTarget.Injectable });
74
113
  MultiUploadService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MultiUploadService, providedIn: 'root' });
@@ -78,4 +117,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
78
117
  providedIn: 'root'
79
118
  }]
80
119
  }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.GrowlBrokerService }, { type: i3.GesdemHandlerService }, { type: i4.ApplicationInfoService }, { type: i5.SdkDictionaryService }, { type: i6.UploadProgressService }]; } });
81
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-upload.service.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-multi-upload/multi-upload.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpEvent, HttpResponse } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { EMPTY, Observable, throwError } from 'rxjs';\nimport { catchError, filter, map, tap } from 'rxjs/operators';\n\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { GrowlType } from '../../foehn-growl/growl-types';\nimport { GesdemHandlerService } from '../../gesdem/gesdem-handler.service';\nimport { ApplicationInfoService } from '../../sdk-appinfo/application-info.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { DocumentReference } from '../document-reference';\nimport { UploadProgressService } from '../foehn-upload-progress-bar/upload-progress.service';\nimport { UploaderHelper } from '../uploader.helper';\nimport { DocumentsWithErrors } from './multi-upload.type';\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class MultiUploadService {\n    // A way to have a unique Id per file\n    private globalSequence = 0;\n    private uploaderHelper: UploaderHelper;\n\n    constructor(\n        private httpClient: HttpClient,\n        private growlService: GrowlBrokerService,\n        private gesdemHandlerService: GesdemHandlerService,\n        private applicationInfoService: ApplicationInfoService,\n        private dictionaryService: SdkDictionaryService,\n        private uploadProgressService: UploadProgressService\n    ) {\n        this.uploaderHelper = new UploaderHelper(\n            dictionaryService,\n            applicationInfoService\n        );\n    }\n\n    uploadDocuments(\n        baseUrl: string,\n        formKey: string,\n        label: string,\n        files: File[],\n        key: string,\n        isMultiple: boolean,\n        language: string,\n        shouldDisplayFileSavedConfirmation: boolean\n    ): Observable<DocumentsWithErrors> {\n        const reference = this.gesdemHandlerService.lastResponse.meta.reference;\n        if (!!reference?.length) {\n            const documents = files.map(file =>\n                this.uploaderHelper.mapToDocumentReference(\n                    file,\n                    label,\n                    key,\n                    isMultiple,\n                    new Date().getTime(),\n                    this.globalSequence++\n                )\n            );\n            const formData = this.uploaderHelper.mapToFormData(\n                documents,\n                formKey,\n                language\n            );\n            const url = `${baseUrl}/upload/${reference}`;\n            return this.httpClient\n                .post<DocumentsWithErrors>(url, formData, {\n                    reportProgress: true,\n                    observe: 'events',\n                    responseType: 'json'\n                })\n                .pipe(\n                    filter((e: HttpEvent<DocumentsWithErrors>) =>\n                        this.uploadProgressService.manageUploadEventFilter(e)\n                    ),\n                    map(\n                        (e: HttpEvent<DocumentsWithErrors>) =>\n                            (e as HttpResponse<DocumentsWithErrors>).body\n                    ),\n                    tap(result => {\n                        const successfulDocumentsCount =\n                            result.documents && result.documents.length;\n                        if (\n                            successfulDocumentsCount &&\n                            shouldDisplayFileSavedConfirmation\n                        ) {\n                            const message = `${successfulDocumentsCount} fichier(s) sauvegardé(s) avec succès`;\n                            this.growlService.addWithType(\n                                GrowlType.SUCCESS,\n                                message\n                            );\n                        }\n                    }),\n                    catchError((e: unknown) => {\n                        this.uploadProgressService.analysisProgress.next(false);\n                        this.uploadProgressService.showProgress.next(false);\n\n                        const message =\n                            'Une erreur est survenue lors de la transmission de vos documents';\n                        this.growlService.addWithType(\n                            GrowlType.DANGER,\n                            message\n                        );\n                        return throwError(() => e);\n                    })\n                );\n        }\n\n        return EMPTY;\n    }\n\n    deleteDocument(\n        baseUrl: string,\n        document: DocumentReference\n    ): Observable<DocumentReference> {\n        const reference = this.gesdemHandlerService.lastResponse.meta.reference;\n        const url = `${baseUrl}/delete/${reference}/doc/${document.reference}`;\n\n        return this.httpClient.delete(url).pipe(\n            // Reflect the document once the update is done to ease chaining observables.\n            map(() => document),\n            tap(() => {\n                const message = `Suppression du fichier ${document.filename} réussie`;\n                this.growlService.addWithType(GrowlType.SUCCESS, message);\n            }),\n            catchError((e: unknown) => {\n                const message =\n                    'Une erreur est survenue lors de la suppression de votre document';\n                this.growlService.addWithType(GrowlType.DANGER, message);\n                return throwError(() => e);\n            })\n        );\n    }\n\n    getDownloadUrl(baseUrl: string, document: DocumentReference): string {\n        const reference = this.gesdemHandlerService.lastResponse.meta.reference;\n        return `${baseUrl}/download/${reference}/doc/${document.reference}`;\n    }\n}\n"]}
120
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-upload.service.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-multi-upload/multi-upload.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpEvent, HttpResponse } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { concat, EMPTY, Observable, throwError, toArray } from 'rxjs';\nimport { catchError, filter, map, tap } from 'rxjs/operators';\n\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { GrowlType } from '../../foehn-growl/growl-types';\nimport { GesdemHandlerService } from '../../gesdem/gesdem-handler.service';\nimport { ApplicationInfoService } from '../../sdk-appinfo/application-info.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { DocumentReference } from '../document-reference';\nimport { UploadProgressService } from '../foehn-upload-progress-bar/upload-progress.service';\nimport { UploaderHelper } from '../uploader.helper';\nimport { DocumentsWithErrors } from './multi-upload.type';\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class MultiUploadService {\n    // A way to have a unique Id per file\n    private globalSequence = 0;\n    private uploaderHelper: UploaderHelper;\n    private readonly NB_CONCURRENT_FILES_TO_UPLOAD = 3;\n\n    constructor(\n        private httpClient: HttpClient,\n        private growlService: GrowlBrokerService,\n        private gesdemHandlerService: GesdemHandlerService,\n        private applicationInfoService: ApplicationInfoService,\n        private dictionaryService: SdkDictionaryService,\n        private uploadProgressService: UploadProgressService\n    ) {\n        this.uploaderHelper = new UploaderHelper(\n            dictionaryService,\n            applicationInfoService\n        );\n    }\n\n    uploadDocuments(\n        baseUrl: string,\n        formKey: string,\n        label: string,\n        files: File[],\n        key: string,\n        isMultiple: boolean,\n        language: string,\n        shouldDisplayFileSavedConfirmation: boolean\n    ): Observable<DocumentsWithErrors> {\n        const reference = this.gesdemHandlerService.lastResponse.meta.reference;\n        if (!!reference?.length) {\n            const url = `${baseUrl}/upload/${reference}`;\n            const filesToGroup = [...files];\n            const groupsOfFiles: File[][] = [];\n\n            while (!!filesToGroup.length) {\n                groupsOfFiles.push(\n                    filesToGroup.splice(0, this.NB_CONCURRENT_FILES_TO_UPLOAD)\n                );\n            }\n\n            const arrayOfDocumentsWithErrorsObservable: Observable<DocumentsWithErrors>[] = groupsOfFiles.map(\n                (groupOfFile, index) => {\n                    const documents = groupOfFile.map(file =>\n                        this.uploaderHelper.mapToDocumentReference(\n                            file,\n                            label,\n                            key,\n                            isMultiple,\n                            new Date().getTime(),\n                            this.globalSequence++\n                        )\n                    );\n                    const formData = this.uploaderHelper.mapToFormData(\n                        documents,\n                        formKey,\n                        language\n                    );\n                    return this.getDocumentsWithErrorsObservable(\n                        url,\n                        formData,\n                        shouldDisplayFileSavedConfirmation,\n                        index,\n                        groupsOfFiles.length\n                    );\n                }\n            );\n\n            if (arrayOfDocumentsWithErrorsObservable.length === 1) {\n                return arrayOfDocumentsWithErrorsObservable[0];\n            }\n\n            //  concat() operator sequentially emits all values from the given Observable and then proceed to the next one.\n            return concat(...arrayOfDocumentsWithErrorsObservable).pipe(\n                //  toArray() operator collects all the values emitted by the source observable into an array and sends it to the observer\n                toArray(),\n                map(this.mergeDocumentsWithErrors.bind(this))\n            );\n        }\n\n        return EMPTY;\n    }\n\n    deleteDocument(\n        baseUrl: string,\n        document: DocumentReference\n    ): Observable<DocumentReference> {\n        const reference = this.gesdemHandlerService.lastResponse.meta.reference;\n        const url = `${baseUrl}/delete/${reference}/doc/${document.reference}`;\n\n        return this.httpClient.delete(url).pipe(\n            // Reflect the document once the update is done to ease chaining observables.\n            map(() => document),\n            tap(() => {\n                const message = this.dictionaryService.getKeySync(\n                    'foehn-uploader.files-deleted-success-message',\n                    { filename: document.filename }\n                );\n                this.growlService.addWithType(GrowlType.SUCCESS, message);\n            }),\n            catchError((e: unknown) => {\n                const message = this.dictionaryService.getKeySync(\n                    'foehn-uploader.delete-error-message'\n                );\n                this.growlService.addWithType(GrowlType.DANGER, message);\n                return throwError(() => e);\n            })\n        );\n    }\n\n    getDownloadUrl(baseUrl: string, document: DocumentReference): string {\n        const reference = this.gesdemHandlerService.lastResponse.meta.reference;\n        return `${baseUrl}/download/${reference}/doc/${document.reference}`;\n    }\n\n    private getDocumentsWithErrorsObservable(\n        url: string,\n        formData: FormData,\n        shouldDisplayFileSavedConfirmation: boolean,\n        packageIndex: number = 0,\n        nbPackages: number = 1\n    ): Observable<DocumentsWithErrors> {\n        return this.httpClient\n            .post<DocumentsWithErrors>(url, formData, {\n                reportProgress: true,\n                observe: 'events',\n                responseType: 'json'\n            })\n            .pipe(\n                filter((e: HttpEvent<DocumentsWithErrors>) =>\n                    this.uploadProgressService.manageUploadEventFilter(\n                        e,\n                        packageIndex,\n                        nbPackages\n                    )\n                ),\n                map(\n                    (e: HttpEvent<DocumentsWithErrors>) =>\n                        (e as HttpResponse<DocumentsWithErrors>).body\n                ),\n                tap(result => {\n                    const successfulDocumentsCount =\n                        result.documents && result.documents.length;\n                    if (\n                        successfulDocumentsCount &&\n                        shouldDisplayFileSavedConfirmation\n                    ) {\n                        const message = this.dictionaryService.getKeySync(\n                            'foehn-uploader.files-saved-success-message',\n                            {\n                                successfulDocumentsCount: successfulDocumentsCount.toString()\n                            }\n                        );\n                        this.growlService.addWithType(\n                            GrowlType.SUCCESS,\n                            message\n                        );\n                    }\n                }),\n                catchError((e: unknown) => {\n                    this.uploadProgressService.analysisProgress.next(false);\n                    this.uploadProgressService.showProgress.next(false);\n\n                    const message = this.dictionaryService.getKeySync(\n                        'foehn-uploader.transmission-error-message'\n                    );\n                    this.growlService.addWithType(GrowlType.DANGER, message);\n                    return throwError(() => e);\n                })\n            );\n    }\n\n    private mergeDocumentsWithErrors = (\n        arrayOfDocumentsWithErrors: DocumentsWithErrors[]\n    ): DocumentsWithErrors => {\n        const mergedResponse: DocumentsWithErrors = new DocumentsWithErrors();\n        arrayOfDocumentsWithErrors.forEach(item => {\n            if (!!item.documents?.length) {\n                if (!mergedResponse.documents) {\n                    mergedResponse.documents = [];\n                }\n                mergedResponse.documents = mergedResponse.documents.concat(\n                    item.documents\n                );\n            }\n            if (!!item.errors?.length) {\n                if (!mergedResponse.errors) {\n                    mergedResponse.errors = [];\n                }\n                mergedResponse.errors = mergedResponse.errors.concat(\n                    item.errors\n                );\n            }\n        });\n        return mergedResponse;\n    };\n}\n"]}