@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.
- package/CHANGELOG.md +32 -2
- package/dsivd-prestations-ng-v16.1.0-beta.3.tgz +0 -0
- package/esm2020/foehn-upload/foehn-bo-multi-upload/bo-multi-upload.service.mjs +64 -23
- package/esm2020/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.mjs +9 -9
- package/esm2020/foehn-upload/foehn-multi-upload/multi-upload.service.mjs +65 -26
- package/esm2020/foehn-upload/foehn-upload-progress-bar/foehn-upload-progress-bar.component.mjs +14 -7
- package/esm2020/foehn-upload/foehn-upload-progress-bar/upload-progress.service.mjs +8 -4
- package/esm2020/foehn-upload/foehn-upload-progress-bar/upload-progress.type.mjs +5 -1
- package/esm2020/gesdem-action-recovery/gesdem-action-recovery-registration/gesdem-action-recovery-registration.component.mjs +14 -7
- package/esm2020/gesdem-action-recovery/gesdem-action-recovery.module.mjs +12 -4
- package/esm2020/sdk-dictionary/default-dictionary.mjs +20 -1
- package/fesm2015/dsivd-prestations-ng.mjs +202 -76
- package/fesm2015/dsivd-prestations-ng.mjs.map +1 -1
- package/fesm2020/dsivd-prestations-ng.mjs +200 -76
- package/fesm2020/dsivd-prestations-ng.mjs.map +1 -1
- package/foehn-upload/foehn-bo-multi-upload/bo-multi-upload.service.d.ts +3 -0
- package/foehn-upload/foehn-bo-multi-upload/foehn-bo-multi-upload.component.d.ts +1 -1
- package/foehn-upload/foehn-multi-upload/multi-upload.service.d.ts +3 -0
- package/foehn-upload/foehn-upload-progress-bar/foehn-upload-progress-bar.component.d.ts +3 -0
- package/foehn-upload/foehn-upload-progress-bar/upload-progress.service.d.ts +5 -5
- package/foehn-upload/foehn-upload-progress-bar/upload-progress.type.d.ts +2 -0
- package/gesdem-action-recovery/gesdem-action-recovery.module.d.ts +3 -1
- package/package.json +1 -1
- 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
|
-
|
|
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
|
|
|
Binary file
|
|
@@ -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
|
|
25
|
-
const
|
|
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 =
|
|
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
|
-
|
|
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":"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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 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
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
.
|
|
38
|
-
const
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
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 =
|
|
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 = '
|
|
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"]}
|