@energycap/components 0.39.4-ECAP-23220-bc-file-upload-dialog.20231219-1515 → 0.39.4-ECAP-23220-bc-file-upload-dialog.20231220-1435

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.
@@ -47,6 +47,10 @@ export class FileUploadComponent extends FormControlBase {
47
47
  * Default: file
48
48
  */
49
49
  this.displayType = 'file';
50
+ /**
51
+ * Optional property to control whether the user can select multiple files
52
+ */
53
+ this.multiSelect = false;
50
54
  }
51
55
  ngOnChanges(changes) {
52
56
  super.ngOnChanges(changes);
@@ -57,7 +61,7 @@ export class FileUploadComponent extends FormControlBase {
57
61
  // Watch for name to change, if the value is cleared we will clear the other
58
62
  // supporting model properties. The name can be cleared by the user manually or via
59
63
  // the clear button
60
- this.formModel?.get('name')?.valueChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(value => {
64
+ this.formModel?.get('names')?.valueChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(value => {
61
65
  if (!value) {
62
66
  this.formModel.patchValue({
63
67
  files: null,
@@ -68,54 +72,24 @@ export class FileUploadComponent extends FormControlBase {
68
72
  }
69
73
  });
70
74
  }
71
- // * New fileChange method
72
75
  async fileChange(files) {
73
- const promises = Array.from(files).map((file) => new Promise((resolve) => {
76
+ if (this.multiSelect) {
77
+ // Since we append the files to the form model when multiSelect=true, we need to clear the form model at the start
78
+ this.clearFormModel();
79
+ await this.handleMultipleFiles(files);
80
+ }
81
+ else {
82
+ const file = files.item(0);
74
83
  if (file) {
75
- const reader = new FileReader();
76
- reader.onloadend = async (e) => {
77
- const base64FileString = reader?.result?.toString().split(",")[1];
78
- await this.processFile(file, base64FileString);
79
- resolve(null);
80
- };
81
- if (this.isBase64FileOutput()) {
82
- reader.readAsDataURL(file);
83
- }
84
- else {
85
- resolve(this.processFile(file));
86
- }
87
- ;
84
+ await this.readFile(file);
88
85
  }
89
- }));
90
- await Promise.all(promises);
86
+ }
91
87
  // Clear the file inputs value, this will allow the user to pick the same filenames again
92
88
  // and cause fileChange to re-trigger.
93
89
  if (this.fileInput) {
94
90
  this.fileInput.nativeElement.value = '';
95
91
  }
96
92
  }
97
- // * Old Method
98
- // public async fileChange(files: FileList): Promise<void> {
99
- // let file = files.item(0);
100
- // // If there is a file selected and then opened again and click cancel you get null so don't try and set anything
101
- // if (file) {
102
- // let reader: FileReader = new FileReader();
103
- // reader.onloadend = async e => {
104
- // let base64FileString: string | undefined = reader?.result?.toString().split(",")[1];
105
- // this.processFile(file!, base64FileString);
106
- // };
107
- // if (this.isBase64FileOutput()) {
108
- // reader.readAsDataURL(file);
109
- // } else {
110
- // await this.processFile(file!);
111
- // }
112
- // // Clear the file inputs value, this will allow the user to pick the same filename and cause
113
- // // the fileChange to trigger.
114
- // if (this.fileInput) {
115
- // this.fileInput.nativeElement.value = '';
116
- // }
117
- // }
118
- // }
119
93
  /**
120
94
  * Checks the file type and updates the file type accept property. This is what determines the file
121
95
  * type choices that the user will be limited to in the file browse dialog
@@ -146,7 +120,7 @@ export class FileUploadComponent extends FormControlBase {
146
120
  this.patchFileResult(file, base64FileString, result);
147
121
  }
148
122
  catch (e) {
149
- // Bummer, we're not going to do anything about it though.
123
+ // Bummer, we're not going to do anything about it though.
150
124
  // We are not patching any of the result so any existing information remains
151
125
  }
152
126
  }
@@ -168,18 +142,24 @@ export class FileUploadComponent extends FormControlBase {
168
142
  * @param base64FileString
169
143
  * @param onFileSelectedResult
170
144
  */
171
- // * Old method
172
145
  patchFileResult(file, base64FileString, onFileSelectedResult) {
146
+ console.log('inside patchFileResult');
173
147
  // Get the current value of files from formModel
174
148
  let files = this.formModel?.get('files')?.value || [];
175
149
  let names = this.formModel?.get('names')?.value || [];
176
- // If the file is not null, append it to the files array
177
- if (file) {
178
- files = [...files, file];
150
+ // If multiSelect is true we need to append the file to the existing array
151
+ // otherwise we'll just set the file to the form model
152
+ if (this.multiSelect) {
153
+ if (file) {
154
+ files = [...files, file];
155
+ }
156
+ if (file?.name) {
157
+ names = [...names, file.name];
158
+ }
179
159
  }
180
- // If the file name is not null, append it to the names array
181
- if (file?.name) {
182
- names = [...names, file.name];
160
+ else {
161
+ files = [file];
162
+ names = [file?.name];
183
163
  }
184
164
  this.formModel?.patchValue({
185
165
  files: files,
@@ -192,13 +172,42 @@ export class FileUploadComponent extends FormControlBase {
192
172
  else {
193
173
  this.formModel.patchValue({ uploadResult: null });
194
174
  }
175
+ // console.log('formModel.value', this.formModel?.value);
176
+ console.log('this.formModel?.get("names")?.value', this.formModel?.get('names')?.value);
177
+ }
178
+ /* ---------------------------- Helper Functions ---------------------------- */
179
+ clearFormModel() {
180
+ this.formModel?.patchValue({
181
+ files: null,
182
+ names: null,
183
+ base64FileString: null,
184
+ uploadResult: null
185
+ });
186
+ }
187
+ async handleMultipleFiles(files) {
188
+ const promises = Array.from(files).map((file) => this.readFile(file));
189
+ await Promise.all(promises);
190
+ }
191
+ ;
192
+ async readFile(file) {
193
+ const reader = new FileReader();
194
+ reader.onloadend = async (e) => {
195
+ const base64FileString = reader?.result?.toString().split(",")[1];
196
+ await this.processFile(file, base64FileString);
197
+ };
198
+ if (this.isBase64FileOutput()) {
199
+ reader.readAsDataURL(file);
200
+ }
201
+ else {
202
+ await this.processFile(file);
203
+ }
195
204
  }
196
205
  }
197
206
  FileUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, deps: [{ token: i1.ValidationMessageService }, { token: i2.FormGroupHelper }], target: i0.ɵɵFactoryTarget.Component });
198
- FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected", displayType: "displayType", buttonLabel: "buttonLabel" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: i5.ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: i6.FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: i7.FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }, { kind: "pipe", type: i8.TranslatePipe, name: "translate" }] });
207
+ FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected", displayType: "displayType", buttonLabel: "buttonLabel", multiSelect: "multiSelect" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"displayType === 'file' ? formModel?.get('names').value : undefined\"\r\n [attr.multiple]=\"multiSelect ? 'multiple' : undefined\">\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('names')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('names').disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_btn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: i5.ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: i6.FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: i7.FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }, { kind: "pipe", type: i8.TranslatePipe, name: "translate" }] });
199
208
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, decorators: [{
200
209
  type: Component,
201
- args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
210
+ args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"displayType === 'file' ? formModel?.get('names').value : undefined\"\r\n [attr.multiple]=\"multiSelect ? 'multiple' : undefined\">\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('names')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('names').disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_btn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
202
211
  }], ctorParameters: function () { return [{ type: i1.ValidationMessageService }, { type: i2.FormGroupHelper }]; }, propDecorators: { placeholder: [{
203
212
  type: Input
204
213
  }], fileType: [{
@@ -213,8 +222,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
213
222
  type: Input
214
223
  }], buttonLabel: [{
215
224
  type: Input
225
+ }], multiSelect: [{
226
+ type: Input
216
227
  }], fileInput: [{
217
228
  type: ViewChild,
218
229
  args: ["fileInput", { read: ElementRef, static: true }]
219
230
  }] } });
220
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.ts","../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn } from '@angular/forms';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ValidationMessageService } from '../../core/validation-message.service';\r\nimport { FormGroupHelper } from '../../shared/form-group.helper';\r\nimport { FormControlBase } from '../form-control-base';\r\n\r\nexport type FileType = 'zip' | 'excel' | 'custom';\r\nexport type FileOutput = 'raw' | 'base64';\r\n\r\nexport const FileTypeExtensions = {\r\n  zip: ['.zip'],\r\n  excel: ['.xls', '.xlsx']\r\n};\r\n\r\n@Component({\r\n  selector: \"ec-file-upload\",\r\n  templateUrl: \"./file-upload.component.html\",\r\n  styleUrls: [\"./file-upload.component.scss\"]\r\n})\r\nexport class FileUploadComponent extends FormControlBase implements OnInit, OnChanges {\r\n\r\n  // static class to create the form group from a parent component\r\n  public static getFormModel(\r\n    validators: ValidatorFn[],\r\n    disabled: boolean = false\r\n  ): UntypedFormGroup {\r\n    let formGroup = new UntypedFormGroup({\r\n      files: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      names: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new UntypedFormControl(null),\r\n      uploadResult: new UntypedFormControl(null)\r\n    });\r\n    if (disabled) {\r\n      formGroup.disable();\r\n    }\r\n    return formGroup;\r\n  }\r\n\r\n  /**\r\n   * The value of the textbox input's placeholder\r\n   */\r\n  @Input() public placeholder?: string = \"Choose file...\";\r\n\r\n  /** Common extensions for a file browsing dialog\r\n   *  Note: Edge does not support the accept attribute on file inputs, therefor all file types\r\n   *        will be shown.  Firefox and Chrome both support this feature.\r\n   */\r\n  @Input() public fileType?: FileType | undefined;\r\n\r\n  /**\r\n   * File output, determines which properties are supplied on the formModel\r\n   */\r\n  @Input() public fileOutput?: FileOutput = 'base64';\r\n\r\n  /** If fileType is set to custom set the acceptable file types based on the custom array */\r\n  @Input() public customExtensions?: Array<string> | undefined;\r\n\r\n  /**\r\n   * Optional callback supported if the hosting page needs to process the file before\r\n   * setting the formModel with the file information. If the promise resolves it will continue\r\n   * and set the file name and contents to the formModel, otherwise on failure it'll do nothing.\r\n   */\r\n  @Input() public onFileSelected?: (files: File | File[]) => Promise<any>;\r\n\r\n  /** \r\n   * Optional display type that controls whether the file input textbox is displayed or\r\n   * simply a button the user clicks to launch the OS file storage dialog.\r\n   * Default: file\r\n   */\r\n  @Input() public displayType?: 'file' | 'button' = 'file';\r\n\r\n  /** \r\n   * When display type is set to button this property will control the button label\r\n   */\r\n  @Input() public buttonLabel?: string;\r\n\r\n  @ViewChild(\"fileInput\", { read: ElementRef, static: true }) public fileInput?: ElementRef;\r\n\r\n  /** Property bound to the file input to filter what file types are shown in the dialog */\r\n  public fileTypeAccept: string | undefined;\r\n\r\n  constructor(\r\n    protected validationMessageService: ValidationMessageService,\r\n    protected formGroupHelper: FormGroupHelper,\r\n  ) {\r\n    super(validationMessageService, formGroupHelper);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges) {\r\n    super.ngOnChanges(changes);\r\n\r\n    this.updateFileTypeAccept();\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    super.ngOnInit();\r\n\r\n    // Watch for name to change, if the value is cleared we will clear the other\r\n    // supporting model properties. The name can be cleared by the user manually or via\r\n    // the clear button\r\n    this.formModel?.get('name')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          files: null,\r\n          names: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n  }\r\n\r\n  // * New fileChange method\r\n  public async fileChange(files: FileList): Promise<void> {\r\n    const promises = Array.from(files).map((file) => new Promise ((resolve) => {\r\n      if (file) {\r\n        const reader: FileReader = new FileReader();\r\n\r\n        reader.onloadend = async e => {\r\n          const base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n          await this.processFile(file, base64FileString);\r\n          resolve(null);\r\n        };\r\n\r\n        if (this.isBase64FileOutput()) {\r\n          reader.readAsDataURL(file);\r\n        } else {\r\n          resolve(this.processFile(file));\r\n        };\r\n      }\r\n    }));\r\n\r\n    await Promise.all(promises);\r\n\r\n    // Clear the file inputs value, this will allow the user to pick the same filenames again\r\n    // and cause fileChange to re-trigger.\r\n    if (this.fileInput) {\r\n      this.fileInput.nativeElement.value = '';\r\n    }\r\n  }\r\n\r\n  // * Old Method\r\n  // public async fileChange(files: FileList): Promise<void> {\r\n  //   let file = files.item(0);\r\n\r\n  //   // If there is a file selected and then opened again and click cancel you get null so don't try and set anything\r\n  //   if (file) {\r\n  //     let reader: FileReader = new FileReader();\r\n\r\n  //     reader.onloadend = async e => {\r\n  //       let base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n  //       this.processFile(file!, base64FileString);\r\n  //     };\r\n\r\n  //     if (this.isBase64FileOutput()) {\r\n  //       reader.readAsDataURL(file);\r\n  //     } else {\r\n  //       await this.processFile(file!);\r\n  //     }\r\n\r\n  //     // Clear the file inputs value, this will allow the user to pick the same filename and cause\r\n  //     // the fileChange to trigger.\r\n  //     if (this.fileInput) {\r\n  //       this.fileInput.nativeElement.value = '';\r\n  //     }\r\n  //   }\r\n  // }\r\n\r\n  /**\r\n   * Checks the file type and updates the file type accept property. This is what determines the file\r\n   * type choices that the user will be limited to in the file browse dialog\r\n   */\r\n  private updateFileTypeAccept(): void {\r\n    if (this.fileType) {\r\n      if (this.fileType !== \"custom\") {\r\n        this.fileTypeAccept = FileTypeExtensions[this.fileType].join(\",\");\r\n      } else {\r\n        if (this.customExtensions) {\r\n          this.fileTypeAccept = this.customExtensions.join(\",\");\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Take a file that was selected by the user and process/patch our form model\r\n   * If the host component requires an action to occur with the file prior to the patch it will call\r\n   * and wait for it to return.\r\n   * @param file \r\n   * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64\r\n   */\r\n  private async processFile(file: File, base64FileString?: string | undefined): Promise<void> {\r\n    if (this.onFileSelected) {\r\n      try {\r\n        let result = await this.onFileSelected(file);\r\n        this.patchFileResult(file, base64FileString, result);\r\n      } catch (e) {\r\n        // Bummer, we're not going to do anything about it though. \r\n        // We are not patching any of the result so any existing information remains\r\n      }\r\n    } else {\r\n      this.patchFileResult(file, base64FileString);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Based on the fileOutput return whether this component is expected to deliver a base64 output\r\n   * @returns \r\n   */\r\n  private isBase64FileOutput(): boolean {\r\n    return this.fileOutput === 'base64';\r\n  }\r\n\r\n  /**\r\n   * When the file was selected and processed patch the file information that the hosting form will\r\n   * be looking for. \r\n   * @param file \r\n   * @param base64FileString \r\n   * @param onFileSelectedResult \r\n   */\r\n  // * Old method\r\n  private patchFileResult(file: File | null, base64FileString?: string, onFileSelectedResult?: any): void {\r\n    // Get the current value of files from formModel\r\n    let files = this.formModel?.get('files')?.value || [];\r\n    let names = this.formModel?.get('names')?.value || [];\r\n\r\n    // If the file is not null, append it to the files array\r\n    if (file) {\r\n      files = [...files, file];\r\n    }\r\n\r\n    // If the file name is not null, append it to the names array\r\n    if (file?.name) {\r\n      names = [...names, file.name];\r\n    }\r\n\r\n    this.formModel?.patchValue({\r\n      files: files,\r\n      names: names,\r\n      base64FileString: base64FileString ?? null\r\n    });\r\n\r\n    if (onFileSelectedResult) {\r\n      this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n    } else {\r\n      this.formModel.patchValue({ uploadResult: null });\r\n    }\r\n\r\n  }\r\n\r\n  // * New method\r\n  // private patchFileResult(files: File[] | null, base64FileString?: string[], onFileSelectedResult?: any): void {\r\n  //   const fileArray = files || [];\r\n  //   const nameArray = fileArray.map(file => file.name);\r\n  //   const base64Array = base64FileString || [];\r\n\r\n  //   this.formModel?.patchValue({\r\n  //     files: fileArray,\r\n  //     name: nameArray,\r\n  //     base64FileString: base64Array\r\n  //   });\r\n    \r\n  //   if(onFileSelectedResult) {\r\n  //     this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n  //   } else {\r\n  //     this.formModel.patchValue({ uploadResult: null });\r\n  //   };\r\n  // }\r\n}\r\n","<ec-form-group [label]=\"label\"\r\n               [formGroup]=\"formModel\"\r\n               class=\"mb-0\">\r\n  <div class=\"d-flex control-group\">\r\n    <div class=\"d-flex flex-grow position-relative\">\r\n      <input #fileInput\r\n             id=\"{{inputId}}_input\"\r\n             type=\"file\"\r\n             tabindex=\"-1\"\r\n             [attr.accept]=\"fileTypeAccept\"\r\n             (change)=\"fileChange($event.target.files)\"\r\n             [class.has-value]=\"formModel?.get('name')?.value\"\r\n             multiple>\r\n      <ec-form-control *ngIf=\"displayType === 'file'\"\r\n                       id=\"{{inputId}}_formControl\"\r\n                       class=\"text-truncate\"\r\n                       [required]=\"required\"\r\n                       [pending]=\"pending\">\r\n        <input id=\"{{inputId}}_name\"\r\n               [formControl]=\"formModel?.get('name')\"\r\n               type=\"text\"\r\n               [placeholder]=\"placeholder\"\r\n               [tabindex]=\"-1\">\r\n      </ec-form-control>\r\n    </div>\r\n    <ec-button *ngIf=\"displayType === 'file'\"\r\n               #browseBtn\r\n               id=\"{{inputId}}_browseBtn\"\r\n               (clicked)=\"fileInput.click()\"\r\n               type=\"secondary\"\r\n               [tabindex]=\"tabindex\"\r\n               [disabled]=\"formModel?.get('name')?.disabled\"\r\n               label=\"Browse\"\r\n               [autofocus]=\"autofocus\">\r\n    </ec-button>\r\n  </div>\r\n  <ec-button *ngIf=\"displayType === 'button'\"\r\n             #selectZipFilesBtn\r\n             id=\"{{inputId}}_browseBtn\"\r\n             [pending]=\"pending\"\r\n             type=\"primary\"\r\n             [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n             (clicked)=\"fileInput.click()\"\r\n             style=\"width: 100%;\">\r\n  </ec-button>\r\n</ec-form-group>"]}
231
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.ts","../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn } from '@angular/forms';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ValidationMessageService } from '../../core/validation-message.service';\r\nimport { FormGroupHelper } from '../../shared/form-group.helper';\r\nimport { FormControlBase } from '../form-control-base';\r\n\r\nexport type FileType = 'zip' | 'excel' | 'custom';\r\nexport type FileOutput = 'raw' | 'base64';\r\n\r\nexport const FileTypeExtensions = {\r\n  zip: ['.zip'],\r\n  excel: ['.xls', '.xlsx']\r\n};\r\n\r\n@Component({\r\n  selector: \"ec-file-upload\",\r\n  templateUrl: \"./file-upload.component.html\",\r\n  styleUrls: [\"./file-upload.component.scss\"]\r\n})\r\nexport class FileUploadComponent extends FormControlBase implements OnInit, OnChanges {\r\n\r\n  // static class to create the form group from a parent component\r\n  public static getFormModel(\r\n    validators: ValidatorFn[],\r\n    disabled: boolean = false\r\n  ): UntypedFormGroup {\r\n    let formGroup = new UntypedFormGroup({\r\n      files: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      names: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new UntypedFormControl(null),\r\n      uploadResult: new UntypedFormControl(null)\r\n    });\r\n    if (disabled) {\r\n      formGroup.disable();\r\n    }\r\n    return formGroup;\r\n  }\r\n\r\n  /**\r\n   * The value of the textbox input's placeholder\r\n   */\r\n  @Input() public placeholder?: string = \"Choose file...\";\r\n\r\n  /** Common extensions for a file browsing dialog\r\n   *  Note: Edge does not support the accept attribute on file inputs, therefor all file types\r\n   *        will be shown.  Firefox and Chrome both support this feature.\r\n   */\r\n  @Input() public fileType?: FileType | undefined;\r\n\r\n  /**\r\n   * File output, determines which properties are supplied on the formModel\r\n   */\r\n  @Input() public fileOutput?: FileOutput = 'base64';\r\n\r\n  /** If fileType is set to custom set the acceptable file types based on the custom array */\r\n  @Input() public customExtensions?: Array<string> | undefined;\r\n\r\n  /**\r\n   * Optional callback supported if the hosting page needs to process the file before\r\n   * setting the formModel with the file information. If the promise resolves it will continue\r\n   * and set the file name and contents to the formModel, otherwise on failure it'll do nothing.\r\n   */\r\n  @Input() public onFileSelected?: (files: File | File[]) => Promise<any>;\r\n\r\n  /** \r\n   * Optional display type that controls whether the file input textbox is displayed or\r\n   * simply a button the user clicks to launch the OS file storage dialog.\r\n   * Default: file\r\n   */\r\n  @Input() public displayType?: 'file' | 'button' = 'file';\r\n\r\n  /** \r\n   * When display type is set to button this property will control the button label\r\n   */\r\n  @Input() public buttonLabel?: string;\r\n\r\n  /**\r\n   * Optional property to control whether the user can select multiple files\r\n   */\r\n  @Input() public multiSelect?: boolean = false;\r\n\r\n  @ViewChild(\"fileInput\", { read: ElementRef, static: true }) public fileInput?: ElementRef;\r\n\r\n  /** Property bound to the file input to filter what file types are shown in the dialog */\r\n  public fileTypeAccept: string | undefined;\r\n\r\n  constructor(\r\n    protected validationMessageService: ValidationMessageService,\r\n    protected formGroupHelper: FormGroupHelper,\r\n  ) {\r\n    super(validationMessageService, formGroupHelper);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges) {\r\n    super.ngOnChanges(changes);\r\n\r\n    this.updateFileTypeAccept();\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    super.ngOnInit();\r\n\r\n    // Watch for name to change, if the value is cleared we will clear the other\r\n    // supporting model properties. The name can be cleared by the user manually or via\r\n    // the clear button\r\n    this.formModel?.get('names')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          files: null,\r\n          names: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n  }\r\n\r\n  public async fileChange(files: FileList): Promise<void> {\r\n\r\n    if (this.multiSelect) {\r\n      // Since we append the files to the form model when multiSelect=true, we need to clear the form model at the start\r\n      this.clearFormModel();\r\n      await this.handleMultipleFiles(files);\r\n    } else {\r\n      const file = files.item(0);\r\n      if (file) {\r\n        await this.readFile(file);\r\n      }\r\n    }\r\n\r\n    // Clear the file inputs value, this will allow the user to pick the same filenames again\r\n    // and cause fileChange to re-trigger.\r\n    if (this.fileInput) {\r\n      this.fileInput.nativeElement.value = '';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Checks the file type and updates the file type accept property. This is what determines the file\r\n   * type choices that the user will be limited to in the file browse dialog\r\n   */\r\n  private updateFileTypeAccept(): void {\r\n    if (this.fileType) {\r\n      if (this.fileType !== \"custom\") {\r\n        this.fileTypeAccept = FileTypeExtensions[this.fileType].join(\",\");\r\n      } else {\r\n        if (this.customExtensions) {\r\n          this.fileTypeAccept = this.customExtensions.join(\",\");\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Take a file that was selected by the user and process/patch our form model\r\n   * If the host component requires an action to occur with the file prior to the patch it will call\r\n   * and wait for it to return.\r\n   * @param file \r\n   * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64\r\n   */\r\n  private async processFile(file: File, base64FileString?: string | undefined): Promise<void> {\r\n    if (this.onFileSelected) {\r\n      try {\r\n        let result = await this.onFileSelected(file);\r\n        this.patchFileResult(file, base64FileString, result);\r\n      } catch (e) {\r\n        // Bummer, we're not going to do anything about it though.\r\n        // We are not patching any of the result so any existing information remains\r\n      }\r\n    } else {\r\n      this.patchFileResult(file, base64FileString);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Based on the fileOutput return whether this component is expected to deliver a base64 output\r\n   * @returns \r\n   */\r\n  private isBase64FileOutput(): boolean {\r\n    return this.fileOutput === 'base64';\r\n  }\r\n\r\n  /**\r\n   * When the file was selected and processed patch the file information that the hosting form will\r\n   * be looking for. \r\n   * @param file \r\n   * @param base64FileString \r\n   * @param onFileSelectedResult \r\n   */\r\n  private patchFileResult(file: File | null, base64FileString?: string, onFileSelectedResult?: any): void {    \r\n    console.log('inside patchFileResult');\r\n\r\n    // Get the current value of files from formModel\r\n    let files = this.formModel?.get('files')?.value || [];\r\n    let names = this.formModel?.get('names')?.value || [];\r\n\r\n    // If multiSelect is true we need to append the file to the existing array\r\n    // otherwise we'll just set the file to the form model\r\n    if (this.multiSelect) {\r\n      if (file) { files = [ ...files, file ]; }\r\n      if (file?.name) { names = [ ...names, file.name ]; }\r\n\r\n    } else {\r\n      files = [ file ];\r\n      names = [ file?.name ];\r\n    }\r\n\r\n    this.formModel?.patchValue({\r\n      files: files,\r\n      names: names,\r\n      base64FileString: base64FileString ?? null\r\n    });\r\n\r\n    if (onFileSelectedResult) {\r\n      this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n    } else {\r\n      this.formModel.patchValue({ uploadResult: null });\r\n    }\r\n\r\n    // console.log('formModel.value', this.formModel?.value);\r\n    console.log('this.formModel?.get(\"names\")?.value', this.formModel?.get('names')?.value);\r\n  }\r\n\r\n  /* ---------------------------- Helper Functions ---------------------------- */\r\n  private clearFormModel(): void {\r\n    this.formModel?.patchValue({\r\n      files: null,\r\n      names: null,\r\n      base64FileString: null,\r\n      uploadResult: null\r\n    });\r\n  }\r\n\r\n  private async handleMultipleFiles(files: FileList): Promise<void> {\r\n    const promises = Array.from(files).map((file) => this.readFile(file));\r\n    await Promise.all(promises);\r\n  };\r\n\r\n  private async readFile(file: File): Promise<void> {\r\n    const reader: FileReader = new FileReader();\r\n\r\n    reader.onloadend = async e => {\r\n      const base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n      await this.processFile(file, base64FileString);\r\n    };\r\n\r\n    if (this.isBase64FileOutput()) {\r\n      reader.readAsDataURL(file);\r\n    } else {\r\n      await this.processFile(file);\r\n    }\r\n  }\r\n}\r\n","<ec-form-group [label]=\"label\"\r\n               [formGroup]=\"formModel\"\r\n               class=\"mb-0\">\r\n  <div class=\"d-flex control-group\">\r\n    <div class=\"d-flex flex-grow position-relative\">\r\n      <input #fileInput\r\n             id=\"{{inputId}}_input\"\r\n             type=\"file\"\r\n             tabindex=\"-1\"\r\n             [attr.accept]=\"fileTypeAccept\"\r\n             (change)=\"fileChange($event.target.files)\"\r\n             [class.has-value]=\"displayType === 'file' ? formModel?.get('names').value : undefined\"\r\n             [attr.multiple]=\"multiSelect ? 'multiple' : undefined\">\r\n            <ec-form-control *ngIf=\"displayType === 'file'\"\r\n                       id=\"{{inputId}}_formControl\"\r\n                       class=\"text-truncate\"\r\n                       [required]=\"required\"\r\n                       [pending]=\"pending\">\r\n        <input id=\"{{inputId}}_name\"\r\n               [formControl]=\"formModel?.get('names')\"\r\n               type=\"text\"\r\n               [placeholder]=\"placeholder\"\r\n               [tabindex]=\"-1\">\r\n      </ec-form-control>\r\n    </div>\r\n    <ec-button *ngIf=\"displayType === 'file'\"\r\n               #browseBtn\r\n               id=\"{{inputId}}_browseBtn\"\r\n               (clicked)=\"fileInput.click()\"\r\n               type=\"secondary\"\r\n               [tabindex]=\"tabindex\"\r\n               [disabled]=\"formModel?.get('names').disabled\"\r\n               label=\"Browse\"\r\n               [autofocus]=\"autofocus\">\r\n    </ec-button>\r\n  </div>\r\n  <ec-button *ngIf=\"displayType === 'button'\"\r\n             #selectZipFilesBtn\r\n             id=\"{{inputId}}_btn\"\r\n             [pending]=\"pending\"\r\n             type=\"primary\"\r\n             [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n             (clicked)=\"fileInput.click()\"\r\n             style=\"width: 100%;\">\r\n  </ec-button>\r\n</ec-form-group>"]}
@@ -3808,6 +3808,10 @@ class FileUploadComponent extends FormControlBase {
3808
3808
  * Default: file
3809
3809
  */
3810
3810
  this.displayType = 'file';
3811
+ /**
3812
+ * Optional property to control whether the user can select multiple files
3813
+ */
3814
+ this.multiSelect = false;
3811
3815
  }
3812
3816
  ngOnChanges(changes) {
3813
3817
  super.ngOnChanges(changes);
@@ -3819,7 +3823,7 @@ class FileUploadComponent extends FormControlBase {
3819
3823
  // Watch for name to change, if the value is cleared we will clear the other
3820
3824
  // supporting model properties. The name can be cleared by the user manually or via
3821
3825
  // the clear button
3822
- (_b = (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.get('name')) === null || _b === void 0 ? void 0 : _b.valueChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(value => {
3826
+ (_b = (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.get('names')) === null || _b === void 0 ? void 0 : _b.valueChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(value => {
3823
3827
  if (!value) {
3824
3828
  this.formModel.patchValue({
3825
3829
  files: null,
@@ -3830,28 +3834,19 @@ class FileUploadComponent extends FormControlBase {
3830
3834
  }
3831
3835
  });
3832
3836
  }
3833
- // * New fileChange method
3834
3837
  fileChange(files) {
3835
3838
  return __awaiter(this, void 0, void 0, function* () {
3836
- const promises = Array.from(files).map((file) => new Promise((resolve) => {
3839
+ if (this.multiSelect) {
3840
+ // Since we append the files to the form model when multiSelect=true, we need to clear the form model at the start
3841
+ this.clearFormModel();
3842
+ yield this.handleMultipleFiles(files);
3843
+ }
3844
+ else {
3845
+ const file = files.item(0);
3837
3846
  if (file) {
3838
- const reader = new FileReader();
3839
- reader.onloadend = (e) => __awaiter(this, void 0, void 0, function* () {
3840
- var _a;
3841
- const base64FileString = (_a = reader === null || reader === void 0 ? void 0 : reader.result) === null || _a === void 0 ? void 0 : _a.toString().split(",")[1];
3842
- yield this.processFile(file, base64FileString);
3843
- resolve(null);
3844
- });
3845
- if (this.isBase64FileOutput()) {
3846
- reader.readAsDataURL(file);
3847
- }
3848
- else {
3849
- resolve(this.processFile(file));
3850
- }
3851
- ;
3847
+ yield this.readFile(file);
3852
3848
  }
3853
- }));
3854
- yield Promise.all(promises);
3849
+ }
3855
3850
  // Clear the file inputs value, this will allow the user to pick the same filenames again
3856
3851
  // and cause fileChange to re-trigger.
3857
3852
  if (this.fileInput) {
@@ -3859,28 +3854,6 @@ class FileUploadComponent extends FormControlBase {
3859
3854
  }
3860
3855
  });
3861
3856
  }
3862
- // * Old Method
3863
- // public async fileChange(files: FileList): Promise<void> {
3864
- // let file = files.item(0);
3865
- // // If there is a file selected and then opened again and click cancel you get null so don't try and set anything
3866
- // if (file) {
3867
- // let reader: FileReader = new FileReader();
3868
- // reader.onloadend = async e => {
3869
- // let base64FileString: string | undefined = reader?.result?.toString().split(",")[1];
3870
- // this.processFile(file!, base64FileString);
3871
- // };
3872
- // if (this.isBase64FileOutput()) {
3873
- // reader.readAsDataURL(file);
3874
- // } else {
3875
- // await this.processFile(file!);
3876
- // }
3877
- // // Clear the file inputs value, this will allow the user to pick the same filename and cause
3878
- // // the fileChange to trigger.
3879
- // if (this.fileInput) {
3880
- // this.fileInput.nativeElement.value = '';
3881
- // }
3882
- // }
3883
- // }
3884
3857
  /**
3885
3858
  * Checks the file type and updates the file type accept property. This is what determines the file
3886
3859
  * type choices that the user will be limited to in the file browse dialog
@@ -3912,7 +3885,7 @@ class FileUploadComponent extends FormControlBase {
3912
3885
  this.patchFileResult(file, base64FileString, result);
3913
3886
  }
3914
3887
  catch (e) {
3915
- // Bummer, we're not going to do anything about it though.
3888
+ // Bummer, we're not going to do anything about it though.
3916
3889
  // We are not patching any of the result so any existing information remains
3917
3890
  }
3918
3891
  }
@@ -3935,19 +3908,25 @@ class FileUploadComponent extends FormControlBase {
3935
3908
  * @param base64FileString
3936
3909
  * @param onFileSelectedResult
3937
3910
  */
3938
- // * Old method
3939
3911
  patchFileResult(file, base64FileString, onFileSelectedResult) {
3940
- var _a, _b, _c, _d, _e;
3912
+ var _a, _b, _c, _d, _e, _f, _g;
3913
+ console.log('inside patchFileResult');
3941
3914
  // Get the current value of files from formModel
3942
3915
  let files = ((_b = (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.get('files')) === null || _b === void 0 ? void 0 : _b.value) || [];
3943
3916
  let names = ((_d = (_c = this.formModel) === null || _c === void 0 ? void 0 : _c.get('names')) === null || _d === void 0 ? void 0 : _d.value) || [];
3944
- // If the file is not null, append it to the files array
3945
- if (file) {
3946
- files = [...files, file];
3917
+ // If multiSelect is true we need to append the file to the existing array
3918
+ // otherwise we'll just set the file to the form model
3919
+ if (this.multiSelect) {
3920
+ if (file) {
3921
+ files = [...files, file];
3922
+ }
3923
+ if (file === null || file === void 0 ? void 0 : file.name) {
3924
+ names = [...names, file.name];
3925
+ }
3947
3926
  }
3948
- // If the file name is not null, append it to the names array
3949
- if (file === null || file === void 0 ? void 0 : file.name) {
3950
- names = [...names, file.name];
3927
+ else {
3928
+ files = [file];
3929
+ names = [file === null || file === void 0 ? void 0 : file.name];
3951
3930
  }
3952
3931
  (_e = this.formModel) === null || _e === void 0 ? void 0 : _e.patchValue({
3953
3932
  files: files,
@@ -3960,13 +3939,48 @@ class FileUploadComponent extends FormControlBase {
3960
3939
  else {
3961
3940
  this.formModel.patchValue({ uploadResult: null });
3962
3941
  }
3942
+ // console.log('formModel.value', this.formModel?.value);
3943
+ console.log('this.formModel?.get("names")?.value', (_g = (_f = this.formModel) === null || _f === void 0 ? void 0 : _f.get('names')) === null || _g === void 0 ? void 0 : _g.value);
3944
+ }
3945
+ /* ---------------------------- Helper Functions ---------------------------- */
3946
+ clearFormModel() {
3947
+ var _a;
3948
+ (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.patchValue({
3949
+ files: null,
3950
+ names: null,
3951
+ base64FileString: null,
3952
+ uploadResult: null
3953
+ });
3954
+ }
3955
+ handleMultipleFiles(files) {
3956
+ return __awaiter(this, void 0, void 0, function* () {
3957
+ const promises = Array.from(files).map((file) => this.readFile(file));
3958
+ yield Promise.all(promises);
3959
+ });
3960
+ }
3961
+ ;
3962
+ readFile(file) {
3963
+ return __awaiter(this, void 0, void 0, function* () {
3964
+ const reader = new FileReader();
3965
+ reader.onloadend = (e) => __awaiter(this, void 0, void 0, function* () {
3966
+ var _a;
3967
+ const base64FileString = (_a = reader === null || reader === void 0 ? void 0 : reader.result) === null || _a === void 0 ? void 0 : _a.toString().split(",")[1];
3968
+ yield this.processFile(file, base64FileString);
3969
+ });
3970
+ if (this.isBase64FileOutput()) {
3971
+ reader.readAsDataURL(file);
3972
+ }
3973
+ else {
3974
+ yield this.processFile(file);
3975
+ }
3976
+ });
3963
3977
  }
3964
3978
  }
3965
3979
  FileUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, deps: [{ token: ValidationMessageService }, { token: FormGroupHelper }], target: i0.ɵɵFactoryTarget.Component });
3966
- FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected", displayType: "displayType", buttonLabel: "buttonLabel" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
3980
+ FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected", displayType: "displayType", buttonLabel: "buttonLabel", multiSelect: "multiSelect" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"displayType === 'file' ? formModel?.get('names').value : undefined\"\r\n [attr.multiple]=\"multiSelect ? 'multiple' : undefined\">\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('names')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('names').disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_btn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
3967
3981
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, decorators: [{
3968
3982
  type: Component,
3969
- args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
3983
+ args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"displayType === 'file' ? formModel?.get('names').value : undefined\"\r\n [attr.multiple]=\"multiSelect ? 'multiple' : undefined\">\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('names')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('names').disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_btn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
3970
3984
  }], ctorParameters: function () { return [{ type: ValidationMessageService }, { type: FormGroupHelper }]; }, propDecorators: { placeholder: [{
3971
3985
  type: Input
3972
3986
  }], fileType: [{
@@ -3981,6 +3995,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
3981
3995
  type: Input
3982
3996
  }], buttonLabel: [{
3983
3997
  type: Input
3998
+ }], multiSelect: [{
3999
+ type: Input
3984
4000
  }], fileInput: [{
3985
4001
  type: ViewChild,
3986
4002
  args: ["fileInput", { read: ElementRef, static: true }]