@energycap/components 0.39.21-ECAP-25650-file-upload-validation-support.20240805-1706 → 0.39.21-ECAP-25650-file-upload-validation-support.20240806-1029
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/controls/file-upload/file-upload.component.mjs +12 -7
- package/esm2020/lib/shared/page/page-base/page-base.component.mjs +5 -1
- package/fesm2015/energycap-components.mjs +18 -8
- package/fesm2015/energycap-components.mjs.map +1 -1
- package/fesm2020/energycap-components.mjs +15 -6
- package/fesm2020/energycap-components.mjs.map +1 -1
- package/package.json +1 -1
@@ -19,8 +19,8 @@ export class FileUploadComponent extends FormControlBase {
|
|
19
19
|
// static class to create the form group from a parent component
|
20
20
|
static getFormModel(validators, disabled = false, fileValidators) {
|
21
21
|
let formGroup = new FormGroup({
|
22
|
-
file: new FormControl({ value: null, disabled: disabled }, validators),
|
23
|
-
name: new FormControl({ value: null, disabled: disabled },
|
22
|
+
file: new FormControl({ value: null, disabled: disabled }, { validators: validators, asyncValidators: fileValidators }),
|
23
|
+
name: new FormControl({ value: null, disabled: disabled }, validators),
|
24
24
|
base64FileString: new FormControl(null),
|
25
25
|
uploadResult: new FormControl(null),
|
26
26
|
});
|
@@ -30,10 +30,10 @@ export class FileUploadComponent extends FormControlBase {
|
|
30
30
|
return formGroup;
|
31
31
|
}
|
32
32
|
static getFileValidator(callback) {
|
33
|
-
return async (
|
34
|
-
if (
|
35
|
-
let file =
|
36
|
-
let base64 =
|
33
|
+
return async (control) => {
|
34
|
+
if (control.value && control.parent) {
|
35
|
+
let file = control.value;
|
36
|
+
let base64 = control.parent.get('base64FileString')?.value;
|
37
37
|
return await callback(file, base64);
|
38
38
|
}
|
39
39
|
return null;
|
@@ -84,6 +84,11 @@ export class FileUploadComponent extends FormControlBase {
|
|
84
84
|
});
|
85
85
|
}
|
86
86
|
});
|
87
|
+
// Sync errors from the file control to the name control whenever the file control status changes
|
88
|
+
this.formModel.get('file')?.statusChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(() => {
|
89
|
+
const errors = this.formModel.get('file')?.errors ?? null;
|
90
|
+
this.formModel.get('name')?.setErrors(errors);
|
91
|
+
});
|
87
92
|
}
|
88
93
|
async fileChange(files) {
|
89
94
|
if (this.multiSelect) {
|
@@ -227,4 +232,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
|
|
227
232
|
type: ViewChild,
|
228
233
|
args: ["fileInput", { read: ElementRef, static: true }]
|
229
234
|
}] } });
|
230
|
-
//# 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    fileValidators?: AsyncValidatorFn | AsyncValidatorFn[]\r\n  ): FormGroup<FileUploadFormGroup> {\r\n    let formGroup = new FormGroup<FileUploadFormGroup>({\r\n      file: new FormControl({ value: null, disabled: disabled }, validators),\r\n      name: new FormControl({ value: null, disabled: disabled }, {validators: validators, asyncValidators: fileValidators}),\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 (nameControl: AbstractControl) => {\r\n      if (nameControl.value && nameControl.parent && nameControl.parent.get('file')?.value) {\r\n        let file = nameControl.parent.get('file')?.value as File;\r\n        let base64 = nameControl.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\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>"]}
|
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":"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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    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>"]}
|
@@ -192,6 +192,8 @@ export class PageBaseComponent {
|
|
192
192
|
* @param event
|
193
193
|
*/
|
194
194
|
async onSave(event) {
|
195
|
+
// Show the saving overlay in case we have async validators in play
|
196
|
+
this.showStatus(PageStatus.Saving);
|
195
197
|
this.formGroup.markAllAsTouched();
|
196
198
|
// If the form has pending async validators, wait for it them to complete before proceeding
|
197
199
|
if (this.formGroup.pending) {
|
@@ -201,6 +203,8 @@ export class PageBaseComponent {
|
|
201
203
|
await this.save();
|
202
204
|
}
|
203
205
|
else {
|
206
|
+
// Remove the saving overlay to display any errors
|
207
|
+
this.showStatus(PageStatus.Loaded);
|
204
208
|
// Only show the banner with the generic message if the parent component hasn't already set one
|
205
209
|
// by implementing onSave for additional validation
|
206
210
|
if (this.errors === '') {
|
@@ -380,4 +384,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
|
|
380
384
|
type: HostBinding,
|
381
385
|
args: ['class']
|
382
386
|
}] } });
|
383
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-base.component.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/shared/page/page-base/page-base.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Directive, HostBinding, Injector, OnDestroy, OnInit } from '@angular/core';\r\nimport { UntypedFormGroup } from '@angular/forms';\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { TranslateService } from '@ngx-translate/core';\r\nimport { Observable, ReplaySubject, Subject, Subscription } from 'rxjs';\r\nimport { filter, take, takeUntil } from 'rxjs/operators';\r\nimport { NavItem } from '../../../controls/navigation/nav-item';\r\nimport { ErrorService } from '../../../core/error.service';\r\nimport { RouterHelper } from '../../../core/router-helper.service';\r\nimport { ScrollService } from '../../../core/scroll.service';\r\nimport { DialogService } from '../../../display/dialog/dialog.service';\r\nimport { Help } from '../../../display/help/help-types';\r\nimport { SplashService } from '../../../display/splash/splash.service';\r\nimport { Overlay } from '../../../display/view-overlay/view-overlay.component';\r\nimport { PageStatus, PageStatuses } from '../page-statuses';\r\n\r\nexport class PageInitResult {\r\n  title?: string = '';\r\n  subTitle?: string = '';\r\n  subTitleUrl?: string = '';\r\n  breadcrumbs?: NavItem[];\r\n  help?: Help;\r\n}\r\n\r\n@Directive()\r\nexport abstract class PageBaseComponent implements OnInit, AfterViewInit, OnDestroy {\r\n  /** Default css classes applied to host element */\r\n  @HostBinding('class') protected classList = 'd-flex flex-grow';\r\n\r\n  /**\r\n   * Title for the page displayed in the header\r\n   */\r\n  public title?: string = '';\r\n\r\n  /**\r\n   * Sub title for the page displayed in the header\r\n   */\r\n  public subTitle?: string = '';\r\n\r\n  /**\r\n   * Sub title url for the page displayed in the header\r\n   */\r\n  public subTitleUrl?: string = '';\r\n\r\n  /**\r\n   * Status overlay for the page\r\n   */\r\n  public status: Overlay = new Overlay('pending', 'Loading');\r\n\r\n  /**\r\n   * Breadcrumbs displayed in the header\r\n   */\r\n  public breadcrumbs?: NavItem[];\r\n\r\n  /**\r\n   * Result returned from the onInit function\r\n   */\r\n  public initResult!: PageInitResult;\r\n\r\n  /**\r\n   * Help information for the component, which can only be set through onInit\r\n   */\r\n  public help: Help | undefined;\r\n\r\n  /**\r\n   * Form group to be used if needed by the page\r\n   */\r\n  public formGroup: UntypedFormGroup = new UntypedFormGroup({});\r\n\r\n  /**\r\n   * Default invalid form message, override if a more specific message is needed\r\n   */\r\n  public defaultInvalidFormMessage: string = 'PleaseCorrectForm_SC';\r\n\r\n  /**\r\n   * Default unknown save error message, override if a more specific message is needed\r\n   */\r\n  public defaultUnknownSaveError: string = 'PageBaseUnknownSaveError_SC';\r\n\r\n  /**\r\n   * Used to opt out of hiding the splash screen when the page is loaded\r\n   */\r\n  public skipHideSplashOnComplete: boolean = false;\r\n\r\n  /**\r\n   * When set to true any navigation end events that are strictly query param changes will\r\n   * not call the onNavigationEnd function. This allows pages to change the view with query\r\n   * params and not trigger a full refresh.\r\n   */\r\n  public ignoreQueryParamsOnNavigationChange: boolean = false;\r\n\r\n  private previousUrl: string = \"\";\r\n\r\n  /**\r\n   * Used to unsubscribe from observables\r\n   */\r\n  public destroyed: Subject<any> = new Subject<any>();\r\n\r\n  /**\r\n   * Used to know when the page has been loaded, this will be completed\r\n   * after onAfterViewInit\r\n   */\r\n  private _afterViewInitComplete: ReplaySubject<void> = new ReplaySubject();\r\n  public afterViewInitComplete: Observable<void> = this._afterViewInitComplete;\r\n\r\n  /**\r\n   * If defined the form will automatically scroll to the first invalid form control\r\n   * if it's outside of the view\r\n   */\r\n  protected scrollContainerSelector: string | undefined;\r\n\r\n  /**\r\n   * Subscription to the router events NavigationEnd\r\n   * Used to know if already subscribed to\r\n   */\r\n  private navigationEndSubscription?: Subscription;\r\n\r\n  /**\r\n   * Common services used by page-base\r\n   */\r\n  protected errorService: ErrorService;\r\n  protected translateService: TranslateService;\r\n  protected scrollService: ScrollService;\r\n  protected router: Router;\r\n  protected activatedRoute: ActivatedRoute;\r\n  protected dialogService: DialogService;\r\n  protected splashService: SplashService;\r\n  protected routerHelper: RouterHelper;\r\n\r\n  constructor(injector: Injector) {\r\n    this.errorService = injector.get(ErrorService);\r\n    this.translateService = injector.get(TranslateService);\r\n    this.scrollService = injector.get(ScrollService);\r\n    this.router = injector.get(Router);\r\n    this.activatedRoute = injector.get(ActivatedRoute);\r\n    this.dialogService = injector.get(DialogService);\r\n    this.splashService = injector.get(SplashService);\r\n    this.routerHelper = injector.get(RouterHelper);\r\n  }\r\n\r\n  /**\r\n   * Errors to be shown for the component.\r\n   * Automatically populated on promise rejection from save and shared with the UI.\r\n   */\r\n  private _errors: string = '';\r\n  public get errors(): string {\r\n    return this._errors;\r\n  }\r\n\r\n  /**\r\n   * Optional lifecycle hook that is called as part of the Angular ngOnInit lifecycle hook.\r\n   * Called before onInit is called for the first time. Add setup logic here that does\r\n   * not need to be run on every reload.\r\n   */\r\n  protected onFirstInit(): void { }\r\n\r\n  /**\r\n   * This method is called in the derived class as part of the Angular ngOnInit\r\n   * lifecycle hook. It must provide a PageInitResult back to the base class to properly\r\n   * set up the page.\r\n   */\r\n  protected abstract onInit(): Promise<PageInitResult>;\r\n  protected initPromise!: Promise<PageInitResult>;\r\n\r\n  public async ngOnInit() {\r\n    try {\r\n      this.showStatus(PageStatus.Loading);\r\n      this.setUpRouterSubscriptions();\r\n      this.onFirstInit();\r\n      this.initPromise = this.onInit();\r\n\r\n      await this.initPromise;\r\n    } catch (error) {\r\n      this.splashService.hideSplash();\r\n      this.onError(error);\r\n    };\r\n  }\r\n\r\n  /**\r\n   * This method is called just after the Angular Lifecycle event ngOnAfterViewInit.\r\n   * It fires after all sub-components have gone through their initialization phase.\r\n   * It is in this phase that the view components should already exist, so a derived class should\r\n   * load data from the API to populate any view components. The promise that\r\n   * is returned must not resolve until all required data has completely loaded\r\n   */\r\n  protected async onAfterViewInit(): Promise<void> { };\r\n  public async ngAfterViewInit(): Promise<void> {\r\n    try {\r\n      const initResult = await this.initPromise;\r\n      this.parseInitResult(initResult);\r\n\r\n      await this.onAfterViewInit();\r\n\r\n      this._afterViewInitComplete.next();\r\n      this._afterViewInitComplete.complete();\r\n\r\n      await this.onFormLoadComplete();\r\n\r\n      if (this.status.status === 'pending') {\r\n        this.showStatus(PageStatus.Loaded);\r\n      }\r\n\r\n      if (!this.skipHideSplashOnComplete) {\r\n        this.splashService.hideSplash();\r\n      }\r\n\r\n    } catch (error) {\r\n      this.splashService.hideSplash();\r\n      this.onError(error);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * A lifecycle method that executes once all data has been loaded\r\n   * through ngAfterViewInit and from local storage if required.\r\n   * It is at this point that you can subscribe to changes on the form\r\n   * reliably\r\n   *\r\n   * @protected\r\n   * @memberof PageBaseComponent\r\n   */\r\n  protected async onFormLoadComplete(): Promise<any> { }\r\n\r\n  /**\r\n   * A lifecycle hook that is called after every NavigationEnd event.\r\n   */\r\n  protected onNavigationEnd(): void { }\r\n\r\n  /**\r\n   * Reload the component kicking off the ngOnInit and ngAfterViewInit lifecycle hooks.\r\n   * Any state that needs to be reset should be done so before calling this method.\r\n   */\r\n  public reload(): void {\r\n    this._afterViewInitComplete = new ReplaySubject();\r\n\r\n    this.showStatus(PageStatus.Loading);\r\n    this.initPromise = this.onInit();\r\n    this.ngAfterViewInit();\r\n  }\r\n\r\n  /**\r\n   * When overridden in a derived class this method will sync the form state\r\n   * into the object that will be persisted to the API. The value should not\r\n   * actually be sent to the API yet.\r\n   */\r\n  protected async applyModelUpdates(): Promise<void> { };\r\n\r\n  /**\r\n   * Save the model state to the API and return a promise that\r\n   * contains the updated entityID. The promise should be rejected\r\n   * if the save fails and errors and the overlay status will be updated\r\n   * accordingly\r\n   */\r\n  protected async saveToApi(): Promise<number | void> { };\r\n\r\n  /**\r\n   * Called after the save has completed successfully.\r\n   * It is the responsibility of the derived class to navigate\r\n   * to a properties/status page or emit appropriately if within a dialog\r\n   * @param entityId\r\n   */\r\n  protected onSaveComplete(entityId?: number): void { };\r\n\r\n  /**\r\n   * Function called when the user clicks save.  We will first touch all controls to show validation messages\r\n   * if the form is invalid.  If the form is valid we will attempt to save and any errors will be reported\r\n   * back to the user\r\n   * @param event\r\n   */\r\n  public async onSave(event: Event): Promise<void> {\r\n    this.formGroup.markAllAsTouched();\r\n\r\n    // If the form has pending async validators, wait for it them to complete before proceeding\r\n    if (this.formGroup.pending) {\r\n      await this.formGroup.statusChanges.pipe(\r\n        filter(status => status !== 'PENDING'),\r\n        take(1),\r\n        takeUntil(this.destroyed)\r\n      ).toPromise();\r\n    }\r\n\r\n    if (this.formGroup.valid) {\r\n      await this.save();\r\n    } else {\r\n      // Only show the banner with the generic message if the parent component hasn't already set one\r\n      // by implementing onSave for additional validation\r\n      if (this.errors === '') {\r\n        this.showErrorBanner(this.defaultInvalidFormMessage);\r\n      }\r\n\r\n      if (this.scrollContainerSelector) {\r\n        setTimeout(() => {\r\n          this.scrollService.scrollToItem(this.scrollContainerSelector!, \".ng-invalid\");\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Function available to trigger a save and bypass the form validation checks.\r\n   * This is useful if the user chooses a remove action from within an edit dialog. If the dialog form is invalid\r\n   * that is no concern to the user if they wish to remove the item they are editing.\r\n   * @param event\r\n   */\r\n  public async onSaveWithoutValidate(event: Event): Promise<void> {\r\n    await this.save();\r\n  }\r\n\r\n  /**\r\n   * Function called when the component is destroyed\r\n   */\r\n  public ngOnDestroy(): void {\r\n    this.destroyed.next();\r\n    this.destroyed.unsubscribe();\r\n  }\r\n\r\n  /**\r\n   * Function that will log an error that is passed in to the console\r\n   * @param e\r\n   */\r\n  protected onError(error: Error): void {\r\n    this.errorService.logConsoleError(error);\r\n    this.showErrorOverlay(error);\r\n  }\r\n\r\n  /**\r\n   * Show an error message in a whole-page overlay. This will prevent any more\r\n   * interactions with the page. It can take a string error message or a raw API\r\n   * error that needs to be parsed. If the API error isn't something we can handle\r\n   * a generic message is shown instead\r\n   * @param {(Error | string)} error\r\n   */\r\n  protected showErrorOverlay(error: Error | string) {\r\n    this.getErrorMessage(error).subscribe((translatedMessage: string) => {\r\n      this.status.setStatus('error', translatedMessage);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Show an error in the banner at the top of the screen.  This will not produce a\r\n   * whole-page overlay like the function showError will.  It can take a string error message\r\n   * or raw API error that needs to be parsed.  If the API error isn't something we can handle,\r\n   * a generic message (customizable via unknownErrorDefault) is shown instead.\r\n   * @param error\r\n   * @param unknownErrorDefault\r\n   */\r\n  protected showErrorBanner(error: Error | string, unknownErrorDefault?: string): void {\r\n    this.getErrorMessage(error, unknownErrorDefault).subscribe((translatedMessage: string) => {\r\n      this._errors = translatedMessage;\r\n      this.showStatus(PageStatus.Loaded);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Parses and translates an API error or error string.\r\n   * @param e\r\n   * @param unknownErrorDefault\r\n   */\r\n  protected getErrorMessage(e: Error | string, unknownErrorDefault?: string): Observable<string> {\r\n    let extractedMessage = this.errorService.parseApiError(e, unknownErrorDefault);\r\n    return this.translateService.get(extractedMessage);\r\n  }\r\n\r\n  /**\r\n   * Clear the errors to hide the errors banner.\r\n   */\r\n  protected hideErrorBanner(): void {\r\n    this._errors = '';\r\n  }\r\n\r\n  /**\r\n   * Change the display status to overlay the whole page while loading or saving\r\n   * and clear it when the form is ready\r\n   * @protected\r\n   * @param {PageStatus} status\r\n   * @memberof FormBaseComponent\r\n   */\r\n  protected showStatus(status: PageStatus) {\r\n    let knownStatus = PageStatuses[status];\r\n    if (knownStatus) {\r\n      this.status.setStatus(knownStatus.status, knownStatus.message, undefined, knownStatus.noDataTemplate, knownStatus.overlayClassList);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Allows the component to enter a pending status and execute a function that on completion will return the status\r\n   * to hasData. If an error occurs within the callback an error banner can be shown.\r\n   * @param callback\r\n   * @param pendingMessage - optional custom pending message\r\n   * @returns\r\n   */\r\n  protected async runBlockingAction(callback: () => Promise<any>, pendingMessage?: string): Promise<any> {\r\n    this.hideErrorBanner();\r\n    this.status.setStatus('pending', pendingMessage);\r\n\r\n    return callback().catch(error => {\r\n      this.showErrorBanner(error);\r\n      // Throw the error for any downstream components that may be listening and need to react\r\n      // to any errors happening within this callback\r\n      throw (error);\r\n    }).finally(() => {\r\n      this.showStatus(PageStatus.Loaded);\r\n    });\r\n  };\r\n\r\n  /**\r\n   * Parses the page init result and sets the necessary properties on the page\r\n   * @param result\r\n   */\r\n  private parseInitResult(result: PageInitResult): void {\r\n    this.initResult = result;\r\n    this.title = result.title;\r\n    this.subTitle = result.subTitle;\r\n    this.subTitleUrl = result.subTitleUrl;\r\n    this.breadcrumbs = result.breadcrumbs;\r\n    this.help = result.help;\r\n  }\r\n\r\n  /**\r\n   * Controls the page status, triggers the save action and handles errors thrown.\r\n   */\r\n  private async save(): Promise<void> {\r\n    this.showStatus(PageStatus.Saving);\r\n\r\n    try {\r\n      await this.applyModelUpdates();\r\n      let entityId = await this.saveToApi() || -1;\r\n\r\n      this._errors = '';\r\n\r\n      this.onSaveComplete(entityId);\r\n    } catch (error) {\r\n      //clear the \"saving\" overlay and show the message at the top of the view through the errors property\r\n      this.showErrorBanner(error, this.defaultUnknownSaveError);\r\n      this.splashService.hideSplash();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Sets up the router event subscriptions\r\n   */\r\n  private setUpRouterSubscriptions(): void {\r\n    if (!this.navigationEndSubscription && !this.destroyed.closed) {\r\n      this.previousUrl = this.router.url;\r\n\r\n      // React to navigation events\r\n      this.navigationEndSubscription = this.router.events.pipe(\r\n        takeUntil(this.destroyed),\r\n        filter(e => e instanceof NavigationEnd),\r\n      ).subscribe(e => {\r\n        try {\r\n          let currentUrl: string = (<NavigationEnd>e).url;\r\n\r\n          if (this.ignoreQueryParamsOnNavigationChange) {\r\n            if (this.previousUrl.split('?')[0] !== currentUrl.split('?')[0]) {\r\n              this.navigationEndHandler((<NavigationEnd>e).url);\r\n            }\r\n          } else {\r\n            if (this.previousUrl !== currentUrl) {\r\n              this.navigationEndHandler((<NavigationEnd>e).url);\r\n            }\r\n          }\r\n        } catch (error) {\r\n          this.onError(error);\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  private navigationEndHandler(url: string): void {\r\n    this.previousUrl = url;\r\n    this.onNavigationEnd();\r\n  }\r\n}\r\n"]}
|
387
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-base.component.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/shared/page/page-base/page-base.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Directive, HostBinding, Injector, OnDestroy, OnInit } from '@angular/core';\r\nimport { UntypedFormGroup } from '@angular/forms';\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { TranslateService } from '@ngx-translate/core';\r\nimport { Observable, ReplaySubject, Subject, Subscription } from 'rxjs';\r\nimport { filter, take, takeUntil } from 'rxjs/operators';\r\nimport { NavItem } from '../../../controls/navigation/nav-item';\r\nimport { ErrorService } from '../../../core/error.service';\r\nimport { RouterHelper } from '../../../core/router-helper.service';\r\nimport { ScrollService } from '../../../core/scroll.service';\r\nimport { DialogService } from '../../../display/dialog/dialog.service';\r\nimport { Help } from '../../../display/help/help-types';\r\nimport { SplashService } from '../../../display/splash/splash.service';\r\nimport { Overlay } from '../../../display/view-overlay/view-overlay.component';\r\nimport { PageStatus, PageStatuses } from '../page-statuses';\r\n\r\nexport class PageInitResult {\r\n  title?: string = '';\r\n  subTitle?: string = '';\r\n  subTitleUrl?: string = '';\r\n  breadcrumbs?: NavItem[];\r\n  help?: Help;\r\n}\r\n\r\n@Directive()\r\nexport abstract class PageBaseComponent implements OnInit, AfterViewInit, OnDestroy {\r\n  /** Default css classes applied to host element */\r\n  @HostBinding('class') protected classList = 'd-flex flex-grow';\r\n\r\n  /**\r\n   * Title for the page displayed in the header\r\n   */\r\n  public title?: string = '';\r\n\r\n  /**\r\n   * Sub title for the page displayed in the header\r\n   */\r\n  public subTitle?: string = '';\r\n\r\n  /**\r\n   * Sub title url for the page displayed in the header\r\n   */\r\n  public subTitleUrl?: string = '';\r\n\r\n  /**\r\n   * Status overlay for the page\r\n   */\r\n  public status: Overlay = new Overlay('pending', 'Loading');\r\n\r\n  /**\r\n   * Breadcrumbs displayed in the header\r\n   */\r\n  public breadcrumbs?: NavItem[];\r\n\r\n  /**\r\n   * Result returned from the onInit function\r\n   */\r\n  public initResult!: PageInitResult;\r\n\r\n  /**\r\n   * Help information for the component, which can only be set through onInit\r\n   */\r\n  public help: Help | undefined;\r\n\r\n  /**\r\n   * Form group to be used if needed by the page\r\n   */\r\n  public formGroup: UntypedFormGroup = new UntypedFormGroup({});\r\n\r\n  /**\r\n   * Default invalid form message, override if a more specific message is needed\r\n   */\r\n  public defaultInvalidFormMessage: string = 'PleaseCorrectForm_SC';\r\n\r\n  /**\r\n   * Default unknown save error message, override if a more specific message is needed\r\n   */\r\n  public defaultUnknownSaveError: string = 'PageBaseUnknownSaveError_SC';\r\n\r\n  /**\r\n   * Used to opt out of hiding the splash screen when the page is loaded\r\n   */\r\n  public skipHideSplashOnComplete: boolean = false;\r\n\r\n  /**\r\n   * When set to true any navigation end events that are strictly query param changes will\r\n   * not call the onNavigationEnd function. This allows pages to change the view with query\r\n   * params and not trigger a full refresh.\r\n   */\r\n  public ignoreQueryParamsOnNavigationChange: boolean = false;\r\n\r\n  private previousUrl: string = \"\";\r\n\r\n  /**\r\n   * Used to unsubscribe from observables\r\n   */\r\n  public destroyed: Subject<any> = new Subject<any>();\r\n\r\n  /**\r\n   * Used to know when the page has been loaded, this will be completed\r\n   * after onAfterViewInit\r\n   */\r\n  private _afterViewInitComplete: ReplaySubject<void> = new ReplaySubject();\r\n  public afterViewInitComplete: Observable<void> = this._afterViewInitComplete;\r\n\r\n  /**\r\n   * If defined the form will automatically scroll to the first invalid form control\r\n   * if it's outside of the view\r\n   */\r\n  protected scrollContainerSelector: string | undefined;\r\n\r\n  /**\r\n   * Subscription to the router events NavigationEnd\r\n   * Used to know if already subscribed to\r\n   */\r\n  private navigationEndSubscription?: Subscription;\r\n\r\n  /**\r\n   * Common services used by page-base\r\n   */\r\n  protected errorService: ErrorService;\r\n  protected translateService: TranslateService;\r\n  protected scrollService: ScrollService;\r\n  protected router: Router;\r\n  protected activatedRoute: ActivatedRoute;\r\n  protected dialogService: DialogService;\r\n  protected splashService: SplashService;\r\n  protected routerHelper: RouterHelper;\r\n\r\n  constructor(injector: Injector) {\r\n    this.errorService = injector.get(ErrorService);\r\n    this.translateService = injector.get(TranslateService);\r\n    this.scrollService = injector.get(ScrollService);\r\n    this.router = injector.get(Router);\r\n    this.activatedRoute = injector.get(ActivatedRoute);\r\n    this.dialogService = injector.get(DialogService);\r\n    this.splashService = injector.get(SplashService);\r\n    this.routerHelper = injector.get(RouterHelper);\r\n  }\r\n\r\n  /**\r\n   * Errors to be shown for the component.\r\n   * Automatically populated on promise rejection from save and shared with the UI.\r\n   */\r\n  private _errors: string = '';\r\n  public get errors(): string {\r\n    return this._errors;\r\n  }\r\n\r\n  /**\r\n   * Optional lifecycle hook that is called as part of the Angular ngOnInit lifecycle hook.\r\n   * Called before onInit is called for the first time. Add setup logic here that does\r\n   * not need to be run on every reload.\r\n   */\r\n  protected onFirstInit(): void { }\r\n\r\n  /**\r\n   * This method is called in the derived class as part of the Angular ngOnInit\r\n   * lifecycle hook. It must provide a PageInitResult back to the base class to properly\r\n   * set up the page.\r\n   */\r\n  protected abstract onInit(): Promise<PageInitResult>;\r\n  protected initPromise!: Promise<PageInitResult>;\r\n\r\n  public async ngOnInit() {\r\n    try {\r\n      this.showStatus(PageStatus.Loading);\r\n      this.setUpRouterSubscriptions();\r\n      this.onFirstInit();\r\n      this.initPromise = this.onInit();\r\n\r\n      await this.initPromise;\r\n    } catch (error) {\r\n      this.splashService.hideSplash();\r\n      this.onError(error);\r\n    };\r\n  }\r\n\r\n  /**\r\n   * This method is called just after the Angular Lifecycle event ngOnAfterViewInit.\r\n   * It fires after all sub-components have gone through their initialization phase.\r\n   * It is in this phase that the view components should already exist, so a derived class should\r\n   * load data from the API to populate any view components. The promise that\r\n   * is returned must not resolve until all required data has completely loaded\r\n   */\r\n  protected async onAfterViewInit(): Promise<void> { };\r\n  public async ngAfterViewInit(): Promise<void> {\r\n    try {\r\n      const initResult = await this.initPromise;\r\n      this.parseInitResult(initResult);\r\n\r\n      await this.onAfterViewInit();\r\n\r\n      this._afterViewInitComplete.next();\r\n      this._afterViewInitComplete.complete();\r\n\r\n      await this.onFormLoadComplete();\r\n\r\n      if (this.status.status === 'pending') {\r\n        this.showStatus(PageStatus.Loaded);\r\n      }\r\n\r\n      if (!this.skipHideSplashOnComplete) {\r\n        this.splashService.hideSplash();\r\n      }\r\n\r\n    } catch (error) {\r\n      this.splashService.hideSplash();\r\n      this.onError(error);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * A lifecycle method that executes once all data has been loaded\r\n   * through ngAfterViewInit and from local storage if required.\r\n   * It is at this point that you can subscribe to changes on the form\r\n   * reliably\r\n   *\r\n   * @protected\r\n   * @memberof PageBaseComponent\r\n   */\r\n  protected async onFormLoadComplete(): Promise<any> { }\r\n\r\n  /**\r\n   * A lifecycle hook that is called after every NavigationEnd event.\r\n   */\r\n  protected onNavigationEnd(): void { }\r\n\r\n  /**\r\n   * Reload the component kicking off the ngOnInit and ngAfterViewInit lifecycle hooks.\r\n   * Any state that needs to be reset should be done so before calling this method.\r\n   */\r\n  public reload(): void {\r\n    this._afterViewInitComplete = new ReplaySubject();\r\n\r\n    this.showStatus(PageStatus.Loading);\r\n    this.initPromise = this.onInit();\r\n    this.ngAfterViewInit();\r\n  }\r\n\r\n  /**\r\n   * When overridden in a derived class this method will sync the form state\r\n   * into the object that will be persisted to the API. The value should not\r\n   * actually be sent to the API yet.\r\n   */\r\n  protected async applyModelUpdates(): Promise<void> { };\r\n\r\n  /**\r\n   * Save the model state to the API and return a promise that\r\n   * contains the updated entityID. The promise should be rejected\r\n   * if the save fails and errors and the overlay status will be updated\r\n   * accordingly\r\n   */\r\n  protected async saveToApi(): Promise<number | void> { };\r\n\r\n  /**\r\n   * Called after the save has completed successfully.\r\n   * It is the responsibility of the derived class to navigate\r\n   * to a properties/status page or emit appropriately if within a dialog\r\n   * @param entityId\r\n   */\r\n  protected onSaveComplete(entityId?: number): void { };\r\n\r\n  /**\r\n   * Function called when the user clicks save.  We will first touch all controls to show validation messages\r\n   * if the form is invalid.  If the form is valid we will attempt to save and any errors will be reported\r\n   * back to the user\r\n   * @param event\r\n   */\r\n  public async onSave(event: Event): Promise<void> {\r\n    // Show the saving overlay in case we have async validators in play\r\n    this.showStatus(PageStatus.Saving);\r\n    \r\n    this.formGroup.markAllAsTouched();\r\n\r\n    // If the form has pending async validators, wait for it them to complete before proceeding\r\n    if (this.formGroup.pending) {\r\n      await this.formGroup.statusChanges.pipe(\r\n        filter(status => status !== 'PENDING'),\r\n        take(1),\r\n        takeUntil(this.destroyed)\r\n      ).toPromise();\r\n    }\r\n\r\n    if (this.formGroup.valid) {\r\n      await this.save();\r\n    } else {\r\n      // Remove the saving overlay to display any errors\r\n      this.showStatus(PageStatus.Loaded);\r\n      \r\n      // Only show the banner with the generic message if the parent component hasn't already set one\r\n      // by implementing onSave for additional validation\r\n      if (this.errors === '') {\r\n        this.showErrorBanner(this.defaultInvalidFormMessage);\r\n      }\r\n\r\n      if (this.scrollContainerSelector) {\r\n        setTimeout(() => {\r\n          this.scrollService.scrollToItem(this.scrollContainerSelector!, \".ng-invalid\");\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Function available to trigger a save and bypass the form validation checks.\r\n   * This is useful if the user chooses a remove action from within an edit dialog. If the dialog form is invalid\r\n   * that is no concern to the user if they wish to remove the item they are editing.\r\n   * @param event\r\n   */\r\n  public async onSaveWithoutValidate(event: Event): Promise<void> {\r\n    await this.save();\r\n  }\r\n\r\n  /**\r\n   * Function called when the component is destroyed\r\n   */\r\n  public ngOnDestroy(): void {\r\n    this.destroyed.next();\r\n    this.destroyed.unsubscribe();\r\n  }\r\n\r\n  /**\r\n   * Function that will log an error that is passed in to the console\r\n   * @param e\r\n   */\r\n  protected onError(error: Error): void {\r\n    this.errorService.logConsoleError(error);\r\n    this.showErrorOverlay(error);\r\n  }\r\n\r\n  /**\r\n   * Show an error message in a whole-page overlay. This will prevent any more\r\n   * interactions with the page. It can take a string error message or a raw API\r\n   * error that needs to be parsed. If the API error isn't something we can handle\r\n   * a generic message is shown instead\r\n   * @param {(Error | string)} error\r\n   */\r\n  protected showErrorOverlay(error: Error | string) {\r\n    this.getErrorMessage(error).subscribe((translatedMessage: string) => {\r\n      this.status.setStatus('error', translatedMessage);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Show an error in the banner at the top of the screen.  This will not produce a\r\n   * whole-page overlay like the function showError will.  It can take a string error message\r\n   * or raw API error that needs to be parsed.  If the API error isn't something we can handle,\r\n   * a generic message (customizable via unknownErrorDefault) is shown instead.\r\n   * @param error\r\n   * @param unknownErrorDefault\r\n   */\r\n  protected showErrorBanner(error: Error | string, unknownErrorDefault?: string): void {\r\n    this.getErrorMessage(error, unknownErrorDefault).subscribe((translatedMessage: string) => {\r\n      this._errors = translatedMessage;\r\n      this.showStatus(PageStatus.Loaded);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Parses and translates an API error or error string.\r\n   * @param e\r\n   * @param unknownErrorDefault\r\n   */\r\n  protected getErrorMessage(e: Error | string, unknownErrorDefault?: string): Observable<string> {\r\n    let extractedMessage = this.errorService.parseApiError(e, unknownErrorDefault);\r\n    return this.translateService.get(extractedMessage);\r\n  }\r\n\r\n  /**\r\n   * Clear the errors to hide the errors banner.\r\n   */\r\n  protected hideErrorBanner(): void {\r\n    this._errors = '';\r\n  }\r\n\r\n  /**\r\n   * Change the display status to overlay the whole page while loading or saving\r\n   * and clear it when the form is ready\r\n   * @protected\r\n   * @param {PageStatus} status\r\n   * @memberof FormBaseComponent\r\n   */\r\n  protected showStatus(status: PageStatus) {\r\n    let knownStatus = PageStatuses[status];\r\n    if (knownStatus) {\r\n      this.status.setStatus(knownStatus.status, knownStatus.message, undefined, knownStatus.noDataTemplate, knownStatus.overlayClassList);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Allows the component to enter a pending status and execute a function that on completion will return the status\r\n   * to hasData. If an error occurs within the callback an error banner can be shown.\r\n   * @param callback\r\n   * @param pendingMessage - optional custom pending message\r\n   * @returns\r\n   */\r\n  protected async runBlockingAction(callback: () => Promise<any>, pendingMessage?: string): Promise<any> {\r\n    this.hideErrorBanner();\r\n    this.status.setStatus('pending', pendingMessage);\r\n\r\n    return callback().catch(error => {\r\n      this.showErrorBanner(error);\r\n      // Throw the error for any downstream components that may be listening and need to react\r\n      // to any errors happening within this callback\r\n      throw (error);\r\n    }).finally(() => {\r\n      this.showStatus(PageStatus.Loaded);\r\n    });\r\n  };\r\n\r\n  /**\r\n   * Parses the page init result and sets the necessary properties on the page\r\n   * @param result\r\n   */\r\n  private parseInitResult(result: PageInitResult): void {\r\n    this.initResult = result;\r\n    this.title = result.title;\r\n    this.subTitle = result.subTitle;\r\n    this.subTitleUrl = result.subTitleUrl;\r\n    this.breadcrumbs = result.breadcrumbs;\r\n    this.help = result.help;\r\n  }\r\n\r\n  /**\r\n   * Controls the page status, triggers the save action and handles errors thrown.\r\n   */\r\n  private async save(): Promise<void> {\r\n    this.showStatus(PageStatus.Saving);\r\n\r\n    try {\r\n      await this.applyModelUpdates();\r\n      let entityId = await this.saveToApi() || -1;\r\n\r\n      this._errors = '';\r\n\r\n      this.onSaveComplete(entityId);\r\n    } catch (error) {\r\n      //clear the \"saving\" overlay and show the message at the top of the view through the errors property\r\n      this.showErrorBanner(error, this.defaultUnknownSaveError);\r\n      this.splashService.hideSplash();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Sets up the router event subscriptions\r\n   */\r\n  private setUpRouterSubscriptions(): void {\r\n    if (!this.navigationEndSubscription && !this.destroyed.closed) {\r\n      this.previousUrl = this.router.url;\r\n\r\n      // React to navigation events\r\n      this.navigationEndSubscription = this.router.events.pipe(\r\n        takeUntil(this.destroyed),\r\n        filter(e => e instanceof NavigationEnd),\r\n      ).subscribe(e => {\r\n        try {\r\n          let currentUrl: string = (<NavigationEnd>e).url;\r\n\r\n          if (this.ignoreQueryParamsOnNavigationChange) {\r\n            if (this.previousUrl.split('?')[0] !== currentUrl.split('?')[0]) {\r\n              this.navigationEndHandler((<NavigationEnd>e).url);\r\n            }\r\n          } else {\r\n            if (this.previousUrl !== currentUrl) {\r\n              this.navigationEndHandler((<NavigationEnd>e).url);\r\n            }\r\n          }\r\n        } catch (error) {\r\n          this.onError(error);\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  private navigationEndHandler(url: string): void {\r\n    this.previousUrl = url;\r\n    this.onNavigationEnd();\r\n  }\r\n}\r\n"]}
|
@@ -4137,8 +4137,8 @@ class FileUploadComponent extends FormControlBase {
|
|
4137
4137
|
// static class to create the form group from a parent component
|
4138
4138
|
static getFormModel(validators, disabled = false, fileValidators) {
|
4139
4139
|
let formGroup = new FormGroup({
|
4140
|
-
file: new FormControl({ value: null, disabled: disabled }, validators),
|
4141
|
-
name: new FormControl({ value: null, disabled: disabled },
|
4140
|
+
file: new FormControl({ value: null, disabled: disabled }, { validators: validators, asyncValidators: fileValidators }),
|
4141
|
+
name: new FormControl({ value: null, disabled: disabled }, validators),
|
4142
4142
|
base64FileString: new FormControl(null),
|
4143
4143
|
uploadResult: new FormControl(null),
|
4144
4144
|
});
|
@@ -4148,11 +4148,11 @@ class FileUploadComponent extends FormControlBase {
|
|
4148
4148
|
return formGroup;
|
4149
4149
|
}
|
4150
4150
|
static getFileValidator(callback) {
|
4151
|
-
return (
|
4152
|
-
var _a
|
4153
|
-
if (
|
4154
|
-
let file =
|
4155
|
-
let base64 = (
|
4151
|
+
return (control) => __awaiter(this, void 0, void 0, function* () {
|
4152
|
+
var _a;
|
4153
|
+
if (control.value && control.parent) {
|
4154
|
+
let file = control.value;
|
4155
|
+
let base64 = (_a = control.parent.get('base64FileString')) === null || _a === void 0 ? void 0 : _a.value;
|
4156
4156
|
return yield callback(file, base64);
|
4157
4157
|
}
|
4158
4158
|
return null;
|
@@ -4190,7 +4190,7 @@ class FileUploadComponent extends FormControlBase {
|
|
4190
4190
|
this.updateFileTypeAccept();
|
4191
4191
|
}
|
4192
4192
|
ngOnInit() {
|
4193
|
-
var _a, _b;
|
4193
|
+
var _a, _b, _c;
|
4194
4194
|
super.ngOnInit();
|
4195
4195
|
// Watch for name to change, if the value is cleared we will clear the other
|
4196
4196
|
// supporting model properties. The name can be cleared by the user manually or via
|
@@ -4204,6 +4204,12 @@ class FileUploadComponent extends FormControlBase {
|
|
4204
4204
|
});
|
4205
4205
|
}
|
4206
4206
|
});
|
4207
|
+
// Sync errors from the file control to the name control whenever the file control status changes
|
4208
|
+
(_c = this.formModel.get('file')) === null || _c === void 0 ? void 0 : _c.statusChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(() => {
|
4209
|
+
var _a, _b, _c;
|
4210
|
+
const errors = (_b = (_a = this.formModel.get('file')) === null || _a === void 0 ? void 0 : _a.errors) !== null && _b !== void 0 ? _b : null;
|
4211
|
+
(_c = this.formModel.get('name')) === null || _c === void 0 ? void 0 : _c.setErrors(errors);
|
4212
|
+
});
|
4207
4213
|
}
|
4208
4214
|
fileChange(files) {
|
4209
4215
|
return __awaiter(this, void 0, void 0, function* () {
|
@@ -10711,6 +10717,8 @@ class PageBaseComponent {
|
|
10711
10717
|
*/
|
10712
10718
|
onSave(event) {
|
10713
10719
|
return __awaiter(this, void 0, void 0, function* () {
|
10720
|
+
// Show the saving overlay in case we have async validators in play
|
10721
|
+
this.showStatus(PageStatus.Saving);
|
10714
10722
|
this.formGroup.markAllAsTouched();
|
10715
10723
|
// If the form has pending async validators, wait for it them to complete before proceeding
|
10716
10724
|
if (this.formGroup.pending) {
|
@@ -10720,6 +10728,8 @@ class PageBaseComponent {
|
|
10720
10728
|
yield this.save();
|
10721
10729
|
}
|
10722
10730
|
else {
|
10731
|
+
// Remove the saving overlay to display any errors
|
10732
|
+
this.showStatus(PageStatus.Loaded);
|
10723
10733
|
// Only show the banner with the generic message if the parent component hasn't already set one
|
10724
10734
|
// by implementing onSave for additional validation
|
10725
10735
|
if (this.errors === '') {
|