@energycap/components 0.39.21-ECAP-25650-file-upload-validation-support.20240806-1029 → 0.39.21-ECAP-25650-file-upload-validation-support.20240806-1534

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.
@@ -17,7 +17,9 @@ export const FileTypeExtensions = {
17
17
  };
18
18
  export class FileUploadComponent extends FormControlBase {
19
19
  // static class to create the form group from a parent component
20
- static getFormModel(validators, disabled = false, fileValidators) {
20
+ static getFormModel(validators, disabled = false,
21
+ /** Any validators required to make sure the selected file is valid. It is recommended that you use `FileUploadComponent.getFileValidator` to help construct the validator(s). NOTE: This currently only works when multiSelect is false. */
22
+ fileValidators) {
21
23
  let formGroup = new FormGroup({
22
24
  file: new FormControl({ value: null, disabled: disabled }, { validators: validators, asyncValidators: fileValidators }),
23
25
  name: new FormControl({ value: null, disabled: disabled }, validators),
@@ -29,10 +31,18 @@ export class FileUploadComponent extends FormControlBase {
29
31
  }
30
32
  return formGroup;
31
33
  }
34
+ /**
35
+ * Helper function that returns an async validator to be used with the file upload control.
36
+ * This is useful for when the file needs to be validated before it can be uploaded.
37
+ *
38
+ * @param callback The callback function that will be called to validate the file. Parameters for the callback are the file and the base64 string for the file.
39
+ * base64 is null if the fileOutput input on the FileUploadComponent is set to raw. Using fileOutput set to base64 is recommended for images.
40
+ */
32
41
  static getFileValidator(callback) {
33
42
  return async (control) => {
34
43
  if (control.value && control.parent) {
35
44
  let file = control.value;
45
+ // For images, we need the base64 string to validate image dimensions
36
46
  let base64 = control.parent.get('base64FileString')?.value;
37
47
  return await callback(file, base64);
38
48
  }
@@ -84,7 +94,8 @@ export class FileUploadComponent extends FormControlBase {
84
94
  });
85
95
  }
86
96
  });
87
- // Sync errors from the file control to the name control whenever the file control status changes
97
+ // Sync errors from the file control to the name control whenever the file control status changes.
98
+ // The name control is the only one displayed to the user so we need to show the errors there.
88
99
  this.formModel.get('file')?.statusChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(() => {
89
100
  const errors = this.formModel.get('file')?.errors ?? null;
90
101
  this.formModel.get('name')?.setErrors(errors);
@@ -129,26 +140,47 @@ export class FileUploadComponent extends FormControlBase {
129
140
  * @param file
130
141
  * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64
131
142
  */
132
- async processFile(file, base64FileString) {
143
+ async processFile(file, base64FileString = null) {
144
+ // If we have async validators on the file control we need to do validation before we trigger the upload
145
+ const validateBeforeUpload = !!this.formModel.controls.file.asyncValidator;
146
+ if (validateBeforeUpload) {
147
+ await this.validateFile(file, base64FileString);
148
+ }
149
+ if (this.onFileSelected) {
150
+ // Only call the onFileSelected callback to upload the file if the form group is valid
151
+ if (this.formModel.valid) {
152
+ try {
153
+ let result = await this.onFileSelected(file);
154
+ // If we did validation, just patch the form result because the file is already in the form
155
+ if (validateBeforeUpload) {
156
+ this.formModel.patchValue({ uploadResult: result ?? null });
157
+ }
158
+ else {
159
+ this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: result ?? null });
160
+ }
161
+ }
162
+ catch (e) {
163
+ // Bummer, we're not going to do anything about it though.
164
+ // We are not patching any of the result so any existing information remains
165
+ }
166
+ }
167
+ // If we don't have an onFileSelected callback we just patch the form model.
168
+ // In the case of pre-upload validation the form already has the file so we don't want to patch again.
169
+ }
170
+ else if (!validateBeforeUpload) {
171
+ this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: null });
172
+ }
173
+ }
174
+ /** Patches the form with the selected file in order to trigger control validation */
175
+ async validateFile(file, base64FileString = null) {
133
176
  // Patch the file first to trigger any file validators
134
- this.patchProcessedFile(file, base64FileString);
177
+ this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: null });
135
178
  // If we have any async validators pending we need to wait for them to complete before we know if the form is valid
136
179
  if (this.formModel.pending) {
137
180
  await this.formModel.statusChanges.pipe(filter(status => status !== 'PENDING'), take(1), takeUntil(this.componentDestroyed)).toPromise();
138
181
  }
139
182
  // Mark the name control as touched so that any validation errors will show
140
183
  this.formModel.controls.name.markAsTouched();
141
- // Only call the onFileSelected callback to upload the file if the form is valid
142
- if (this.onFileSelected && this.formModel.valid) {
143
- try {
144
- let result = await this.onFileSelected(file);
145
- this.formModel.patchValue({ uploadResult: result ?? null });
146
- }
147
- catch (e) {
148
- // Bummer, we're not going to do anything about it though.
149
- // We are not patching any of the result so any existing information remains
150
- }
151
- }
152
184
  }
153
185
  /**
154
186
  * Based on the fileOutput return whether this component is expected to deliver a base64 output
@@ -157,21 +189,6 @@ export class FileUploadComponent extends FormControlBase {
157
189
  isBase64FileOutput() {
158
190
  return this.fileOutput === 'base64';
159
191
  }
160
- /**
161
- * When the file was selected and processed patch the file information that the hosting form will
162
- * be looking for.
163
- * @param file
164
- * @param base64FileString
165
- * @param onFileSelectedResult
166
- */
167
- patchProcessedFile(file, base64FileString) {
168
- this.formModel?.patchValue({
169
- file: file,
170
- name: file?.name,
171
- base64FileString: base64FileString ?? null,
172
- uploadResult: null
173
- });
174
- }
175
192
  /** Maps the files to an array of File objects and sends them along
176
193
  * to the derived onMultipleFileSelected method in the hosting component
177
194
  */
@@ -232,4 +249,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
232
249
  type: ViewChild,
233
250
  args: ["fileInput", { read: ElementRef, static: true }]
234
251
  }] } });
235
- //# 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":"AAAA,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,KAAK,EAAoC,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1G,OAAO,EAAqC,WAAW,EAAE,SAAS,EAAe,MAAM,gBAAgB,CAAC;AACxG,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAGzD,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;;;;;;;;;;AAKvD,MAAM,CAAC,MAAM,kBAAkB,GAAG;IAChC,GAAG,EAAE,CAAC,MAAM,CAAC;IACb,KAAK,EAAE,CAAC,MAAM,EAAE,OAAO,CAAC;CACzB,CAAC;AAoBF,MAAM,OAAO,mBAAoB,SAAQ,eAAe;IAEtD,gEAAgE;IACzD,MAAM,CAAC,YAAY,CACxB,UAAyB,EACzB,WAAoB,KAAK,EACzB,cAAsD;QAEtD,IAAI,SAAS,GAAG,IAAI,SAAS,CAAsB;YACjD,IAAI,EAAE,IAAI,WAAW,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,EAAC,UAAU,EAAE,UAAU,EAAE,eAAe,EAAE,cAAc,EAAC,CAAC;YACrH,IAAI,EAAE,IAAI,WAAW,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,UAAU,CAAC;YACtE,gBAAgB,EAAE,IAAI,WAAW,CAAC,IAAI,CAAC;YACvC,YAAY,EAAE,IAAI,WAAW,CAAC,IAAI,CAAC;SACpC,CAAC,CAAC;QACH,IAAI,QAAQ,EAAE;YACZ,SAAS,CAAC,OAAO,EAAE,CAAC;SACrB;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAEM,MAAM,CAAC,gBAAgB,CAAC,QAA+B;QAC5D,OAAO,KAAK,EAAE,OAAwB,EAAE,EAAE;YACxC,IAAI,OAAO,CAAC,KAAK,IAAI,OAAO,CAAC,MAAM,EAAE;gBACnC,IAAI,IAAI,GAAG,OAAO,CAAC,KAAa,CAAC;gBACjC,IAAI,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,kBAAkB,CAAC,EAAE,KAAe,CAAC;gBACrE,OAAO,MAAM,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;aACrC;YACD,OAAO,IAAI,CAAC;QACd,CAAC,CAAA;IACH,CAAC;IAgED,YACY,wBAAkD,EAClD,eAAgC;QAE1C,KAAK,CAAC,wBAAwB,EAAE,eAAe,CAAC,CAAC;QAHvC,6BAAwB,GAAxB,wBAAwB,CAA0B;QAClD,oBAAe,GAAf,eAAe,CAAiB;QA9D5C;;WAEG;QACa,gBAAW,GAAY,gBAAgB,CAAC;QAQxD;;WAEG;QACa,eAAU,GAAgB,QAAQ,CAAC;QAmBnD;;;;WAIG;QACa,gBAAW,GAAuB,MAAM,CAAC;QAOzD;;WAEG;QACa,eAAU,GAAY,SAAS,CAAC;QAEhD;;WAEG;QACa,gBAAW,GAAa,KAAK,CAAC;IAY9C,CAAC;IAEM,WAAW,CAAC,OAAsB;QACvC,KAAK,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QAE3B,IAAI,CAAC,oBAAoB,EAAE,CAAC;IAC9B,CAAC;IAEM,QAAQ;QACb,KAAK,CAAC,QAAQ,EAAE,CAAC;QAEjB,4EAA4E;QAC5E,mFAAmF;QACnF,mBAAmB;QACnB,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,MAAM,CAAC,EAAE,YAAY,CAAC,IAAI,CAC5C,SAAS,CAAC,IAAI,CAAC,kBAAkB,CAAC,CACnC,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YAClB,IAAI,CAAC,KAAK,EAAE;gBACV,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC;oBACxB,IAAI,EAAE,IAAI;oBACV,gBAAgB,EAAE,IAAI;oBACtB,YAAY,EAAE,IAAI;iBACnB,CAAC,CAAC;aACJ;QACH,CAAC,CAAC,CAAC;QAEH,iGAAiG;QACjG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,aAAa,CAAC,IAAI,CAC5C,SAAS,CAAC,IAAI,CAAC,kBAAkB,CAAC,CACnC,CAAC,SAAS,CAAC,GAAG,EAAE;YACf,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,MAAM,IAAI,IAAI,CAAC;YAC1D,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,SAAS,CAAC,MAAM,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;IACL,CAAC;IAEM,KAAK,CAAC,UAAU,CAAC,KAAe;QACrC,IAAI,IAAI,CAAC,WAAW,EAAE;YACpB,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;SACjC;aAAM;YACL,MAAM,IAAI,GAAgB,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YACxC,IAAI,IAAI,EAAE;gBACR,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;aAC3B;SACF;QAED,yFAAyF;QACzF,sCAAsC;QACtC,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,KAAK,GAAG,EAAE,CAAC;SACzC;IACH,CAAC;IAED;;;OAGG;IACK,oBAAoB;QAC1B,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,IAAI,CAAC,QAAQ,KAAK,QAAQ,EAAE;gBAC9B,IAAI,CAAC,cAAc,GAAG,kBAAkB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;aACnE;iBAAM;gBACL,IAAI,IAAI,CAAC,gBAAgB,EAAE;oBACzB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;iBACvD;aACF;SACF;IACH,CAAC;IAED;;;;;;OAMG;IACK,KAAK,CAAC,WAAW,CAAC,IAAU,EAAE,gBAAqC;QACzE,sDAAsD;QACtD,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,gBAAgB,CAAC,CAAC;QAEhD,mHAAmH;QACnH,IAAI,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE;YAC1B,MAAM,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,IAAI,CACrC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,KAAK,SAAS,CAAC,EACtC,IAAI,CAAC,CAAC,CAAC,EACP,SAAS,CAAC,IAAI,CAAC,kBAAkB,CAAC,CACnC,CAAC,SAAS,EAAE,CAAC;SACf;QAED,2EAA2E;QAC3E,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;QAE7C,gFAAgF;QAChF,IAAI,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE;YAC/C,IAAI;gBACF,IAAI,MAAM,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;gBAC7C,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,EAAE,YAAY,EAAE,MAAM,IAAI,IAAI,EAAE,CAAC,CAAC;aAC7D;YAAC,OAAO,CAAC,EAAE;gBACV,0DAA0D;gBAC1D,4EAA4E;aAC7E;SACF;IACH,CAAC;IAED;;;OAGG;IACK,kBAAkB;QACxB,OAAO,IAAI,CAAC,UAAU,KAAK,QAAQ,CAAC;IACtC,CAAC;IAED;;;;;;OAMG;IACK,kBAAkB,CAAC,IAAiB,EAAE,gBAAyB;QACrE,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC;YACzB,IAAI,EAAE,IAAI;YACV,IAAI,EAAE,IAAI,EAAE,IAAI;YAChB,gBAAgB,EAAE,gBAAgB,IAAI,IAAI;YAC1C,YAAY,EAAE,IAAI;SACnB,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,mBAAmB,CAAC,KAAe;QAC/C,MAAM,UAAU,GAAG,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAErC,IAAI,IAAI,CAAC,uBAAuB,EAAE;YAChC,IAAI;gBACF,IAAI,MAAM,GAAG,MAAM,IAAI,CAAC,uBAAuB,CAAC,UAAU,CAAC,CAAC;aAC7D;YAAC,OAAO,CAAC,EAAE;gBACV,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;aAC3B;SACF;IACH,CAAC;IAAA,CAAC;IAEM,KAAK,CAAC,QAAQ,CAAC,IAAU;QAC/B,MAAM,MAAM,GAAe,IAAI,UAAU,EAAE,CAAC;QAE5C,MAAM,CAAC,SAAS,GAAG,KAAK,EAAC,CAAC,EAAC,EAAE;YAC3B,IAAI,gBAAgB,GAAuB,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;YACpF,MAAM,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,gBAAgB,CAAC,CAAC;QACjD,CAAC,CAAC;QAEF,IAAI,IAAI,CAAC,kBAAkB,EAAE,EAAE;YAC7B,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;SAC5B;aAAM;YACL,MAAM,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;SAC9B;IACH,CAAC;;gHA7PU,mBAAmB;oGAAnB,mBAAmB,meAwFE,UAAU,uFCzH5C,iiEA8CgB;2FDbH,mBAAmB;kBAL/B,SAAS;+BACE,gBAAgB;6IAmCV,SAAS;sBAAxB,KAAK;gBAKU,WAAW;sBAA1B,KAAK;gBAMU,QAAQ;sBAAvB,KAAK;gBAKU,UAAU;sBAAzB,KAAK;gBAGU,gBAAgB;sBAA/B,KAAK;gBAOU,cAAc;sBAA7B,KAAK;gBAOU,uBAAuB;sBAAtC,KAAK;gBAOU,WAAW;sBAA1B,KAAK;gBAKU,WAAW;sBAA1B,KAAK;gBAKU,UAAU;sBAAzB,KAAK;gBAKU,WAAW;sBAA1B,KAAK;gBAE6D,SAAS;sBAA3E,SAAS;uBAAC,WAAW,EAAE,EAAE,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { AbstractControl, AsyncValidatorFn, FormControl, FormGroup, ValidatorFn } from '@angular/forms';\r\nimport { filter, take, 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\nexport type FileUploadFormGroup = {\r\n  file: FormControl<File | null>;\r\n  name: FormControl<string | null>;\r\n  base64FileString: FormControl<string | null>;\r\n  uploadResult: FormControl<any>;\r\n};\r\n\r\nexport type InvalidFileError = {\r\n  invalidFile: true;\r\n};\r\n\r\nexport type FileValidatorCallback = (file: File | null | undefined, base64: string | null | undefined) => Promise<InvalidFileError | null>;\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    fileValidators?: AsyncValidatorFn | AsyncValidatorFn[]\r\n  ): FormGroup<FileUploadFormGroup> {\r\n    let formGroup = new FormGroup<FileUploadFormGroup>({\r\n      file: new FormControl({ value: null, disabled: disabled }, {validators: validators, asyncValidators: fileValidators}),\r\n      name: new FormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new FormControl(null),\r\n      uploadResult: new FormControl(null),\r\n    });\r\n    if (disabled) {\r\n      formGroup.disable();\r\n    }\r\n    return formGroup;\r\n  }\r\n\r\n  public static getFileValidator(callback: FileValidatorCallback): AsyncValidatorFn {\r\n    return async (control: AbstractControl) => {\r\n      if (control.value && control.parent) {\r\n        let file = control.value as File;\r\n        let base64 = control.parent.get('base64FileString')?.value as string;\r\n        return await callback(file, base64);\r\n      }\r\n      return null;\r\n    }\r\n  }\r\n\r\n  @Input() public formModel!: FormGroup<FileUploadFormGroup>;\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?: (file: File) => Promise<any>;\r\n\r\n  /**\r\n   * Optional callback supported for when the hosting page needs to process multiple files\r\n   * We simply map the FileList to an array of File objects and pass that array to the \r\n   * callback and let the hosting page handle the processing\r\n   */\r\n  @Input() public onMultipleFilesSelected?: (files: 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   * When display type is set to button this property will control the button type\r\n   */\r\n  @Input() public buttonType?: string = 'primary';\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('name')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          file: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n\r\n    // Sync errors from the file control to the name control whenever the file control status changes\r\n    this.formModel.get('file')?.statusChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(() => {\r\n      const errors = this.formModel.get('file')?.errors ?? null;\r\n      this.formModel.get('name')?.setErrors(errors);\r\n    });\r\n  }\r\n\r\n  public async fileChange(files: FileList): Promise<void> {\r\n    if (this.multiSelect) {\r\n      this.handleMultipleFiles(files);\r\n    } else {\r\n      const file: File | null = 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    // Patch the file first to trigger any file validators\r\n    this.patchProcessedFile(file, base64FileString);\r\n\r\n    // If we have any async validators pending we need to wait for them to complete before we know if the form is valid\r\n    if (this.formModel.pending) {\r\n      await this.formModel.statusChanges.pipe(\r\n        filter(status => status !== 'PENDING'),\r\n        take(1),\r\n        takeUntil(this.componentDestroyed)\r\n      ).toPromise();\r\n    }\r\n\r\n    // Mark the name control as touched so that any validation errors will show\r\n    this.formModel.controls.name.markAsTouched();\r\n\r\n    // Only call the onFileSelected callback to upload the file if the form is valid\r\n    if (this.onFileSelected && this.formModel.valid) {\r\n      try {\r\n        let result = await this.onFileSelected(file);\r\n        this.formModel.patchValue({ uploadResult: result ?? null });\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    }\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 patchProcessedFile(file: File | null, base64FileString?: string): void {\r\n    this.formModel?.patchValue({ \r\n      file: file, \r\n      name: file?.name, \r\n      base64FileString: base64FileString ?? null,\r\n      uploadResult: null\r\n    });\r\n  }\r\n\r\n  /** Maps the files to an array of File objects and sends them along\r\n   * to the derived onMultipleFileSelected method in the hosting component\r\n   */\r\n  private async handleMultipleFiles(files: FileList): Promise<void> {\r\n    const filesArray = Array.from(files);\r\n\r\n    if (this.onMultipleFilesSelected) {\r\n      try {\r\n        let result = await this.onMultipleFilesSelected(filesArray);\r\n      } catch (e) {\r\n        console.log('error: ', e);\r\n      }  \r\n    }\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      let 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               [helpPopover]=\"helpPopover\"\r\n               [helpPopoverPosition]=\"helpPopoverPosition\"\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('name').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 || formModel?.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             id=\"{{inputId}}_btn\"\r\n             [pending]=\"pending\"\r\n             [type]=\"buttonType\"\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>"]}
252
+ //# 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 { AbstractControl, AsyncValidatorFn, FormControl, FormGroup, ValidatorFn } from '@angular/forms';\r\nimport { filter, take, 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\nexport type FileUploadFormGroup = {\r\n  file: FormControl<File | null>;\r\n  name: FormControl<string | null>;\r\n  base64FileString: FormControl<string | null>;\r\n  uploadResult: FormControl<any>;\r\n};\r\n\r\nexport type InvalidFileError = {\r\n  invalidFile: true;\r\n};\r\n\r\nexport type FileValidatorCallback = (file: File | null | undefined, base64: string | null | undefined) => Promise<InvalidFileError | null>;\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    /** Any validators required to make sure the selected file is valid. It is recommended that you use `FileUploadComponent.getFileValidator` to help construct the validator(s). NOTE: This currently only works when multiSelect is false. */\r\n    fileValidators?: AsyncValidatorFn | AsyncValidatorFn[]\r\n  ): FormGroup<FileUploadFormGroup> {\r\n    let formGroup = new FormGroup<FileUploadFormGroup>({\r\n      file: new FormControl({ value: null, disabled: disabled }, {validators: validators, asyncValidators: fileValidators}),\r\n      name: new FormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new FormControl(null),\r\n      uploadResult: new FormControl(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   * Helper function that returns an async validator to be used with the file upload control.\r\n   * This is useful for when the file needs to be validated before it can be uploaded.\r\n   * \r\n   * @param callback The callback function that will be called to validate the file. Parameters for the callback are the file and the base64 string for the file.\r\n   * base64 is null if the fileOutput input on the FileUploadComponent is set to raw. Using fileOutput set to base64 is recommended for images.\r\n   */\r\n  public static getFileValidator(callback: FileValidatorCallback): AsyncValidatorFn {\r\n    return async (control: AbstractControl) => {\r\n      if (control.value && control.parent) {\r\n        let file = control.value as File;\r\n        // For images, we need the base64 string to validate image dimensions\r\n        let base64 = control.parent.get('base64FileString')?.value as string;\r\n        return await callback(file, base64);\r\n      }\r\n      return null;\r\n    }\r\n  }\r\n\r\n  @Input() public formModel!: FormGroup<FileUploadFormGroup>;\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?: (file: File) => Promise<any>;\r\n\r\n  /**\r\n   * Optional callback supported for when the hosting page needs to process multiple files\r\n   * We simply map the FileList to an array of File objects and pass that array to the \r\n   * callback and let the hosting page handle the processing\r\n   */\r\n  @Input() public onMultipleFilesSelected?: (files: 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   * When display type is set to button this property will control the button type\r\n   */\r\n  @Input() public buttonType?: string = 'primary';\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('name')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          file: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n\r\n    // Sync errors from the file control to the name control whenever the file control status changes.\r\n    // The name control is the only one displayed to the user so we need to show the errors there.\r\n    this.formModel.get('file')?.statusChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(() => {\r\n      const errors = this.formModel.get('file')?.errors ?? null;\r\n      this.formModel.get('name')?.setErrors(errors);\r\n    });\r\n  }\r\n\r\n  public async fileChange(files: FileList): Promise<void> {\r\n    if (this.multiSelect) {\r\n      this.handleMultipleFiles(files);\r\n    } else {\r\n      const file: File | null = 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 | null = null): Promise<void> {\r\n    // If we have async validators on the file control we need to do validation before we trigger the upload\r\n    const validateBeforeUpload = !!this.formModel.controls.file.asyncValidator;\r\n    if (validateBeforeUpload) {\r\n      await this.validateFile(file, base64FileString);\r\n    }\r\n    \r\n    if (this.onFileSelected) {\r\n      // Only call the onFileSelected callback to upload the file if the form group is valid\r\n      if (this.formModel.valid) {\r\n        try {\r\n          let result = await this.onFileSelected(file);\r\n  \r\n          // If we did validation, just patch the form result because the file is already in the form\r\n          if (validateBeforeUpload) {\r\n            this.formModel.patchValue({ uploadResult: result ?? null });\r\n          } else {\r\n            this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: result ?? null });\r\n          }\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      }\r\n    // If we don't have an onFileSelected callback we just patch the form model.\r\n    // In the case of pre-upload validation the form already has the file so we don't want to patch again.\r\n    } else if (!validateBeforeUpload) {\r\n      this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: null });\r\n    }\r\n  }\r\n\r\n  /** Patches the form with the selected file in order to trigger control validation */\r\n  private async validateFile(file: File, base64FileString: string | null = null): Promise<void> {\r\n    // Patch the file first to trigger any file validators\r\n    this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: null });\r\n\r\n    // If we have any async validators pending we need to wait for them to complete before we know if the form is valid\r\n    if (this.formModel.pending) {\r\n      await this.formModel.statusChanges.pipe(\r\n        filter(status => status !== 'PENDING'),\r\n        take(1),\r\n        takeUntil(this.componentDestroyed)\r\n      ).toPromise();\r\n    }\r\n\r\n    // Mark the name control as touched so that any validation errors will show\r\n    this.formModel.controls.name.markAsTouched();\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  /** Maps the files to an array of File objects and sends them along\r\n   * to the derived onMultipleFileSelected method in the hosting component\r\n   */\r\n  private async handleMultipleFiles(files: FileList): Promise<void> {\r\n    const filesArray = Array.from(files);\r\n\r\n    if (this.onMultipleFilesSelected) {\r\n      try {\r\n        let result = await this.onMultipleFilesSelected(filesArray);\r\n      } catch (e) {\r\n        console.log('error: ', e);\r\n      }  \r\n    }\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      let 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               [helpPopover]=\"helpPopover\"\r\n               [helpPopoverPosition]=\"helpPopoverPosition\"\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('name').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 || formModel?.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             id=\"{{inputId}}_btn\"\r\n             [pending]=\"pending\"\r\n             [type]=\"buttonType\"\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>"]}
@@ -4135,7 +4135,9 @@ const FileTypeExtensions = {
4135
4135
  };
4136
4136
  class FileUploadComponent extends FormControlBase {
4137
4137
  // static class to create the form group from a parent component
4138
- static getFormModel(validators, disabled = false, fileValidators) {
4138
+ static getFormModel(validators, disabled = false,
4139
+ /** Any validators required to make sure the selected file is valid. It is recommended that you use `FileUploadComponent.getFileValidator` to help construct the validator(s). NOTE: This currently only works when multiSelect is false. */
4140
+ fileValidators) {
4139
4141
  let formGroup = new FormGroup({
4140
4142
  file: new FormControl({ value: null, disabled: disabled }, { validators: validators, asyncValidators: fileValidators }),
4141
4143
  name: new FormControl({ value: null, disabled: disabled }, validators),
@@ -4147,11 +4149,19 @@ class FileUploadComponent extends FormControlBase {
4147
4149
  }
4148
4150
  return formGroup;
4149
4151
  }
4152
+ /**
4153
+ * Helper function that returns an async validator to be used with the file upload control.
4154
+ * This is useful for when the file needs to be validated before it can be uploaded.
4155
+ *
4156
+ * @param callback The callback function that will be called to validate the file. Parameters for the callback are the file and the base64 string for the file.
4157
+ * base64 is null if the fileOutput input on the FileUploadComponent is set to raw. Using fileOutput set to base64 is recommended for images.
4158
+ */
4150
4159
  static getFileValidator(callback) {
4151
4160
  return (control) => __awaiter(this, void 0, void 0, function* () {
4152
4161
  var _a;
4153
4162
  if (control.value && control.parent) {
4154
4163
  let file = control.value;
4164
+ // For images, we need the base64 string to validate image dimensions
4155
4165
  let base64 = (_a = control.parent.get('base64FileString')) === null || _a === void 0 ? void 0 : _a.value;
4156
4166
  return yield callback(file, base64);
4157
4167
  }
@@ -4204,7 +4214,8 @@ class FileUploadComponent extends FormControlBase {
4204
4214
  });
4205
4215
  }
4206
4216
  });
4207
- // Sync errors from the file control to the name control whenever the file control status changes
4217
+ // Sync errors from the file control to the name control whenever the file control status changes.
4218
+ // The name control is the only one displayed to the user so we need to show the errors there.
4208
4219
  (_c = this.formModel.get('file')) === null || _c === void 0 ? void 0 : _c.statusChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(() => {
4209
4220
  var _a, _b, _c;
4210
4221
  const errors = (_b = (_a = this.formModel.get('file')) === null || _a === void 0 ? void 0 : _a.errors) !== null && _b !== void 0 ? _b : null;
@@ -4252,27 +4263,50 @@ class FileUploadComponent extends FormControlBase {
4252
4263
  * @param file
4253
4264
  * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64
4254
4265
  */
4255
- processFile(file, base64FileString) {
4266
+ processFile(file, base64FileString = null) {
4267
+ return __awaiter(this, void 0, void 0, function* () {
4268
+ // If we have async validators on the file control we need to do validation before we trigger the upload
4269
+ const validateBeforeUpload = !!this.formModel.controls.file.asyncValidator;
4270
+ if (validateBeforeUpload) {
4271
+ yield this.validateFile(file, base64FileString);
4272
+ }
4273
+ if (this.onFileSelected) {
4274
+ // Only call the onFileSelected callback to upload the file if the form group is valid
4275
+ if (this.formModel.valid) {
4276
+ try {
4277
+ let result = yield this.onFileSelected(file);
4278
+ // If we did validation, just patch the form result because the file is already in the form
4279
+ if (validateBeforeUpload) {
4280
+ this.formModel.patchValue({ uploadResult: result !== null && result !== void 0 ? result : null });
4281
+ }
4282
+ else {
4283
+ this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: result !== null && result !== void 0 ? result : null });
4284
+ }
4285
+ }
4286
+ catch (e) {
4287
+ // Bummer, we're not going to do anything about it though.
4288
+ // We are not patching any of the result so any existing information remains
4289
+ }
4290
+ }
4291
+ // If we don't have an onFileSelected callback we just patch the form model.
4292
+ // In the case of pre-upload validation the form already has the file so we don't want to patch again.
4293
+ }
4294
+ else if (!validateBeforeUpload) {
4295
+ this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: null });
4296
+ }
4297
+ });
4298
+ }
4299
+ /** Patches the form with the selected file in order to trigger control validation */
4300
+ validateFile(file, base64FileString = null) {
4256
4301
  return __awaiter(this, void 0, void 0, function* () {
4257
4302
  // Patch the file first to trigger any file validators
4258
- this.patchProcessedFile(file, base64FileString);
4303
+ this.formModel.patchValue({ file, name: file.name, base64FileString, uploadResult: null });
4259
4304
  // If we have any async validators pending we need to wait for them to complete before we know if the form is valid
4260
4305
  if (this.formModel.pending) {
4261
4306
  yield this.formModel.statusChanges.pipe(filter(status => status !== 'PENDING'), take(1), takeUntil(this.componentDestroyed)).toPromise();
4262
4307
  }
4263
4308
  // Mark the name control as touched so that any validation errors will show
4264
4309
  this.formModel.controls.name.markAsTouched();
4265
- // Only call the onFileSelected callback to upload the file if the form is valid
4266
- if (this.onFileSelected && this.formModel.valid) {
4267
- try {
4268
- let result = yield this.onFileSelected(file);
4269
- this.formModel.patchValue({ uploadResult: result !== null && result !== void 0 ? result : null });
4270
- }
4271
- catch (e) {
4272
- // Bummer, we're not going to do anything about it though.
4273
- // We are not patching any of the result so any existing information remains
4274
- }
4275
- }
4276
4310
  });
4277
4311
  }
4278
4312
  /**
@@ -4282,22 +4316,6 @@ class FileUploadComponent extends FormControlBase {
4282
4316
  isBase64FileOutput() {
4283
4317
  return this.fileOutput === 'base64';
4284
4318
  }
4285
- /**
4286
- * When the file was selected and processed patch the file information that the hosting form will
4287
- * be looking for.
4288
- * @param file
4289
- * @param base64FileString
4290
- * @param onFileSelectedResult
4291
- */
4292
- patchProcessedFile(file, base64FileString) {
4293
- var _a;
4294
- (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.patchValue({
4295
- file: file,
4296
- name: file === null || file === void 0 ? void 0 : file.name,
4297
- base64FileString: base64FileString !== null && base64FileString !== void 0 ? base64FileString : null,
4298
- uploadResult: null
4299
- });
4300
- }
4301
4319
  /** Maps the files to an array of File objects and sends them along
4302
4320
  * to the derived onMultipleFileSelected method in the hosting component
4303
4321
  */