@energycap/components 0.39.32-ECAP-27075-confirm-bug-textbox-always-showing-even-when-not-requested.20241023-1706 → 0.39.33-ECAP-26866-table-multi-sort-improvements.20241024-0933

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.
@@ -65,7 +65,7 @@ export class ConfirmComponent {
65
65
  /** Form Group to hold any form controls needed */
66
66
  this.formGroup = new UntypedFormGroup({});
67
67
  this.status = new Overlay('hasData');
68
- this.showTextBox = false;
68
+ this.showTextBox = true;
69
69
  this.destroyed = new Subject();
70
70
  this.onDialogSave = new EventEmitter();
71
71
  this.onDialogCancel = new EventEmitter();
@@ -111,13 +111,11 @@ export class ConfirmComponent {
111
111
  });
112
112
  }
113
113
  setValidations() {
114
- const showCheckBox = this.formGroup.get('checkbox');
115
- const textBoxControl = this.formGroup.controls['textbox'];
116
- if (showCheckBox) {
117
- this.showTextBox ? textBoxControl?.enable() : textBoxControl?.disable();
114
+ if (this.showTextBox) {
115
+ this.formGroup.controls['textbox']?.enable();
118
116
  }
119
117
  else {
120
- textBoxControl?.enable();
118
+ this.formGroup.controls['textbox']?.disable();
121
119
  }
122
120
  }
123
121
  /**
@@ -165,4 +163,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
165
163
  }], onDialogCancel: [{
166
164
  type: Output
167
165
  }] } });
168
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"confirm.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/confirm/confirm.component.ts","../../../../../../projects/components/src/lib/display/confirm/confirm.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn, Validators } from '@angular/forms';\r\n\r\nimport { ButtonType } from '../../controls/button/button.component';\r\nimport { TextboxType } from '../../controls/textbox/textbox.component';\r\nimport { DialogContent } from '../dialog/dialog-content';\r\nimport { Overlay } from '../view-overlay/view-overlay.component';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { Subject } from 'rxjs';\r\n\r\ntype SaveSource = 'primary'|'alternate';\r\n\r\nexport type CustomActionResult = {\r\n  message: string;\r\n  isSuccess: boolean;\r\n};\r\n\r\nexport class ConfirmDialogContext {\r\n  /** Id to distinguish between confirm component contexts emitted by the dialog service */\r\n  public id: string = '';\r\n\r\n  /** Optional title of the confirm */\r\n  public title?: string;\r\n\r\n  /** Optional message */\r\n  public message?: string;\r\n\r\n  /** Optional checkConfirm\r\n   * this will determine if ec-checkbox is available or no\r\n   * in order to delete or move to no parent the entity\r\n   */\r\n  public checkConfirm?:boolean;\r\n\r\n  /**\r\n   * checkboxContent\r\n   * Determine the label next to the checkbox\r\n   * if checkConfirm is true otherwise is not visible\r\n   */\r\n\r\n  public checkboxText?:string;\r\n\r\n  /**\r\n   * Optional textbox type, this will determine whether\r\n   * an input is visible or not\r\n   */\r\n  public textboxType?: TextboxType;\r\n\r\n  /** Optional textbox label */\r\n  public textboxLabel?: string;\r\n\r\n  /**\r\n   * Optional required flag for the textbox, will set\r\n   * a validator on the input if true\r\n   */\r\n  public textboxRequired?: boolean = false;\r\n\r\n  /** Optional array of validators to add to the textbox form control. */\r\n  public textboxValidators?: ValidatorFn[];\r\n\r\n  /** Sets the textbox's uppercase parameter */\r\n  public textboxUppercase?: boolean = false;\r\n\r\n  /**\r\n   * Optional textbox height (in rows).\r\n   * Only applies to multi-line type.\r\n   * Default 3.\r\n   */\r\n  public textareaRows?: number = 3;\r\n\r\n  /** Optional textbox placeholder text */\r\n  public textboxPlaceholder?: string;\r\n\r\n  /** Optional error message to display when textbox */\r\n  public validationErrorMessage?: string;\r\n\r\n  /** Value returned from the input if enabled */\r\n  public textboxValue?: string;\r\n  public checkboxValue?:boolean;\r\n  public confirmLabel?:string;\r\n\r\n  /** Optional save button label */\r\n  public saveLabel?: string = 'Save'; // LOCO\r\n\r\n  /** Optional save button type */\r\n  public saveButtonType?: ButtonType = 'primary';\r\n\r\n  /** Optional cancel button label */\r\n  public cancelLabel?: string = 'Cancel'; // LOCO\r\n\r\n  /** Optional flag for hiding the cancel (secondary) button */\r\n  public hideCancel?: boolean = false;\r\n\r\n  /** When true, adds the submit tag to the save button so hitting enter triggers the save */\r\n  public saveOnEnter?: boolean = false;\r\n\r\n  /** Optional alternate save button label */\r\n  public alternateSaveLabel?: string = undefined;\r\n\r\n  /** Optional alternate save button type */\r\n  public alternateSaveButtonType?: ButtonType = undefined;\r\n\r\n  /**\r\n   * Used to know which save button was selected in case the calling component\r\n   * is utilizing the alternate save button\r\n   */\r\n  public saveSource?: SaveSource = 'primary';\r\n\r\n  /**\r\n   * When provided, the confirm will show a pending state and call this, then upon\r\n   * completion display the result message and a close button. If the promise is resolved\r\n   * successfully, the close button will emit onDialogSave. If the promise is rejected,\r\n   * the close button will emit onDialogCancel.\r\n   */\r\n  public inlineConfirmAction?: () => Promise<CustomActionResult>;\r\n\r\n  /** The message to be displayed while inlineConfirmAction is pending. */\r\n  public inlineConfirmPendingMessage?: string = '';\r\n}\r\n\r\n@Component({\r\n  selector: 'ec-confirm',\r\n  templateUrl: './confirm.component.html',\r\n  styleUrls: ['./confirm.component.scss']\r\n})\r\nexport class ConfirmComponent implements OnInit, DialogContent<ConfirmDialogContext>, OnDestroy{\r\n\r\n  public dialogId: string = 'Confirm';\r\n\r\n  /** Form Group to hold any form controls needed */\r\n  public formGroup: UntypedFormGroup = new UntypedFormGroup({});\r\n\r\n  public status = new Overlay('hasData');\r\n\r\n  public showTextBox:boolean = false;\r\n\r\n  private destroyed = new Subject<void>();\r\n\r\n  /**\r\n   * Provided by the calling component to tell the confirm what to show\r\n   * and is used on save to send back any entered values from the user\r\n   */\r\n  @Input() public context!: ConfirmDialogContext;\r\n\r\n  @Output() public onDialogSave: EventEmitter<ConfirmDialogContext> = new EventEmitter<ConfirmDialogContext>();\r\n  @Output() public onDialogCancel: EventEmitter<void> = new EventEmitter<void>();\r\n\r\n  public onOpen(openContext?: ConfirmDialogContext): void {\r\n    this.context = openContext!;\r\n    this.dialogId = openContext!.id;\r\n    if (this.context.alternateSaveLabel) {\r\n      this.context.saveOnEnter = false;\r\n    }\r\n  }\r\n\r\n  /** Error to be displayed in the error banner */\r\n  public error?: string;\r\n\r\n  constructor() { }\r\n\r\n  public ngOnInit(): void {\r\n    this.addFormControls();\r\n    this.setValidations();\r\n    this.listenCheckBox();\r\n  }\r\n\r\n  public onSave(source: SaveSource): void {\r\n    this.formGroup.markAllAsTouched();\r\n\r\n    if (this.formGroup.valid) {\r\n      let formValues = this.formGroup.getRawValue();\r\n      if (formValues.textbox) {\r\n        this.context.textboxValue = formValues.textbox;\r\n      }\r\n      if(this.context.checkConfirm) {\r\n        this.context.checkboxValue = formValues.checkbox ? formValues.checkbox : false;\r\n      }\r\n\r\n      this.context.saveSource = source;\r\n\r\n      if (this.context.inlineConfirmAction) {\r\n        this.doInlineConfirmAction();\r\n      } else {\r\n        this.onDialogSave.emit(this.context);\r\n      }\r\n      \r\n    } else if (this.context.validationErrorMessage) {\r\n      this.error = this.context.validationErrorMessage;\r\n    }\r\n  }\r\n\r\n  public onCancel(): void {\r\n    this.onDialogCancel.emit();\r\n  }\r\n\r\n  public ngOnDestroy(): void {\r\n    this.destroyed.next();\r\n    this.destroyed.complete();\r\n  }\r\n\r\n  public listenCheckBox():void {\r\n    this.formGroup.get('checkbox')?.valueChanges.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe(value => {\r\n      this.showTextBox = value;\r\n      this.setValidations();\r\n    });\r\n  }\r\n\r\n  private setValidations(): void {\r\n    const showCheckBox = this.formGroup.get('checkbox');\r\n    const textBoxControl = this.formGroup.controls['textbox'];\r\n\r\n    if (showCheckBox) {\r\n        this.showTextBox ? textBoxControl?.enable() : textBoxControl?.disable();\r\n    } else {\r\n        textBoxControl?.enable();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Adds the textbox form control to our form group if its configured to be visible\r\n   * Will also add the required validator if requested\r\n   */\r\n  private addFormControls(): void {\r\n    if (this.context.textboxType) {\r\n      let validators: ValidatorFn[] = [];\r\n      if (this.context.textboxRequired) {\r\n        validators.push(Validators.required);\r\n      }\r\n      if (this.context.textboxValidators) {\r\n        validators.push(...this.context.textboxValidators);\r\n      }\r\n      this.formGroup.addControl('textbox', new UntypedFormControl('', validators));\r\n    }\r\n\r\n    if(this.context.checkConfirm){\r\n      this.showTextBox = false;\r\n      const checkboxControl = new UntypedFormControl('');\r\n      this.formGroup.addControl('checkbox', checkboxControl);      \r\n    }\r\n  }\r\n\r\n  private async doInlineConfirmAction() {\r\n    this.status.setStatus('pending', this.context.inlineConfirmPendingMessage);\r\n\r\n    let result = await this.context.inlineConfirmAction!();\r\n\r\n    this.status.setStatus('noData', result.message, {\r\n      id: 'inlineConfirmClose',\r\n      label: 'Close',\r\n      icon: result.isSuccess ? 'icon-check-circle' : 'icon-error',\r\n      classlist: result.isSuccess ? 'font-color-success' : 'font-color-error',\r\n      onClick: () => { result.isSuccess ? this.onDialogSave.emit(this.context) : this.onDialogCancel.emit() }\r\n    });\r\n  }\r\n}\r\n","<section ecOverlay\r\n         [status]=\"status?.status\"\r\n         [message]=\"status?.message\"\r\n         [action]=\"status?.action\"\r\n         [noDataTemplate]=\"inlineConfirmResult\"\r\n         overlayClassList=\"p-0\"\r\n         class=\"bg-body flex-grow d-flex\">\r\n  <form [formGroup]=\"formGroup\" class=\"flex-grow flex-column confirm-content text-body-1\">\r\n    <section class=\"flex-grow\">\r\n      <h2 class=\"font-weight-bold mb-3\" *ngIf=\"context?.title\">{{context?.title}}</h2>\r\n      <ec-banner *ngIf=\"error\" [text]=\"error | translate\" bannerStyle=\"normal\" class=\"mb-3\"></ec-banner>\r\n      <div *ngIf=\"context?.message\" [innerHtml]=\"context?.message | translate\" class=\"mb-3\"></div>\r\n      \r\n      <ec-checkbox *ngIf=\"context?.checkConfirm\"\r\n                    [formModel]=\"formGroup.get('checkbox')\"\r\n                    name=\"confirmCheckbox\"\r\n                    [label]=\"context?.checkboxText\"\r\n                    [autofocus]=\"true\"\r\n                    ></ec-checkbox>\r\n      <div *ngIf=\"showTextBox\" [innerHtml]=\"context.confirmLabel\" class=\"mb-3\"></div>\r\n      <ec-textbox id=\"confirmTextbox\"\r\n                  *ngIf=\"(context?.textboxType && !context?.checkConfirm) || showTextBox\"\r\n                  [autofocus]=\"true\"\r\n                  [formModel]=\"formGroup.get('textbox')\"\r\n                  [required]=\"context?.textboxRequired\"\r\n                  [label]=\"context?.textboxLabel\"\r\n                  [type]=\"context?.textboxType\"\r\n                  [rows]=\"context?.textareaRows\"\r\n                  [placeholder]=\"context?.textboxPlaceholder\"\r\n                  [upperCase]=\"context?.textboxUppercase\">\r\n      </ec-textbox>\r\n    </section>\r\n    <footer class=\"mt-auto flex-shrink d-flex py-2 flex-row-reverse\">\r\n      <ec-button id=\"saveConfirmButton\"\r\n                  class=\"ml-2\"\r\n                  [type]=\"context?.saveButtonType ? context?.saveButtonType : 'primary'\"\r\n                  [label]=\"context?.saveLabel ? context?.saveLabel : 'Save'\"\r\n                  [autofocus]=\"!context?.textboxType\"\r\n                  [isSubmit]=\"context?.saveOnEnter\"\r\n                  (clicked)=\"onSave('primary')\">\r\n      </ec-button>\r\n      <ec-button id=\"alternateSaveConfirmButton\"\r\n                  *ngIf=\"context?.alternateSaveLabel\"\r\n                  class=\"ml-2\"\r\n                  [type]=\"context?.alternateSaveButtonType ? context?.alternateSaveButtonType : 'primary'\"\r\n                  [label]=\"context?.alternateSaveLabel\"\r\n                  (clicked)=\"onSave('alternate')\">\r\n      </ec-button>\r\n      <ec-button  *ngIf=\"!context?.hideCancel\"\r\n                  id=\"cancelConfirmButton\"\r\n                  class=\"ml-auto\"\r\n                  type=\"secondary\"\r\n                  [label]=\"context?.cancelLabel ? context?.cancelLabel : 'Cancel'\"\r\n                  (clicked)=\"onCancel()\">\r\n      </ec-button>\r\n    </footer>\r\n  </form>\r\n</section>\r\n\r\n<ng-template #inlineConfirmResult>\r\n  <section class=\"flex-grow align-self-stretch confirm-content text-body-1\">\r\n    <div class=\"d-flex\">\r\n      <i class=\"ec-icon {{status?.action?.icon}} {{status?.action?.classlist}} inline-confirm-result-icon\"></i>\r\n      <div [innerHTML]=\"status?.message | translate\"></div>\r\n    </div>\r\n    <div class=\"d-flex pt-2 mt-auto\">\r\n      <ec-button id=\"inlineConfirmClose\"\r\n                 label=\"Close\"\r\n                 type=\"secondary\"\r\n                 class=\"ml-auto\"\r\n                 (clicked)=\"status?.action?.onClick!()\"\r\n                 [autofocus]=\"true\">\r\n      </ec-button>\r\n    </div>\r\n  </section>\r\n</ng-template>"]}
166
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"confirm.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/confirm/confirm.component.ts","../../../../../../projects/components/src/lib/display/confirm/confirm.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn, Validators } from '@angular/forms';\r\n\r\nimport { ButtonType } from '../../controls/button/button.component';\r\nimport { TextboxType } from '../../controls/textbox/textbox.component';\r\nimport { DialogContent } from '../dialog/dialog-content';\r\nimport { Overlay } from '../view-overlay/view-overlay.component';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { Subject } from 'rxjs';\r\n\r\ntype SaveSource = 'primary'|'alternate';\r\n\r\nexport type CustomActionResult = {\r\n  message: string;\r\n  isSuccess: boolean;\r\n};\r\n\r\nexport class ConfirmDialogContext {\r\n  /** Id to distinguish between confirm component contexts emitted by the dialog service */\r\n  public id: string = '';\r\n\r\n  /** Optional title of the confirm */\r\n  public title?: string;\r\n\r\n  /** Optional message */\r\n  public message?: string;\r\n\r\n  /** Optional checkConfirm\r\n   * this will determine if ec-checkbox is available or no\r\n   * in order to delete or move to no parent the entity\r\n   */\r\n  public checkConfirm?:boolean;\r\n\r\n  /**\r\n   * checkboxContent\r\n   * Determine the label next to the checkbox\r\n   * if checkConfirm is true otherwise is not visible\r\n   */\r\n\r\n  public checkboxText?:string;\r\n\r\n  /**\r\n   * Optional textbox type, this will determine whether\r\n   * an input is visible or not\r\n   */\r\n  public textboxType?: TextboxType;\r\n\r\n  /** Optional textbox label */\r\n  public textboxLabel?: string;\r\n\r\n  /**\r\n   * Optional required flag for the textbox, will set\r\n   * a validator on the input if true\r\n   */\r\n  public textboxRequired?: boolean = false;\r\n\r\n  /** Optional array of validators to add to the textbox form control. */\r\n  public textboxValidators?: ValidatorFn[];\r\n\r\n  /** Sets the textbox's uppercase parameter */\r\n  public textboxUppercase?: boolean = false;\r\n\r\n  /**\r\n   * Optional textbox height (in rows).\r\n   * Only applies to multi-line type.\r\n   * Default 3.\r\n   */\r\n  public textareaRows?: number = 3;\r\n\r\n  /** Optional textbox placeholder text */\r\n  public textboxPlaceholder?: string;\r\n\r\n  /** Optional error message to display when textbox */\r\n  public validationErrorMessage?: string;\r\n\r\n  /** Value returned from the input if enabled */\r\n  public textboxValue?: string;\r\n  public checkboxValue?:boolean;\r\n  public confirmLabel?:string;\r\n\r\n  /** Optional save button label */\r\n  public saveLabel?: string = 'Save'; // LOCO\r\n\r\n  /** Optional save button type */\r\n  public saveButtonType?: ButtonType = 'primary';\r\n\r\n  /** Optional cancel button label */\r\n  public cancelLabel?: string = 'Cancel'; // LOCO\r\n\r\n  /** Optional flag for hiding the cancel (secondary) button */\r\n  public hideCancel?: boolean = false;\r\n\r\n  /** When true, adds the submit tag to the save button so hitting enter triggers the save */\r\n  public saveOnEnter?: boolean = false;\r\n\r\n  /** Optional alternate save button label */\r\n  public alternateSaveLabel?: string = undefined;\r\n\r\n  /** Optional alternate save button type */\r\n  public alternateSaveButtonType?: ButtonType = undefined;\r\n\r\n  /**\r\n   * Used to know which save button was selected in case the calling component\r\n   * is utilizing the alternate save button\r\n   */\r\n  public saveSource?: SaveSource = 'primary';\r\n\r\n  /**\r\n   * When provided, the confirm will show a pending state and call this, then upon\r\n   * completion display the result message and a close button. If the promise is resolved\r\n   * successfully, the close button will emit onDialogSave. If the promise is rejected,\r\n   * the close button will emit onDialogCancel.\r\n   */\r\n  public inlineConfirmAction?: () => Promise<CustomActionResult>;\r\n\r\n  /** The message to be displayed while inlineConfirmAction is pending. */\r\n  public inlineConfirmPendingMessage?: string = '';\r\n}\r\n\r\n@Component({\r\n  selector: 'ec-confirm',\r\n  templateUrl: './confirm.component.html',\r\n  styleUrls: ['./confirm.component.scss']\r\n})\r\nexport class ConfirmComponent implements OnInit, DialogContent<ConfirmDialogContext>, OnDestroy{\r\n\r\n  public dialogId: string = 'Confirm';\r\n\r\n  /** Form Group to hold any form controls needed */\r\n  public formGroup: UntypedFormGroup = new UntypedFormGroup({});\r\n\r\n  public status = new Overlay('hasData');\r\n\r\n  public showTextBox:boolean = true;\r\n\r\n  private destroyed = new Subject<void>();\r\n\r\n  /**\r\n   * Provided by the calling component to tell the confirm what to show\r\n   * and is used on save to send back any entered values from the user\r\n   */\r\n  @Input() public context!: ConfirmDialogContext;\r\n\r\n  @Output() public onDialogSave: EventEmitter<ConfirmDialogContext> = new EventEmitter<ConfirmDialogContext>();\r\n  @Output() public onDialogCancel: EventEmitter<void> = new EventEmitter<void>();\r\n\r\n  public onOpen(openContext?: ConfirmDialogContext): void {\r\n    this.context = openContext!;\r\n    this.dialogId = openContext!.id;\r\n\r\n    if (this.context.alternateSaveLabel) {\r\n      this.context.saveOnEnter = false;\r\n    }\r\n  }\r\n\r\n  /** Error to be displayed in the error banner */\r\n  public error?: string;\r\n\r\n  constructor() { }\r\n\r\n  public ngOnInit(): void {\r\n    this.addFormControls();\r\n    this.setValidations();\r\n    this.listenCheckBox();\r\n  }\r\n\r\n  public onSave(source: SaveSource): void {\r\n    this.formGroup.markAllAsTouched();\r\n\r\n    if (this.formGroup.valid) {\r\n      let formValues = this.formGroup.getRawValue();\r\n      if (formValues.textbox) {\r\n        this.context.textboxValue = formValues.textbox;\r\n      }\r\n      if(this.context.checkConfirm) {\r\n        this.context.checkboxValue = formValues.checkbox ? formValues.checkbox : false;\r\n      }\r\n\r\n      this.context.saveSource = source;\r\n\r\n      if (this.context.inlineConfirmAction) {\r\n        this.doInlineConfirmAction();\r\n      } else {\r\n        this.onDialogSave.emit(this.context);\r\n      }\r\n    } else if (this.context.validationErrorMessage) {\r\n      this.error = this.context.validationErrorMessage;\r\n    }\r\n  }\r\n\r\n  public onCancel(): void {\r\n    this.onDialogCancel.emit();\r\n  }\r\n\r\n  public ngOnDestroy(): void {\r\n    this.destroyed.next();\r\n    this.destroyed.complete();\r\n  }\r\n\r\n  public listenCheckBox():void {\r\n    this.formGroup.get('checkbox')?.valueChanges.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe(value => {\r\n      this.showTextBox = value;\r\n      this.setValidations();\r\n    });\r\n  }\r\n\r\n  private setValidations():void {\r\n    if(this.showTextBox){\r\n      this.formGroup.controls['textbox']?.enable();\r\n    } else {\r\n      this.formGroup.controls['textbox']?.disable();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Adds the textbox form control to our form group if its configured to be visible\r\n   * Will also add the required validator if requested\r\n   */\r\n  private addFormControls(): void {\r\n    if (this.context.textboxType) {\r\n      let validators: ValidatorFn[] = [];\r\n      if (this.context.textboxRequired) {\r\n        validators.push(Validators.required);\r\n      }\r\n      if (this.context.textboxValidators) {\r\n        validators.push(...this.context.textboxValidators);\r\n      }\r\n      this.formGroup.addControl('textbox', new UntypedFormControl('', validators));\r\n    }\r\n\r\n    if(this.context.checkConfirm){\r\n      this.showTextBox = false;\r\n      const checkboxControl = new UntypedFormControl('');\r\n      this.formGroup.addControl('checkbox', checkboxControl);      \r\n    }\r\n  }\r\n\r\n  private async doInlineConfirmAction() {\r\n    this.status.setStatus('pending', this.context.inlineConfirmPendingMessage);\r\n\r\n    let result = await this.context.inlineConfirmAction!();\r\n\r\n    this.status.setStatus('noData', result.message, {\r\n      id: 'inlineConfirmClose',\r\n      label: 'Close',\r\n      icon: result.isSuccess ? 'icon-check-circle' : 'icon-error',\r\n      classlist: result.isSuccess ? 'font-color-success' : 'font-color-error',\r\n      onClick: () => { result.isSuccess ? this.onDialogSave.emit(this.context) : this.onDialogCancel.emit() }\r\n    });\r\n  }\r\n}\r\n","<section ecOverlay\r\n         [status]=\"status?.status\"\r\n         [message]=\"status?.message\"\r\n         [action]=\"status?.action\"\r\n         [noDataTemplate]=\"inlineConfirmResult\"\r\n         overlayClassList=\"p-0\"\r\n         class=\"bg-body flex-grow d-flex\">\r\n  <form [formGroup]=\"formGroup\" class=\"flex-grow flex-column confirm-content text-body-1\">\r\n    <section class=\"flex-grow\">\r\n      <h2 class=\"font-weight-bold mb-3\" *ngIf=\"context?.title\">{{context?.title}}</h2>\r\n      <ec-banner *ngIf=\"error\" [text]=\"error | translate\" bannerStyle=\"normal\" class=\"mb-3\"></ec-banner>\r\n      <div *ngIf=\"context?.message\" [innerHtml]=\"context?.message | translate\" class=\"mb-3\"></div>\r\n      \r\n      <ec-checkbox *ngIf=\"context?.checkConfirm\"\r\n                    [formModel]=\"formGroup.get('checkbox')\"\r\n                    name=\"confirmCheckbox\"\r\n                    [label]=\"context?.checkboxText\"\r\n                    [autofocus]=\"true\"\r\n                    ></ec-checkbox>\r\n      <div *ngIf=\"showTextBox\" [innerHtml]=\"context.confirmLabel\" class=\"mb-3\"></div>\r\n      <ec-textbox id=\"confirmTextbox\"\r\n                  *ngIf=\"(context?.textboxType && !context?.checkConfirm) || showTextBox\"\r\n                  [autofocus]=\"true\"\r\n                  [formModel]=\"formGroup.get('textbox')\"\r\n                  [required]=\"context?.textboxRequired\"\r\n                  [label]=\"context?.textboxLabel\"\r\n                  [type]=\"context?.textboxType\"\r\n                  [rows]=\"context?.textareaRows\"\r\n                  [placeholder]=\"context?.textboxPlaceholder\"\r\n                  [upperCase]=\"context?.textboxUppercase\">\r\n      </ec-textbox>\r\n    </section>\r\n    <footer class=\"mt-auto flex-shrink d-flex py-2 flex-row-reverse\">\r\n      <ec-button id=\"saveConfirmButton\"\r\n                  class=\"ml-2\"\r\n                  [type]=\"context?.saveButtonType ? context?.saveButtonType : 'primary'\"\r\n                  [label]=\"context?.saveLabel ? context?.saveLabel : 'Save'\"\r\n                  [autofocus]=\"!context?.textboxType\"\r\n                  [isSubmit]=\"context?.saveOnEnter\"\r\n                  (clicked)=\"onSave('primary')\">\r\n      </ec-button>\r\n      <ec-button id=\"alternateSaveConfirmButton\"\r\n                  *ngIf=\"context?.alternateSaveLabel\"\r\n                  class=\"ml-2\"\r\n                  [type]=\"context?.alternateSaveButtonType ? context?.alternateSaveButtonType : 'primary'\"\r\n                  [label]=\"context?.alternateSaveLabel\"\r\n                  (clicked)=\"onSave('alternate')\">\r\n      </ec-button>\r\n      <ec-button  *ngIf=\"!context?.hideCancel\"\r\n                  id=\"cancelConfirmButton\"\r\n                  class=\"ml-auto\"\r\n                  type=\"secondary\"\r\n                  [label]=\"context?.cancelLabel ? context?.cancelLabel : 'Cancel'\"\r\n                  (clicked)=\"onCancel()\">\r\n      </ec-button>\r\n    </footer>\r\n  </form>\r\n</section>\r\n\r\n<ng-template #inlineConfirmResult>\r\n  <section class=\"flex-grow align-self-stretch confirm-content text-body-1\">\r\n    <div class=\"d-flex\">\r\n      <i class=\"ec-icon {{status?.action?.icon}} {{status?.action?.classlist}} inline-confirm-result-icon\"></i>\r\n      <div [innerHTML]=\"status?.message | translate\"></div>\r\n    </div>\r\n    <div class=\"d-flex pt-2 mt-auto\">\r\n      <ec-button id=\"inlineConfirmClose\"\r\n                 label=\"Close\"\r\n                 type=\"secondary\"\r\n                 class=\"ml-auto\"\r\n                 (clicked)=\"status?.action?.onClick!()\"\r\n                 [autofocus]=\"true\">\r\n      </ec-button>\r\n    </div>\r\n  </section>\r\n</ng-template>"]}
@@ -108,4 +108,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
108
108
  }], itemToggled: [{
109
109
  type: Output
110
110
  }] } });
111
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchy-base.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/hierarchy/hierarchy-base.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, EventEmitter, Injector, Input, OnChanges, OnDestroy, Output, SimpleChanges } from '@angular/core';\r\nimport { Params, QueryParamsHandling } from '@angular/router';\r\nimport { Subject } from 'rxjs';\r\nimport { LinkItem } from '../../controls/navigation/link-item';\r\nimport { TelemetryService } from '../../core/telemetry.service';\r\nimport { ViewStatus } from '../../shared/display';\r\n\r\nexport class HierarchyItem implements LinkItem {\r\n  id: string = '';\r\n\r\n  /** Label to display for the item */\r\n  label: string = '';\r\n\r\n  /** Text to be displayed in the item's tooltip. When not provided, the label will be displayed. */\r\n  tooltip?: string;\r\n\r\n  /** Indicates how many levels deep the item is in the parent structure */\r\n  level: number = 0;\r\n\r\n  url?: string;\r\n\r\n  queryParams?: Params | null;\r\n\r\n  queryParamsHandling?: QueryParamsHandling;\r\n\r\n  icon?: string;\r\n\r\n  /** Display item as a link, a heading, or with a divider line below it */\r\n  display?: 'default' | 'heading' | 'divider' = 'default';\r\n\r\n  /** Set to hide toggle, even if item has children */\r\n  hideToggle?: boolean = false;\r\n\r\n  /** Flag to indicate whether the item has children or not, mostly drives\r\n   *  whether you get the expand/collapse button\r\n   */\r\n  hasChildren: boolean = false;\r\n\r\n  /** Array of children if any for the item */\r\n  children: Array<HierarchyItem> = [];\r\n\r\n  /** Flag to indicate if the item has been expanded */\r\n  expanded?: boolean = false;\r\n\r\n  /** Flag to indicate if the item is selected */\r\n  selected?: boolean = false;\r\n\r\n  /** Flag to indicate if the item can be selected */\r\n  selectable?: boolean = false;\r\n\r\n  readonly?: boolean = false;\r\n\r\n  /** Status of each item to indicate whether children lookup is pending or\r\n   *  if an error occurred retrieving data\r\n   */\r\n  status?: ViewStatus = \"hasData\";\r\n\r\n  /** Indicates if the item is a top level entity, e.g. Organization or Cost Center */\r\n  topLevel?: boolean = false;\r\n\r\n  /**\r\n   * When checking the url for activeness, the url must match exactly or partially\r\n   * @see https://angular.io/guide/router#active-router-links\r\n   */\r\n  isActiveExactMatch?: boolean = true;\r\n\r\n  /**\r\n   * Reference to the parent item. Useful if you need to walk the tree in reverse. An example usage\r\n   * would be to expand all parent items above a node that is the current selection.\r\n   */\r\n  parentItem?: HierarchyItem;\r\n\r\n  /**\r\n   * Set to have no level indentation for the item. Useful for top level items in a tree that is not hierarchical\r\n   */\r\n  noIndent?: boolean | undefined;\r\n}\r\n\r\n/**\r\n * Represents a HierarchyItem that has been clicked to toggle its expansion state.\r\n */\r\nexport type ToggledHierarchyItem = {\r\n  item: HierarchyItem;\r\n  expanded: boolean;\r\n};\r\n\r\n@Directive()\r\nexport abstract class HierarchyBase implements OnChanges, OnDestroy {\r\n\r\n  /** First item to start the hierarchy structure */\r\n  @Input() public rootNode!: HierarchyItem;\r\n\r\n  /** Event emitter to request children for a given item */\r\n  @Output() public getItemChildren: EventEmitter<HierarchyItem> = new EventEmitter<HierarchyItem>();\r\n\r\n  /** Event emitted when an item is toggled. */\r\n  @Output() public itemToggled: EventEmitter<ToggledHierarchyItem> = new EventEmitter<ToggledHierarchyItem>();\r\n\r\n  /**\r\n  * Used to unsubscribe from observables\r\n  */\r\n  private _destroyed: Subject<any> = new Subject<any>();\r\n  public get destroyed(): Subject<any> {\r\n    return this._destroyed;\r\n  }\r\n\r\n  protected telemetryService: TelemetryService;\r\n\r\n  constructor(injector: Injector) {\r\n    this.telemetryService = injector.get(TelemetryService);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes && changes.rootNode && changes.rootNode.currentValue) {\r\n      /** If the rootNode has children and we don't know about them yet we must be loading\r\n      *  the view without the new hierarchy api\r\n      */\r\n      if (this.rootNode.hasChildren && this.rootNode.children.length === 0) {\r\n        this.getItemChildren.emit(this.rootNode);\r\n      }\r\n    }\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  /** Handler for the toggle button */\r\n  public toggleItemClicked(item: HierarchyItem, expanded: boolean) {\r\n    item.expanded = expanded;\r\n    this.telemetryService.trackEvent('toggle-tree-node', {\r\n      newExpandState: item.expanded ? 'expanded' : 'collapsed',\r\n      itemDepth: `${item.level}`,\r\n    });\r\n\r\n    this.itemToggled.emit({ item, expanded });\r\n\r\n    if (item.expanded && item.hasChildren && item.children.length === 0) {\r\n      item.status = \"pending\";\r\n      this.getItemChildren.emit(item);\r\n    }\r\n  }\r\n\r\n  /** Method exposed to the parent component to set the items children once they are available\r\n   */\r\n  public setItemChildren(parentItem: HierarchyItem, items: Array<HierarchyItem>): void {\r\n    parentItem.children = items;\r\n    parentItem.status = \"hasData\";\r\n  }\r\n\r\n  /** Method exposed to the parent component to indicate that something happened on the\r\n   *  item that caused an error, could be retrieving children\r\n  */\r\n  public setItemError(parentItem: HierarchyItem): void {\r\n    parentItem.status = \"error\";\r\n  }\r\n}\r\n"]}
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchy-base.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/hierarchy/hierarchy-base.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, EventEmitter, Injector, Input, OnChanges, OnDestroy, Output, SimpleChanges } from '@angular/core';\r\nimport { Params, QueryParamsHandling } from '@angular/router';\r\nimport { Subject } from 'rxjs';\r\nimport { LinkItem } from '../../controls/navigation/link-item';\r\nimport { TelemetryService } from '../../core/telemetry.service';\r\nimport { ViewStatus } from '../../shared/display';\r\n\r\nexport class HierarchyItem implements LinkItem {\r\n  id: string = '';\r\n\r\n  /** Label to display for the item */\r\n  label: string = '';\r\n\r\n  /** Text to be displayed in the item's tooltip. When not provided, the label will be displayed. */\r\n  tooltip?: string;\r\n\r\n  /** Text to be displayed in the item's icon tooltip. */\r\n  iconTooltip?: string;\r\n\r\n  /** Indicates how many levels deep the item is in the parent structure */\r\n  level: number = 0;\r\n\r\n  url?: string;\r\n\r\n  queryParams?: Params | null;\r\n\r\n  queryParamsHandling?: QueryParamsHandling;\r\n\r\n  icon?: string;\r\n\r\n  /** Display item as a link, a heading, or with a divider line below it */\r\n  display?: 'default' | 'heading' | 'divider' = 'default';\r\n\r\n  /** Set to hide toggle, even if item has children */\r\n  hideToggle?: boolean = false;\r\n\r\n  /** Flag to indicate whether the item has children or not, mostly drives\r\n   *  whether you get the expand/collapse button\r\n   */\r\n  hasChildren: boolean = false;\r\n\r\n  /** Array of children if any for the item */\r\n  children: Array<HierarchyItem> = [];\r\n\r\n  /** Flag to indicate if the item has been expanded */\r\n  expanded?: boolean = false;\r\n\r\n  /** Flag to indicate if the item is selected */\r\n  selected?: boolean = false;\r\n\r\n  /** Flag to indicate if the item can be selected */\r\n  selectable?: boolean = false;\r\n\r\n  readonly?: boolean = false;\r\n\r\n  /** Status of each item to indicate whether children lookup is pending or\r\n   *  if an error occurred retrieving data\r\n   */\r\n  status?: ViewStatus = \"hasData\";\r\n\r\n  /** Indicates if the item is a top level entity, e.g. Organization or Cost Center */\r\n  topLevel?: boolean = false;\r\n\r\n  /**\r\n   * When checking the url for activeness, the url must match exactly or partially\r\n   * @see https://angular.io/guide/router#active-router-links\r\n   */\r\n  isActiveExactMatch?: boolean = true;\r\n\r\n  /**\r\n   * Reference to the parent item. Useful if you need to walk the tree in reverse. An example usage\r\n   * would be to expand all parent items above a node that is the current selection.\r\n   */\r\n  parentItem?: HierarchyItem;\r\n\r\n  /**\r\n   * Set to have no level indentation for the item. Useful for top level items in a tree that is not hierarchical\r\n   */\r\n  noIndent?: boolean | undefined;\r\n}\r\n\r\n/**\r\n * Represents a HierarchyItem that has been clicked to toggle its expansion state.\r\n */\r\nexport type ToggledHierarchyItem = {\r\n  item: HierarchyItem;\r\n  expanded: boolean;\r\n};\r\n\r\n@Directive()\r\nexport abstract class HierarchyBase implements OnChanges, OnDestroy {\r\n\r\n  /** First item to start the hierarchy structure */\r\n  @Input() public rootNode!: HierarchyItem;\r\n\r\n  /** Event emitter to request children for a given item */\r\n  @Output() public getItemChildren: EventEmitter<HierarchyItem> = new EventEmitter<HierarchyItem>();\r\n\r\n  /** Event emitted when an item is toggled. */\r\n  @Output() public itemToggled: EventEmitter<ToggledHierarchyItem> = new EventEmitter<ToggledHierarchyItem>();\r\n\r\n  /**\r\n  * Used to unsubscribe from observables\r\n  */\r\n  private _destroyed: Subject<any> = new Subject<any>();\r\n  public get destroyed(): Subject<any> {\r\n    return this._destroyed;\r\n  }\r\n\r\n  protected telemetryService: TelemetryService;\r\n\r\n  constructor(injector: Injector) {\r\n    this.telemetryService = injector.get(TelemetryService);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes && changes.rootNode && changes.rootNode.currentValue) {\r\n      /** If the rootNode has children and we don't know about them yet we must be loading\r\n      *  the view without the new hierarchy api\r\n      */\r\n      if (this.rootNode.hasChildren && this.rootNode.children.length === 0) {\r\n        this.getItemChildren.emit(this.rootNode);\r\n      }\r\n    }\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  /** Handler for the toggle button */\r\n  public toggleItemClicked(item: HierarchyItem, expanded: boolean) {\r\n    item.expanded = expanded;\r\n    this.telemetryService.trackEvent('toggle-tree-node', {\r\n      newExpandState: item.expanded ? 'expanded' : 'collapsed',\r\n      itemDepth: `${item.level}`,\r\n    });\r\n\r\n    this.itemToggled.emit({ item, expanded });\r\n\r\n    if (item.expanded && item.hasChildren && item.children.length === 0) {\r\n      item.status = \"pending\";\r\n      this.getItemChildren.emit(item);\r\n    }\r\n  }\r\n\r\n  /** Method exposed to the parent component to set the items children once they are available\r\n   */\r\n  public setItemChildren(parentItem: HierarchyItem, items: Array<HierarchyItem>): void {\r\n    parentItem.children = items;\r\n    parentItem.status = \"hasData\";\r\n  }\r\n\r\n  /** Method exposed to the parent component to indicate that something happened on the\r\n   *  item that caused an error, could be retrieving children\r\n  */\r\n  public setItemError(parentItem: HierarchyItem): void {\r\n    parentItem.status = \"error\";\r\n  }\r\n}\r\n"]}
@@ -45,10 +45,10 @@ export class HierarchyTreeComponent extends HierarchyBase {
45
45
  }
46
46
  }
47
47
  HierarchyTreeComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: HierarchyTreeComponent, deps: [{ token: i1.ScrollService }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Component });
48
- HierarchyTreeComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: HierarchyTreeComponent, selector: "ec-hierarchy-tree", inputs: { id: "id", hideRootNode: "hideRootNode", customItemTemplate: "customItemTemplate" }, outputs: { itemSelected: "itemSelected" }, usesInheritance: true, ngImport: i0, template: "<ul id=\"{{scrollContainerId}}\"\r\n class=\"flex-grow scroll-y py-1\">\r\n\r\n <li *ngIf=\"!hideRootNode && rootNode\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{rootNode?.tooltip ?? rootNode?.label}}\"\r\n [ngClass]=\"{'is-heading': rootNode?.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"rootNode?.url\"\r\n [ecNavItemActiveExactMatch]='rootNode?.isActiveExactMatch'>\r\n <a id=\"rootNode_{{rootNode?.id}}_link\"\r\n class=\"item flex-grow d-flex align-items-center\"\r\n routerLink=\"{{rootNode?.url}}\"\r\n [queryParams]=\"rootNode?.queryParams\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: rootNode }\"></ng-container>\r\n </a>\r\n </div>\r\n </li>\r\n\r\n <ng-template #hierarchyView\r\n let-items>\r\n <li *ngFor=\"let item of items; index as index; first as isFirst\"\r\n [ngClass]=\"{'divider': item.display === 'divider'}\"\r\n id=\"treeItem_{{item.id}}\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{item.tooltip ?? item.label}}\"\r\n [ngClass]=\"{'is-heading': item.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"item.url\"\r\n [ecNavItemActiveExactMatch]=\"item.isActiveExactMatch\"\r\n [ecNavItemActiveQueryParams]=\"item.queryParams\">\r\n\r\n <!-- This element provides the indentation for each level -->\r\n <span id=\"indent_{{item.id}}\"\r\n *ngIf=\"!item.noIndent\"\r\n class=\"d-block h-100\"\r\n [style.width.px]=\"(indent) * (item.level - 1) + (item.level * 4) + (item.hasChildren ? 0 : indent)\">\r\n </span>\r\n\r\n <!-- Toggle the button icon to be a spinner when item.status is pending -->\r\n <ec-collapsible-toggle id=\"toggle_{{item.id}}\"\r\n class=\"flex-shrink\"\r\n [style.width.px]=\"indent\"\r\n *ngIf=\"item.hasChildren && !item.hideToggle\"\r\n [hidden]=\"item.status === 'pending'\"\r\n [expanded]=\"item.expanded\"\r\n (expandedChange)=\"toggleItemClicked(item, $event)\">\r\n </ec-collapsible-toggle>\r\n\r\n <i class=\"ec-icon icon-loading my-1\"\r\n [hidden]=\"item.status !== 'pending'\"></i>\r\n\r\n <a id=\"treeItem_{{item.id}}_link\"\r\n class=\"item\"\r\n *ngIf=\"item.url\"\r\n (click)=\"selectItem(item)\"\r\n [routerLink]=\"item.url\"\r\n [queryParams]=\"item.queryParams\"\r\n [queryParamsHandling]=\"item.queryParamsHandling || ''\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </a>\r\n <div id=\"treeItem_{{item.id}}_heading\"\r\n class=\"item\"\r\n *ngIf=\"!item.url\"\r\n (click)=\"selectItem(item)\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </div>\r\n </div>\r\n\r\n\r\n <ul *ngIf=\"item.children.length > 0 && item.expanded\">\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: item.children }\"></ng-container>\r\n </ul>\r\n </li>\r\n </ng-template>\r\n\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: rootNode?.children }\"></ng-container>\r\n</ul>\r\n\r\n<ng-template #defaultItemTemplate\r\n let-item>\r\n <i class=\"ec-icon {{item.icon}} mx-1 flex-shrink\"\r\n *ngIf=\"item.icon\"></i>\r\n <span class=\"mx-1 text-truncate\">{{item.label}}</span>\r\n</ng-template>", styles: [":host{display:block;font-size:var(--ec-menu-font-size, var(--ec-font-size-action));background-color:var(--ec-menu-background-color, var(--ec-background-color))}ul{padding:0;margin:0;list-style:none;overflow-x:hidden}ul li{white-space:nowrap;padding:0}ul li.divider{border-bottom:1px solid var(--ec-border-color);padding-bottom:.25rem;margin-bottom:.25rem}ul .item-wrapper{display:flex;align-items:center;padding:0 .25rem;margin:0 .25rem;height:1.75rem;border-radius:var(--ec-border-radius)}ul .item-wrapper.is-heading{cursor:pointer;color:var(--ec-color-hint-dark);text-transform:uppercase;font-weight:var(--ec-font-weight-bold);font-size:var(--ec-font-size-label)}ul .item-wrapper:hover{background-color:var(--ec-background-color-hover)}ul .item-wrapper.is-selected{font-weight:700}ul .item-wrapper a{text-decoration:none}ul .item{color:inherit;display:flex;align-items:center;flex:1 1;min-height:0;min-width:0}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i3.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: i4.NavItemActiveDirective, selector: "[ecNavItemActive]", inputs: ["ecNavItemActive", "ecNavItemActiveExactMatch", "ecNavItemActiveQueryParams", "ecNavItemActiveUrl"], outputs: ["routerLinkActivated"] }, { kind: "component", type: i5.CollapsibleToggleComponent, selector: "ec-collapsible-toggle", inputs: ["id", "expanded", "tabindex"], outputs: ["expandedChange"] }] });
48
+ HierarchyTreeComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: HierarchyTreeComponent, selector: "ec-hierarchy-tree", inputs: { id: "id", hideRootNode: "hideRootNode", customItemTemplate: "customItemTemplate" }, outputs: { itemSelected: "itemSelected" }, usesInheritance: true, ngImport: i0, template: "<ul id=\"{{scrollContainerId}}\"\r\n class=\"flex-grow scroll-y py-1\">\r\n\r\n <li *ngIf=\"!hideRootNode && rootNode\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{rootNode?.tooltip ?? rootNode?.label}}\"\r\n [ngClass]=\"{'is-heading': rootNode?.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"rootNode?.url\"\r\n [ecNavItemActiveExactMatch]='rootNode?.isActiveExactMatch'>\r\n <a id=\"rootNode_{{rootNode?.id}}_link\"\r\n class=\"item flex-grow d-flex align-items-center\"\r\n routerLink=\"{{rootNode?.url}}\"\r\n [queryParams]=\"rootNode?.queryParams\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: rootNode }\"></ng-container>\r\n </a>\r\n </div>\r\n </li>\r\n\r\n <ng-template #hierarchyView\r\n let-items>\r\n <li *ngFor=\"let item of items; index as index; first as isFirst\"\r\n [ngClass]=\"{'divider': item.display === 'divider'}\"\r\n id=\"treeItem_{{item.id}}\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{item.tooltip ?? item.label}}\"\r\n [ngClass]=\"{'is-heading': item.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"item.url\"\r\n [ecNavItemActiveExactMatch]=\"item.isActiveExactMatch\"\r\n [ecNavItemActiveQueryParams]=\"item.queryParams\">\r\n\r\n <!-- This element provides the indentation for each level -->\r\n <span id=\"indent_{{item.id}}\"\r\n *ngIf=\"!item.noIndent\"\r\n class=\"d-block h-100\"\r\n [style.width.px]=\"(indent) * (item.level - 1) + (item.level * 4) + (item.hasChildren ? 0 : indent)\">\r\n </span>\r\n\r\n <!-- Toggle the button icon to be a spinner when item.status is pending -->\r\n <ec-collapsible-toggle id=\"toggle_{{item.id}}\"\r\n class=\"flex-shrink\"\r\n [style.width.px]=\"indent\"\r\n *ngIf=\"item.hasChildren && !item.hideToggle\"\r\n [hidden]=\"item.status === 'pending'\"\r\n [expanded]=\"item.expanded\"\r\n (expandedChange)=\"toggleItemClicked(item, $event)\">\r\n </ec-collapsible-toggle>\r\n\r\n <i class=\"ec-icon icon-loading my-1\"\r\n [hidden]=\"item.status !== 'pending'\"></i>\r\n\r\n <a id=\"treeItem_{{item.id}}_link\"\r\n class=\"item\"\r\n *ngIf=\"item.url\"\r\n (click)=\"selectItem(item)\"\r\n [routerLink]=\"item.url\"\r\n [queryParams]=\"item.queryParams\"\r\n [queryParamsHandling]=\"item.queryParamsHandling || ''\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </a>\r\n <div id=\"treeItem_{{item.id}}_heading\"\r\n class=\"item\"\r\n *ngIf=\"!item.url\"\r\n (click)=\"selectItem(item)\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </div>\r\n </div>\r\n\r\n\r\n <ul *ngIf=\"item.children.length > 0 && item.expanded\">\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: item.children }\"></ng-container>\r\n </ul>\r\n </li>\r\n </ng-template>\r\n\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: rootNode?.children }\"></ng-container>\r\n</ul>\r\n\r\n<ng-template #defaultItemTemplate\r\n let-item>\r\n <i class=\"ec-icon {{item.icon}} mx-1 flex-shrink\"\r\n [attr.title]=\"item.iconTooltip\"\r\n *ngIf=\"item.icon\"></i>\r\n <span class=\"mx-1 text-truncate\">{{item.label}}</span>\r\n</ng-template>", styles: [":host{display:block;font-size:var(--ec-menu-font-size, var(--ec-font-size-action));background-color:var(--ec-menu-background-color, var(--ec-background-color))}ul{padding:0;margin:0;list-style:none;overflow-x:hidden}ul li{white-space:nowrap;padding:0}ul li.divider{border-bottom:1px solid var(--ec-border-color);padding-bottom:.25rem;margin-bottom:.25rem}ul .item-wrapper{display:flex;align-items:center;padding:0 .25rem;margin:0 .25rem;height:1.75rem;border-radius:var(--ec-border-radius)}ul .item-wrapper.is-heading{cursor:pointer;color:var(--ec-color-hint-dark);text-transform:uppercase;font-weight:var(--ec-font-weight-bold);font-size:var(--ec-font-size-label)}ul .item-wrapper:hover{background-color:var(--ec-background-color-hover)}ul .item-wrapper.is-selected{font-weight:700}ul .item-wrapper a{text-decoration:none}ul .item{color:inherit;display:flex;align-items:center;flex:1 1;min-height:0;min-width:0}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i3.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: i4.NavItemActiveDirective, selector: "[ecNavItemActive]", inputs: ["ecNavItemActive", "ecNavItemActiveExactMatch", "ecNavItemActiveQueryParams", "ecNavItemActiveUrl"], outputs: ["routerLinkActivated"] }, { kind: "component", type: i5.CollapsibleToggleComponent, selector: "ec-collapsible-toggle", inputs: ["id", "expanded", "tabindex"], outputs: ["expandedChange"] }] });
49
49
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: HierarchyTreeComponent, decorators: [{
50
50
  type: Component,
51
- args: [{ selector: 'ec-hierarchy-tree', template: "<ul id=\"{{scrollContainerId}}\"\r\n class=\"flex-grow scroll-y py-1\">\r\n\r\n <li *ngIf=\"!hideRootNode && rootNode\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{rootNode?.tooltip ?? rootNode?.label}}\"\r\n [ngClass]=\"{'is-heading': rootNode?.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"rootNode?.url\"\r\n [ecNavItemActiveExactMatch]='rootNode?.isActiveExactMatch'>\r\n <a id=\"rootNode_{{rootNode?.id}}_link\"\r\n class=\"item flex-grow d-flex align-items-center\"\r\n routerLink=\"{{rootNode?.url}}\"\r\n [queryParams]=\"rootNode?.queryParams\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: rootNode }\"></ng-container>\r\n </a>\r\n </div>\r\n </li>\r\n\r\n <ng-template #hierarchyView\r\n let-items>\r\n <li *ngFor=\"let item of items; index as index; first as isFirst\"\r\n [ngClass]=\"{'divider': item.display === 'divider'}\"\r\n id=\"treeItem_{{item.id}}\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{item.tooltip ?? item.label}}\"\r\n [ngClass]=\"{'is-heading': item.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"item.url\"\r\n [ecNavItemActiveExactMatch]=\"item.isActiveExactMatch\"\r\n [ecNavItemActiveQueryParams]=\"item.queryParams\">\r\n\r\n <!-- This element provides the indentation for each level -->\r\n <span id=\"indent_{{item.id}}\"\r\n *ngIf=\"!item.noIndent\"\r\n class=\"d-block h-100\"\r\n [style.width.px]=\"(indent) * (item.level - 1) + (item.level * 4) + (item.hasChildren ? 0 : indent)\">\r\n </span>\r\n\r\n <!-- Toggle the button icon to be a spinner when item.status is pending -->\r\n <ec-collapsible-toggle id=\"toggle_{{item.id}}\"\r\n class=\"flex-shrink\"\r\n [style.width.px]=\"indent\"\r\n *ngIf=\"item.hasChildren && !item.hideToggle\"\r\n [hidden]=\"item.status === 'pending'\"\r\n [expanded]=\"item.expanded\"\r\n (expandedChange)=\"toggleItemClicked(item, $event)\">\r\n </ec-collapsible-toggle>\r\n\r\n <i class=\"ec-icon icon-loading my-1\"\r\n [hidden]=\"item.status !== 'pending'\"></i>\r\n\r\n <a id=\"treeItem_{{item.id}}_link\"\r\n class=\"item\"\r\n *ngIf=\"item.url\"\r\n (click)=\"selectItem(item)\"\r\n [routerLink]=\"item.url\"\r\n [queryParams]=\"item.queryParams\"\r\n [queryParamsHandling]=\"item.queryParamsHandling || ''\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </a>\r\n <div id=\"treeItem_{{item.id}}_heading\"\r\n class=\"item\"\r\n *ngIf=\"!item.url\"\r\n (click)=\"selectItem(item)\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </div>\r\n </div>\r\n\r\n\r\n <ul *ngIf=\"item.children.length > 0 && item.expanded\">\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: item.children }\"></ng-container>\r\n </ul>\r\n </li>\r\n </ng-template>\r\n\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: rootNode?.children }\"></ng-container>\r\n</ul>\r\n\r\n<ng-template #defaultItemTemplate\r\n let-item>\r\n <i class=\"ec-icon {{item.icon}} mx-1 flex-shrink\"\r\n *ngIf=\"item.icon\"></i>\r\n <span class=\"mx-1 text-truncate\">{{item.label}}</span>\r\n</ng-template>", styles: [":host{display:block;font-size:var(--ec-menu-font-size, var(--ec-font-size-action));background-color:var(--ec-menu-background-color, var(--ec-background-color))}ul{padding:0;margin:0;list-style:none;overflow-x:hidden}ul li{white-space:nowrap;padding:0}ul li.divider{border-bottom:1px solid var(--ec-border-color);padding-bottom:.25rem;margin-bottom:.25rem}ul .item-wrapper{display:flex;align-items:center;padding:0 .25rem;margin:0 .25rem;height:1.75rem;border-radius:var(--ec-border-radius)}ul .item-wrapper.is-heading{cursor:pointer;color:var(--ec-color-hint-dark);text-transform:uppercase;font-weight:var(--ec-font-weight-bold);font-size:var(--ec-font-size-label)}ul .item-wrapper:hover{background-color:var(--ec-background-color-hover)}ul .item-wrapper.is-selected{font-weight:700}ul .item-wrapper a{text-decoration:none}ul .item{color:inherit;display:flex;align-items:center;flex:1 1;min-height:0;min-width:0}\n"] }]
51
+ args: [{ selector: 'ec-hierarchy-tree', template: "<ul id=\"{{scrollContainerId}}\"\r\n class=\"flex-grow scroll-y py-1\">\r\n\r\n <li *ngIf=\"!hideRootNode && rootNode\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{rootNode?.tooltip ?? rootNode?.label}}\"\r\n [ngClass]=\"{'is-heading': rootNode?.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"rootNode?.url\"\r\n [ecNavItemActiveExactMatch]='rootNode?.isActiveExactMatch'>\r\n <a id=\"rootNode_{{rootNode?.id}}_link\"\r\n class=\"item flex-grow d-flex align-items-center\"\r\n routerLink=\"{{rootNode?.url}}\"\r\n [queryParams]=\"rootNode?.queryParams\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: rootNode }\"></ng-container>\r\n </a>\r\n </div>\r\n </li>\r\n\r\n <ng-template #hierarchyView\r\n let-items>\r\n <li *ngFor=\"let item of items; index as index; first as isFirst\"\r\n [ngClass]=\"{'divider': item.display === 'divider'}\"\r\n id=\"treeItem_{{item.id}}\">\r\n <div class=\"item-wrapper\"\r\n title=\"{{item.tooltip ?? item.label}}\"\r\n [ngClass]=\"{'is-heading': item.display === 'heading'}\"\r\n ecNavItemActive=\"is-selected\"\r\n [ecNavItemActiveUrl]=\"item.url\"\r\n [ecNavItemActiveExactMatch]=\"item.isActiveExactMatch\"\r\n [ecNavItemActiveQueryParams]=\"item.queryParams\">\r\n\r\n <!-- This element provides the indentation for each level -->\r\n <span id=\"indent_{{item.id}}\"\r\n *ngIf=\"!item.noIndent\"\r\n class=\"d-block h-100\"\r\n [style.width.px]=\"(indent) * (item.level - 1) + (item.level * 4) + (item.hasChildren ? 0 : indent)\">\r\n </span>\r\n\r\n <!-- Toggle the button icon to be a spinner when item.status is pending -->\r\n <ec-collapsible-toggle id=\"toggle_{{item.id}}\"\r\n class=\"flex-shrink\"\r\n [style.width.px]=\"indent\"\r\n *ngIf=\"item.hasChildren && !item.hideToggle\"\r\n [hidden]=\"item.status === 'pending'\"\r\n [expanded]=\"item.expanded\"\r\n (expandedChange)=\"toggleItemClicked(item, $event)\">\r\n </ec-collapsible-toggle>\r\n\r\n <i class=\"ec-icon icon-loading my-1\"\r\n [hidden]=\"item.status !== 'pending'\"></i>\r\n\r\n <a id=\"treeItem_{{item.id}}_link\"\r\n class=\"item\"\r\n *ngIf=\"item.url\"\r\n (click)=\"selectItem(item)\"\r\n [routerLink]=\"item.url\"\r\n [queryParams]=\"item.queryParams\"\r\n [queryParamsHandling]=\"item.queryParamsHandling || ''\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </a>\r\n <div id=\"treeItem_{{item.id}}_heading\"\r\n class=\"item\"\r\n *ngIf=\"!item.url\"\r\n (click)=\"selectItem(item)\">\r\n <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n </div>\r\n </div>\r\n\r\n\r\n <ul *ngIf=\"item.children.length > 0 && item.expanded\">\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: item.children }\"></ng-container>\r\n </ul>\r\n </li>\r\n </ng-template>\r\n\r\n <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: rootNode?.children }\"></ng-container>\r\n</ul>\r\n\r\n<ng-template #defaultItemTemplate\r\n let-item>\r\n <i class=\"ec-icon {{item.icon}} mx-1 flex-shrink\"\r\n [attr.title]=\"item.iconTooltip\"\r\n *ngIf=\"item.icon\"></i>\r\n <span class=\"mx-1 text-truncate\">{{item.label}}</span>\r\n</ng-template>", styles: [":host{display:block;font-size:var(--ec-menu-font-size, var(--ec-font-size-action));background-color:var(--ec-menu-background-color, var(--ec-background-color))}ul{padding:0;margin:0;list-style:none;overflow-x:hidden}ul li{white-space:nowrap;padding:0}ul li.divider{border-bottom:1px solid var(--ec-border-color);padding-bottom:.25rem;margin-bottom:.25rem}ul .item-wrapper{display:flex;align-items:center;padding:0 .25rem;margin:0 .25rem;height:1.75rem;border-radius:var(--ec-border-radius)}ul .item-wrapper.is-heading{cursor:pointer;color:var(--ec-color-hint-dark);text-transform:uppercase;font-weight:var(--ec-font-weight-bold);font-size:var(--ec-font-size-label)}ul .item-wrapper:hover{background-color:var(--ec-background-color-hover)}ul .item-wrapper.is-selected{font-weight:700}ul .item-wrapper a{text-decoration:none}ul .item{color:inherit;display:flex;align-items:center;flex:1 1;min-height:0;min-width:0}\n"] }]
52
52
  }], ctorParameters: function () { return [{ type: i1.ScrollService }, { type: i0.Injector }]; }, propDecorators: { id: [{
53
53
  type: Input
54
54
  }], hideRootNode: [{
@@ -58,4 +58,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
58
58
  }], itemSelected: [{
59
59
  type: Output
60
60
  }] } });
61
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchy-tree.component.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/display/hierarchy/hierarchy-tree/hierarchy-tree.component.ts","../../../../../../../projects/components/src/lib/display/hierarchy/hierarchy-tree/hierarchy-tree.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAY,KAAK,EAAU,MAAM,EAAe,MAAM,eAAe,CAAC;AACxF,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAE/B,OAAO,EAAE,aAAa,EAAiB,MAAM,mBAAmB,CAAC;;;;;;;AAOjE,MAAM,OAAO,sBAAuB,SAAQ,aAAa;IAoBvD,YACU,aAA4B,EACpC,QAAkB;QAChB,KAAK,CAAC,QAAQ,CAAC,CAAC;QAFV,kBAAa,GAAb,aAAa,CAAe;QAnBtC,+FAA+F;QAC/E,OAAE,GAAW,eAAe,CAAC;QAQ7C,iEAAiE;QAChD,iBAAY,GAA2B,IAAI,OAAO,EAAE,CAAC;QAEtE,4GAA4G;QACrG,WAAM,GAAG,EAAE,CAAC;QAEnB,uHAAuH;QAChH,sBAAiB,GAAG,EAAE,CAAC;IAKT,CAAC;IAEf,QAAQ;QACb,IAAI,CAAC,iBAAiB,GAAG,GAAG,IAAI,CAAC,EAAE,OAAO,CAAC;IAC7C,CAAC;IAED;;;OAGG;IACI,oBAAoB;QACzB,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;YACrB,MAAM,YAAY,GAAG,2BAA2B,CAAC;YACjD,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,iBAAiB,EAAE,EAAE,YAAY,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,UAAU,CAAC,IAAmB;QACnC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC7B,IAAI,IAAI,CAAC,OAAO,KAAK,SAAS,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YAClD,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAC9C;IACH,CAAC;;mHAjDU,sBAAsB;uGAAtB,sBAAsB,yNCVnC,83HAoFc;2FD1ED,sBAAsB;kBALlC,SAAS;+BACE,mBAAmB;2HAOb,EAAE;sBAAjB,KAAK;gBAGU,YAAY;sBAA3B,KAAK;gBAGU,kBAAkB;sBAAjC,KAAK;gBAGW,YAAY;sBAA5B,MAAM","sourcesContent":["import { Component, Injector, Input, OnInit, Output, TemplateRef } from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { ScrollService } from '../../../core/scroll.service';\r\nimport { HierarchyBase, HierarchyItem } from '../hierarchy-base';\r\n\r\n@Component({\r\n  selector: 'ec-hierarchy-tree',\r\n  templateUrl: './hierarchy-tree.component.html',\r\n  styleUrls: ['./hierarchy-tree.component.scss']\r\n})\r\nexport class HierarchyTreeComponent extends HierarchyBase implements OnInit {\r\n\r\n  /** The value of the id attribute of the tree's root HTMLUListElement, suffixed with '_root' */\r\n  @Input() public id: string = 'HierarchyTree';\r\n\r\n  /** Hide the root node */\r\n  @Input() public hideRootNode?: boolean;\r\n\r\n  /** Custom item template to use in place of the default one */\r\n  @Input() public customItemTemplate?: TemplateRef<any>;\r\n\r\n  /** Emits a HierarchyItem whenever one is selected by clicking */\r\n  @Output() public itemSelected: Subject<HierarchyItem> = new Subject();\r\n\r\n  /** The width of the spacing before a tree node, multipled by a HierarchyItem's level. Value is in pixels */\r\n  public indent = 16;\r\n\r\n  /** The value of the id attribute of the tree's scroll container element, used to scroll a selected item into view.  */\r\n  public scrollContainerId = '';\r\n\r\n  constructor(\r\n    private scrollService: ScrollService,\r\n    injector: Injector,\r\n  ) { super(injector); }\r\n\r\n  public ngOnInit(): void {\r\n    this.scrollContainerId = `${this.id}_root`;\r\n  }\r\n\r\n  /**\r\n   * Scroll to the item currently marked as is-selected. Wait a tick for the\r\n   * NavItemActiveDirective to update the selected class.\r\n   */\r\n  public scrollToSelectedItem(): void {\r\n    window.setTimeout(() => {\r\n      const linkSelector = `.item-wrapper.is-selected`;\r\n      this.scrollService.scrollToItem(`#${this.scrollContainerId}`, linkSelector);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * When a HierarchyItem is selected, update the value of the activeUrl property with the value\r\n   * of that item's url property and emit the item to any subscribers.\r\n   */\r\n  public selectItem(item: HierarchyItem): void {\r\n    this.itemSelected.next(item);\r\n    if (item.display === 'heading' && !item.hideToggle) {\r\n      this.toggleItemClicked(item, !item.expanded);\r\n    }\r\n  }\r\n}\r\n","<ul id=\"{{scrollContainerId}}\"\r\n    class=\"flex-grow scroll-y py-1\">\r\n\r\n  <li *ngIf=\"!hideRootNode && rootNode\">\r\n    <div class=\"item-wrapper\"\r\n         title=\"{{rootNode?.tooltip ?? rootNode?.label}}\"\r\n         [ngClass]=\"{'is-heading': rootNode?.display === 'heading'}\"\r\n         ecNavItemActive=\"is-selected\"\r\n         [ecNavItemActiveUrl]=\"rootNode?.url\"\r\n         [ecNavItemActiveExactMatch]='rootNode?.isActiveExactMatch'>\r\n      <a id=\"rootNode_{{rootNode?.id}}_link\"\r\n         class=\"item flex-grow d-flex align-items-center\"\r\n         routerLink=\"{{rootNode?.url}}\"\r\n         [queryParams]=\"rootNode?.queryParams\">\r\n        <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: rootNode }\"></ng-container>\r\n      </a>\r\n    </div>\r\n  </li>\r\n\r\n  <ng-template #hierarchyView\r\n               let-items>\r\n    <li *ngFor=\"let item of items; index as index; first as isFirst\"\r\n        [ngClass]=\"{'divider': item.display === 'divider'}\"\r\n        id=\"treeItem_{{item.id}}\">\r\n      <div class=\"item-wrapper\"\r\n           title=\"{{item.tooltip ?? item.label}}\"\r\n           [ngClass]=\"{'is-heading': item.display === 'heading'}\"\r\n           ecNavItemActive=\"is-selected\"\r\n           [ecNavItemActiveUrl]=\"item.url\"\r\n           [ecNavItemActiveExactMatch]=\"item.isActiveExactMatch\"\r\n           [ecNavItemActiveQueryParams]=\"item.queryParams\">\r\n\r\n        <!-- This element provides the indentation for each level -->\r\n        <span id=\"indent_{{item.id}}\"\r\n              *ngIf=\"!item.noIndent\"\r\n              class=\"d-block h-100\"\r\n              [style.width.px]=\"(indent) * (item.level - 1) + (item.level * 4) + (item.hasChildren ? 0 : indent)\">\r\n        </span>\r\n\r\n        <!-- Toggle the button icon to be a spinner when item.status is pending -->\r\n        <ec-collapsible-toggle id=\"toggle_{{item.id}}\"\r\n                               class=\"flex-shrink\"\r\n                               [style.width.px]=\"indent\"\r\n                               *ngIf=\"item.hasChildren && !item.hideToggle\"\r\n                               [hidden]=\"item.status === 'pending'\"\r\n                               [expanded]=\"item.expanded\"\r\n                               (expandedChange)=\"toggleItemClicked(item, $event)\">\r\n        </ec-collapsible-toggle>\r\n\r\n        <i class=\"ec-icon icon-loading my-1\"\r\n           [hidden]=\"item.status !== 'pending'\"></i>\r\n\r\n        <a id=\"treeItem_{{item.id}}_link\"\r\n           class=\"item\"\r\n           *ngIf=\"item.url\"\r\n           (click)=\"selectItem(item)\"\r\n           [routerLink]=\"item.url\"\r\n           [queryParams]=\"item.queryParams\"\r\n           [queryParamsHandling]=\"item.queryParamsHandling || ''\">\r\n           <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n        </a>\r\n        <div id=\"treeItem_{{item.id}}_heading\"\r\n             class=\"item\"\r\n             *ngIf=\"!item.url\"\r\n             (click)=\"selectItem(item)\">\r\n             <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n        </div>\r\n      </div>\r\n\r\n\r\n      <ul *ngIf=\"item.children.length > 0 && item.expanded\">\r\n        <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: item.children }\"></ng-container>\r\n      </ul>\r\n    </li>\r\n  </ng-template>\r\n\r\n  <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: rootNode?.children }\"></ng-container>\r\n</ul>\r\n\r\n<ng-template #defaultItemTemplate\r\n             let-item>\r\n  <i class=\"ec-icon {{item.icon}} mx-1 flex-shrink\"\r\n     *ngIf=\"item.icon\"></i>\r\n  <span class=\"mx-1 text-truncate\">{{item.label}}</span>\r\n</ng-template>"]}
61
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchy-tree.component.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/display/hierarchy/hierarchy-tree/hierarchy-tree.component.ts","../../../../../../../projects/components/src/lib/display/hierarchy/hierarchy-tree/hierarchy-tree.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAY,KAAK,EAAU,MAAM,EAAe,MAAM,eAAe,CAAC;AACxF,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAE/B,OAAO,EAAE,aAAa,EAAiB,MAAM,mBAAmB,CAAC;;;;;;;AAOjE,MAAM,OAAO,sBAAuB,SAAQ,aAAa;IAoBvD,YACU,aAA4B,EACpC,QAAkB;QAChB,KAAK,CAAC,QAAQ,CAAC,CAAC;QAFV,kBAAa,GAAb,aAAa,CAAe;QAnBtC,+FAA+F;QAC/E,OAAE,GAAW,eAAe,CAAC;QAQ7C,iEAAiE;QAChD,iBAAY,GAA2B,IAAI,OAAO,EAAE,CAAC;QAEtE,4GAA4G;QACrG,WAAM,GAAG,EAAE,CAAC;QAEnB,uHAAuH;QAChH,sBAAiB,GAAG,EAAE,CAAC;IAKT,CAAC;IAEf,QAAQ;QACb,IAAI,CAAC,iBAAiB,GAAG,GAAG,IAAI,CAAC,EAAE,OAAO,CAAC;IAC7C,CAAC;IAED;;;OAGG;IACI,oBAAoB;QACzB,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;YACrB,MAAM,YAAY,GAAG,2BAA2B,CAAC;YACjD,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,iBAAiB,EAAE,EAAE,YAAY,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,UAAU,CAAC,IAAmB;QACnC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC7B,IAAI,IAAI,CAAC,OAAO,KAAK,SAAS,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YAClD,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAC9C;IACH,CAAC;;mHAjDU,sBAAsB;uGAAtB,sBAAsB,yNCVnC,w6HAqFc;2FD3ED,sBAAsB;kBALlC,SAAS;+BACE,mBAAmB;2HAOb,EAAE;sBAAjB,KAAK;gBAGU,YAAY;sBAA3B,KAAK;gBAGU,kBAAkB;sBAAjC,KAAK;gBAGW,YAAY;sBAA5B,MAAM","sourcesContent":["import { Component, Injector, Input, OnInit, Output, TemplateRef } from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { ScrollService } from '../../../core/scroll.service';\r\nimport { HierarchyBase, HierarchyItem } from '../hierarchy-base';\r\n\r\n@Component({\r\n  selector: 'ec-hierarchy-tree',\r\n  templateUrl: './hierarchy-tree.component.html',\r\n  styleUrls: ['./hierarchy-tree.component.scss']\r\n})\r\nexport class HierarchyTreeComponent extends HierarchyBase implements OnInit {\r\n\r\n  /** The value of the id attribute of the tree's root HTMLUListElement, suffixed with '_root' */\r\n  @Input() public id: string = 'HierarchyTree';\r\n\r\n  /** Hide the root node */\r\n  @Input() public hideRootNode?: boolean;\r\n\r\n  /** Custom item template to use in place of the default one */\r\n  @Input() public customItemTemplate?: TemplateRef<any>;\r\n\r\n  /** Emits a HierarchyItem whenever one is selected by clicking */\r\n  @Output() public itemSelected: Subject<HierarchyItem> = new Subject();\r\n\r\n  /** The width of the spacing before a tree node, multipled by a HierarchyItem's level. Value is in pixels */\r\n  public indent = 16;\r\n\r\n  /** The value of the id attribute of the tree's scroll container element, used to scroll a selected item into view.  */\r\n  public scrollContainerId = '';\r\n\r\n  constructor(\r\n    private scrollService: ScrollService,\r\n    injector: Injector,\r\n  ) { super(injector); }\r\n\r\n  public ngOnInit(): void {\r\n    this.scrollContainerId = `${this.id}_root`;\r\n  }\r\n\r\n  /**\r\n   * Scroll to the item currently marked as is-selected. Wait a tick for the\r\n   * NavItemActiveDirective to update the selected class.\r\n   */\r\n  public scrollToSelectedItem(): void {\r\n    window.setTimeout(() => {\r\n      const linkSelector = `.item-wrapper.is-selected`;\r\n      this.scrollService.scrollToItem(`#${this.scrollContainerId}`, linkSelector);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * When a HierarchyItem is selected, update the value of the activeUrl property with the value\r\n   * of that item's url property and emit the item to any subscribers.\r\n   */\r\n  public selectItem(item: HierarchyItem): void {\r\n    this.itemSelected.next(item);\r\n    if (item.display === 'heading' && !item.hideToggle) {\r\n      this.toggleItemClicked(item, !item.expanded);\r\n    }\r\n  }\r\n}\r\n","<ul id=\"{{scrollContainerId}}\"\r\n    class=\"flex-grow scroll-y py-1\">\r\n\r\n  <li *ngIf=\"!hideRootNode && rootNode\">\r\n    <div class=\"item-wrapper\"\r\n         title=\"{{rootNode?.tooltip ?? rootNode?.label}}\"\r\n         [ngClass]=\"{'is-heading': rootNode?.display === 'heading'}\"\r\n         ecNavItemActive=\"is-selected\"\r\n         [ecNavItemActiveUrl]=\"rootNode?.url\"\r\n         [ecNavItemActiveExactMatch]='rootNode?.isActiveExactMatch'>\r\n      <a id=\"rootNode_{{rootNode?.id}}_link\"\r\n         class=\"item flex-grow d-flex align-items-center\"\r\n         routerLink=\"{{rootNode?.url}}\"\r\n         [queryParams]=\"rootNode?.queryParams\">\r\n        <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: rootNode }\"></ng-container>\r\n      </a>\r\n    </div>\r\n  </li>\r\n\r\n  <ng-template #hierarchyView\r\n               let-items>\r\n    <li *ngFor=\"let item of items; index as index; first as isFirst\"\r\n        [ngClass]=\"{'divider': item.display === 'divider'}\"\r\n        id=\"treeItem_{{item.id}}\">\r\n      <div class=\"item-wrapper\"\r\n           title=\"{{item.tooltip ?? item.label}}\"\r\n           [ngClass]=\"{'is-heading': item.display === 'heading'}\"\r\n           ecNavItemActive=\"is-selected\"\r\n           [ecNavItemActiveUrl]=\"item.url\"\r\n           [ecNavItemActiveExactMatch]=\"item.isActiveExactMatch\"\r\n           [ecNavItemActiveQueryParams]=\"item.queryParams\">\r\n\r\n        <!-- This element provides the indentation for each level -->\r\n        <span id=\"indent_{{item.id}}\"\r\n              *ngIf=\"!item.noIndent\"\r\n              class=\"d-block h-100\"\r\n              [style.width.px]=\"(indent) * (item.level - 1) + (item.level * 4) + (item.hasChildren ? 0 : indent)\">\r\n        </span>\r\n\r\n        <!-- Toggle the button icon to be a spinner when item.status is pending -->\r\n        <ec-collapsible-toggle id=\"toggle_{{item.id}}\"\r\n                               class=\"flex-shrink\"\r\n                               [style.width.px]=\"indent\"\r\n                               *ngIf=\"item.hasChildren && !item.hideToggle\"\r\n                               [hidden]=\"item.status === 'pending'\"\r\n                               [expanded]=\"item.expanded\"\r\n                               (expandedChange)=\"toggleItemClicked(item, $event)\">\r\n        </ec-collapsible-toggle>\r\n\r\n        <i class=\"ec-icon icon-loading my-1\"\r\n           [hidden]=\"item.status !== 'pending'\"></i>\r\n\r\n        <a id=\"treeItem_{{item.id}}_link\"\r\n           class=\"item\"\r\n           *ngIf=\"item.url\"\r\n           (click)=\"selectItem(item)\"\r\n           [routerLink]=\"item.url\"\r\n           [queryParams]=\"item.queryParams\"\r\n           [queryParamsHandling]=\"item.queryParamsHandling || ''\">\r\n           <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n        </a>\r\n        <div id=\"treeItem_{{item.id}}_heading\"\r\n             class=\"item\"\r\n             *ngIf=\"!item.url\"\r\n             (click)=\"selectItem(item)\">\r\n             <ng-container *ngTemplateOutlet=\"customItemTemplate || defaultItemTemplate; context: { $implicit: item }\"></ng-container>\r\n        </div>\r\n      </div>\r\n\r\n\r\n      <ul *ngIf=\"item.children.length > 0 && item.expanded\">\r\n        <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: item.children }\"></ng-container>\r\n      </ul>\r\n    </li>\r\n  </ng-template>\r\n\r\n  <ng-container *ngTemplateOutlet=\"hierarchyView; context:{ $implicit: rootNode?.children }\"></ng-container>\r\n</ul>\r\n\r\n<ng-template #defaultItemTemplate\r\n             let-item>\r\n  <i class=\"ec-icon {{item.icon}} mx-1 flex-shrink\"\r\n     [attr.title]=\"item.iconTooltip\"\r\n     *ngIf=\"item.icon\"></i>\r\n  <span class=\"mx-1 text-truncate\">{{item.label}}</span>\r\n</ng-template>"]}
@@ -1,5 +1,5 @@
1
1
  import { Component, ContentChildren, EventEmitter, HostBinding, Input, Output, ViewEncapsulation } from '@angular/core';
2
- import { isEqual } from 'lodash';
2
+ import { cloneDeep, isEqual } from 'lodash';
3
3
  import { Subject } from 'rxjs';
4
4
  import { takeUntil } from 'rxjs/operators';
5
5
  import { ResizableColumnComponent } from './resizable-column.component';
@@ -77,7 +77,7 @@ export class TableComponent {
77
77
  if (this.sortable && changes.sort && changes.sort.currentValue && !changes.sort.firstChange) {
78
78
  const newValue = changes.sort.currentValue;
79
79
  if (!this.currentSort || !isEqual(newValue, this.currentSort)) {
80
- this.onSort(newValue.field, newValue.direction);
80
+ this.onSort(newValue);
81
81
  }
82
82
  }
83
83
  }
@@ -126,7 +126,14 @@ export class TableComponent {
126
126
  if (th.classList.contains('is-resizable') && event.target && event.target.classList.contains('handle')) {
127
127
  return;
128
128
  }
129
- this.onSort(field);
129
+ let direction;
130
+ if (th.classList.contains('is-sorted-desc')) {
131
+ direction = 'asc';
132
+ }
133
+ else {
134
+ direction = 'desc';
135
+ }
136
+ this.onSort({ field, direction });
130
137
  });
131
138
  // Store unlisten function to remove listener when component is destroyed
132
139
  this.unlisteners.push(unlisten);
@@ -138,7 +145,7 @@ export class TableComponent {
138
145
  });
139
146
  // Set the initial sort
140
147
  if (this.sort) {
141
- this.onSort(this.sort.field, this.sort.direction);
148
+ this.onSort(this.sort);
142
149
  }
143
150
  }
144
151
  /**
@@ -147,32 +154,29 @@ export class TableComponent {
147
154
  * Emit the current field and direction of the sorted column
148
155
  * @param field
149
156
  */
150
- onSort(field, direction) {
151
- const th = this.sortableCols[field];
152
- if (!th) {
153
- return;
154
- }
155
- if (this.currentSort && this.currentSort.field !== field) {
156
- this.renderer.removeClass(this.sortableCols[this.currentSort.field], `is-sorted-${this.currentSort.direction}`);
157
- }
158
- if (th.classList.contains('is-sorted-desc')) {
159
- this.renderer.removeClass(th, 'is-sorted-desc');
160
- if (!direction) {
161
- direction = 'asc';
162
- }
157
+ onSort(sort) {
158
+ this.clearSort();
159
+ if (Array.isArray(sort)) {
160
+ sort.forEach(s => this.applySort(s));
163
161
  }
164
162
  else {
165
- if (th.classList.contains('is-sorted-asc')) {
166
- this.renderer.removeClass(th, 'is-sorted-asc');
167
- }
168
- if (!direction) {
169
- direction = 'desc';
170
- }
163
+ this.applySort(sort);
164
+ }
165
+ this.currentSort = cloneDeep(sort);
166
+ this.sortChange.emit(this.currentSort);
167
+ }
168
+ clearSort() {
169
+ Object.values(this.sortableCols).forEach(th => {
170
+ this.renderer.removeClass(th, `is-sorted-asc`);
171
+ this.renderer.removeClass(th, `is-sorted-desc`);
172
+ });
173
+ }
174
+ applySort(sort) {
175
+ const th = this.sortableCols[sort.field];
176
+ if (!th) {
177
+ return;
171
178
  }
172
- this.renderer.addClass(th, `is-sorted-${direction}`);
173
- const sortEvent = { field: field, direction: direction };
174
- this.currentSort = sortEvent;
175
- this.sortChange.emit(sortEvent);
179
+ this.renderer.addClass(th, `is-sorted-${sort.direction}`);
176
180
  }
177
181
  initSelectable() {
178
182
  this.selectionContext.rowCheckboxes.valueChanges.pipe(takeUntil(this.destroyed)).subscribe((value) => {
@@ -241,4 +245,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
241
245
  type: Input,
242
246
  args: ['resizableColumns']
243
247
  }] } });
244
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/table/table.component.ts","../../../../../../projects/components/src/lib/display/table/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterContentInit, AfterViewInit, Component, ContentChildren, ElementRef, EventEmitter, HostBinding, Input, OnChanges, OnDestroy, Output, QueryList, Renderer2, SimpleChanges, TemplateRef, ViewEncapsulation } from '@angular/core';\r\nimport { isEqual } from 'lodash';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ResizableColumnComponent } from './resizable-column.component';\r\nimport { TableMasterRowComponent } from './table-master-row.component';\r\nimport { TableSelectableRowContext } from './table-selectable-row.component';\r\n\r\nexport type TableSortDirection = 'desc' | 'asc';\r\nexport interface TableSort {\r\n  field: string;\r\n  direction: TableSortDirection;\r\n}\r\n\r\n@Component({\r\n  selector: 'ec-table',\r\n  templateUrl: './table.component.html',\r\n  styleUrls: ['./table.component.scss'],\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class TableComponent implements AfterViewInit, OnDestroy, OnChanges, AfterContentInit{\r\n  @HostBinding('attr.id')\r\n  @Input() id = '';\r\n\r\n  /**\r\n   * Enables table scrolling with a sticky header and footer. Use\r\n   * <thead> and <tfoot> elements for styles to be applied correctly. \r\n   */\r\n  @HostBinding('class.is-scrollable')\r\n  @Input() scrollable = false;\r\n  \r\n  /**\r\n   * Enables resizable table columns. Apply the 'is-resizable' class\r\n   * to the <th> of a table column than can be resized.\r\n   */\r\n  @HostBinding('class.is-resizable')\r\n  @Input() resizable = false;\r\n  \r\n  /** \r\n   * Turns on condensed table styling. By default a condensed table\r\n   * has no row borders. To show borders, apply the 'has-borders' class\r\n   * to the <ec-table> element\r\n   */\r\n  @HostBinding('class.is-condensed')\r\n  @Input() condensed = false;\r\n  \r\n  /** \r\n   * Enables sorting by clicking on table columns. To make a column sortable\r\n   * apply the `data-sortable` attribute to the columns `<th>` element with \r\n   * the value of the field used to sort the column. When a column is clicked\r\n   * the fields and current sort direction is emitted from the `sortChange`\r\n   * output\r\n   * \r\n   * @example\r\n   * ```html\r\n   * <ec-table [sortable]=\"true\" [sort]=\"currentSort\" (sortChange)=\"onSort($event)\">\r\n   *  <thead>\r\n   *    <th data-sortable=\"name\">Name</th>\r\n   *    <th data-sortable=\"qty\">Qty.</th>\r\n   *    <th data-sortable=\"price\">Price</th>\r\n   *  </thead>\r\n   *  ...\r\n   * </ec-table>\r\n   * ```\r\n   */\r\n  @HostBinding('class.is-sortable')\r\n  @Input() sortable = false;\r\n  \r\n  /**\r\n   * Shared context of TableSelectableRowComponents in the table. Will turn on highlighting on row hover.\r\n   * Add the `TableSelectableRowComponent` to the table content to add checkboxes to each row and assign a \r\n   * TemplateRef to the `[selectionToolbarTemplate]` input that includes actions to take on selected items \r\n   * that will appear over the table header when 1 or more rows are selected.\r\n   */\r\n  @Input() selectionContext?: TableSelectableRowContext;\r\n\r\n  /** Include actions to display in a bar over the table header when 1 or more rows are selected. Requires `[selectionContext]` to be defined */\r\n  @Input() selectionToolbarTemplate?: TemplateRef<any>;\r\n  \r\n  /** Set to true if selectionContext is defined */\r\n  @HostBinding('class.is-selectable')\r\n  @Input() selectable = false;\r\n\r\n  /** Hides/shows the selection toolbar template when 1 or more rows is selected */\r\n  hasSelection = false\r\n\r\n  /**\r\n   * Enables form table styling to allow form controls inside table cells.\r\n   */\r\n  @HostBinding('class.is-form-table')\r\n  @Input() isForm = false;\r\n  \r\n  /**\r\n   * The current sort field and direction ('desc' or 'asc')\r\n   */\r\n  @Input() sort?: TableSort;\r\n\r\n  /**\r\n   * Emits the new sort value whenever a sortable column is clicked\r\n   */\r\n  @Output() sortChange: EventEmitter<TableSort> = new EventEmitter();\r\n\r\n  /**\r\n   * True if a TableMasterRowComponent exists in the table content\r\n   */\r\n  @HostBinding('class.is-master-detail') hasMasterDetailRows = false;\r\n\r\n  @ContentChildren(TableMasterRowComponent, {descendants: true}) masterRows!: QueryList<TableMasterRowComponent>;\r\n\r\n  public resizableColumns!: QueryList<ResizableColumnComponent>;\r\n  @ContentChildren(ResizableColumnComponent, {descendants: true}) _resizableColumns!: QueryList<ResizableColumnComponent>;\r\n\r\n  @Input('resizableColumns') searchableTableResizableColumns?: QueryList<ResizableColumnComponent>;\r\n\r\n  /**\r\n   * Unlisten function returned by `renderer.listen()` that are called\r\n   * OnDestroy to clean up all listeners\r\n   */\r\n  private unlisteners: (() => void)[] = [];\r\n  \r\n  /** The current sort value */\r\n  private currentSort?: TableSort;\r\n  \r\n  /** A map of sortable `<th>` elements */\r\n  private sortableCols: {[field: string]: HTMLTableHeaderCellElement} = {};\r\n\r\n  private destroyed: Subject<void> = new Subject();\r\n  \r\n  constructor(private renderer: Renderer2, public el: ElementRef) {}\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (this.sortable && changes.sort && changes.sort.currentValue && !changes.sort.firstChange) {\r\n      const newValue: TableSort = changes.sort.currentValue;\r\n      if (!this.currentSort || !isEqual(newValue, this.currentSort)){\r\n        this.onSort(newValue.field, newValue.direction);\r\n      }\r\n    }\r\n  }\r\n\r\n  ngAfterContentInit(): void {\r\n    this.hasMasterDetailRows = !!this.masterRows.length;\r\n    \r\n    this.masterRows.changes.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe(() => {\r\n      this.hasMasterDetailRows = !!this.masterRows.length;\r\n    });\r\n\r\n    this.resizableColumns = this.searchableTableResizableColumns || this._resizableColumns;\r\n  }\r\n  \r\n  ngAfterViewInit(): void {\r\n    if (this.sortable) {\r\n      this.initSortable();\r\n    }\r\n\r\n    if (this.selectable) {\r\n      this.initSelectable();\r\n    }\r\n  }\r\n  \r\n  ngOnDestroy(): void {\r\n    if (this.unlisteners.length) {\r\n      this.unlisteners.forEach(unlisten => unlisten());\r\n    }\r\n    \r\n    this.destroyed.next();\r\n    this.destroyed.unsubscribe();\r\n  }\r\n\r\n  /**\r\n   * Get all elements inside the host element that match a selector \r\n   * @param selector - Any valid selector for [querySelectorAll](https://developer.mozilla.org/en-US/docs/Web/API/Document/querySelectorAll)\r\n   */\r\n  private queryElements<T>(selector: string): T[] {\r\n    const table = this.el.nativeElement;\r\n    return Array.from(table.querySelectorAll(selector));\r\n  }\r\n\r\n  /**\r\n   * Store a `<th>` for each sortable column and attach a click\r\n   * event listener. \r\n   */\r\n  private initSortable(): void {\r\n    this.queryElements<HTMLTableHeaderCellElement>('th').forEach(th => {\r\n      if (th.hasAttribute('data-sortfield')) {\r\n        \r\n        const field = th.getAttribute('data-sortfield');\r\n        if (field) {\r\n          this.sortableCols[field] = th;\r\n          const unlisten = this.renderer.listen(th, 'click', (event: MouseEvent) => {\r\n\r\n            // Do not sort if the event target was the resizable handle\r\n            if (th.classList.contains('is-resizable') && event.target && (<HTMLElement>event.target).classList.contains('handle')) {\r\n              return;\r\n            }\r\n\r\n            this.onSort(field);\r\n          });\r\n  \r\n          // Store unlisten function to remove listener when component is destroyed\r\n          this.unlisteners.push(unlisten);\r\n        } else {\r\n          console.error(`data-sortfield attribute was not assigned a value.`, th);\r\n        }\r\n      }\r\n    });\r\n\r\n    // Set the initial sort\r\n    if (this.sort) {\r\n      this.onSort(this.sort.field, this.sort.direction);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Find the sorted column and apply the desc or asc class in order\r\n   * to show the correct sort icon. If another field was sorted, reset it.\r\n   * Emit the current field and direction of the sorted column\r\n   * @param field \r\n   */\r\n  private onSort(field: string, direction?: TableSortDirection): void {\r\n    const th = this.sortableCols[field];\r\n    if (!th) { return; }\r\n    \r\n    if (this.currentSort && this.currentSort.field !== field) {\r\n      this.renderer.removeClass(this.sortableCols[this.currentSort.field], `is-sorted-${this.currentSort.direction}`);\r\n    } \r\n    \r\n    if (th.classList.contains('is-sorted-desc')) {\r\n      this.renderer.removeClass(th, 'is-sorted-desc');\r\n\r\n      if (!direction) {\r\n        direction = 'asc';\r\n      }\r\n\r\n    } else {\r\n      if (th.classList.contains('is-sorted-asc')) {\r\n        this.renderer.removeClass(th, 'is-sorted-asc');\r\n      }\r\n\r\n      if (!direction) {\r\n        direction = 'desc';\r\n      }\r\n    }\r\n\r\n    this.renderer.addClass(th, `is-sorted-${direction}`);\r\n\r\n    const sortEvent = {field: field, direction: direction};\r\n    this.currentSort = sortEvent;\r\n    this.sortChange.emit(sortEvent);\r\n  }\r\n\r\n  private initSelectable(): void {\r\n    this.selectionContext!.rowCheckboxes.valueChanges.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe((value: boolean[]) => {\r\n      this.hasSelection = value.some(v => v);\r\n    });\r\n  }\r\n}","<div class=\"selectable-table-toolbar\"\r\n     *ngIf=\"hasSelection && selectionToolbarTemplate\">\r\n  <ng-container *ngTemplateOutlet=\"selectionToolbarTemplate\"></ng-container>\r\n</div>\r\n<div id=\"{{id}}-scroll-container\" \r\n     class=\"table-scroll-container\">\r\n  <table class=\"main-table\"\r\n         id=\"{{id}}\"\r\n         [ecResizableTable]=\"resizable\"\r\n         [containerEl]=\"el\"\r\n         [sortableTable]=\"sortable\"\r\n         [resizableColumns]=\"resizableColumns\">\r\n    <ng-content></ng-content>\r\n  </table>\r\n</div>"]}
248
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/table/table.component.ts","../../../../../../projects/components/src/lib/display/table/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterContentInit, AfterViewInit, Component, ContentChildren, ElementRef, EventEmitter, HostBinding, Input, OnChanges, OnDestroy, Output, QueryList, Renderer2, SimpleChanges, TemplateRef, ViewEncapsulation } from '@angular/core';\r\nimport { cloneDeep, isEqual } from 'lodash';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ResizableColumnComponent } from './resizable-column.component';\r\nimport { TableMasterRowComponent } from './table-master-row.component';\r\nimport { TableSelectableRowContext } from './table-selectable-row.component';\r\n\r\nexport type TableSortDirection = 'desc' | 'asc';\r\nexport interface TableSort {\r\n  field: string;\r\n  direction: TableSortDirection;\r\n}\r\n\r\n@Component({\r\n  selector: 'ec-table',\r\n  templateUrl: './table.component.html',\r\n  styleUrls: ['./table.component.scss'],\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class TableComponent implements AfterViewInit, OnDestroy, OnChanges, AfterContentInit{\r\n  @HostBinding('attr.id')\r\n  @Input() id = '';\r\n\r\n  /**\r\n   * Enables table scrolling with a sticky header and footer. Use\r\n   * <thead> and <tfoot> elements for styles to be applied correctly. \r\n   */\r\n  @HostBinding('class.is-scrollable')\r\n  @Input() scrollable = false;\r\n  \r\n  /**\r\n   * Enables resizable table columns. Apply the 'is-resizable' class\r\n   * to the <th> of a table column than can be resized.\r\n   */\r\n  @HostBinding('class.is-resizable')\r\n  @Input() resizable = false;\r\n  \r\n  /** \r\n   * Turns on condensed table styling. By default a condensed table\r\n   * has no row borders. To show borders, apply the 'has-borders' class\r\n   * to the <ec-table> element\r\n   */\r\n  @HostBinding('class.is-condensed')\r\n  @Input() condensed = false;\r\n  \r\n  /** \r\n   * Enables sorting by clicking on table columns. To make a column sortable\r\n   * apply the `data-sortable` attribute to the columns `<th>` element with \r\n   * the value of the field used to sort the column. When a column is clicked\r\n   * the fields and current sort direction is emitted from the `sortChange`\r\n   * output\r\n   * \r\n   * @example\r\n   * ```html\r\n   * <ec-table [sortable]=\"true\" [sort]=\"currentSort\" (sortChange)=\"onSort($event)\">\r\n   *  <thead>\r\n   *    <th data-sortable=\"name\">Name</th>\r\n   *    <th data-sortable=\"qty\">Qty.</th>\r\n   *    <th data-sortable=\"price\">Price</th>\r\n   *  </thead>\r\n   *  ...\r\n   * </ec-table>\r\n   * ```\r\n   */\r\n  @HostBinding('class.is-sortable')\r\n  @Input() sortable = false;\r\n  \r\n  /**\r\n   * Shared context of TableSelectableRowComponents in the table. Will turn on highlighting on row hover.\r\n   * Add the `TableSelectableRowComponent` to the table content to add checkboxes to each row and assign a \r\n   * TemplateRef to the `[selectionToolbarTemplate]` input that includes actions to take on selected items \r\n   * that will appear over the table header when 1 or more rows are selected.\r\n   */\r\n  @Input() selectionContext?: TableSelectableRowContext;\r\n\r\n  /** Include actions to display in a bar over the table header when 1 or more rows are selected. Requires `[selectionContext]` to be defined */\r\n  @Input() selectionToolbarTemplate?: TemplateRef<any>;\r\n  \r\n  /** Set to true if selectionContext is defined */\r\n  @HostBinding('class.is-selectable')\r\n  @Input() selectable = false;\r\n\r\n  /** Hides/shows the selection toolbar template when 1 or more rows is selected */\r\n  hasSelection = false\r\n\r\n  /**\r\n   * Enables form table styling to allow form controls inside table cells.\r\n   */\r\n  @HostBinding('class.is-form-table')\r\n  @Input() isForm = false;\r\n  \r\n  /**\r\n   * The current sort field and direction ('desc' or 'asc')\r\n   */\r\n  @Input() sort?: TableSort | TableSort[];\r\n\r\n  /**\r\n   * Emits the new sort value whenever a sortable column is clicked\r\n   */\r\n  @Output() sortChange: EventEmitter<TableSort | TableSort[]> = new EventEmitter();\r\n\r\n  /**\r\n   * True if a TableMasterRowComponent exists in the table content\r\n   */\r\n  @HostBinding('class.is-master-detail') hasMasterDetailRows = false;\r\n\r\n  @ContentChildren(TableMasterRowComponent, {descendants: true}) masterRows!: QueryList<TableMasterRowComponent>;\r\n\r\n  public resizableColumns!: QueryList<ResizableColumnComponent>;\r\n  @ContentChildren(ResizableColumnComponent, {descendants: true}) _resizableColumns!: QueryList<ResizableColumnComponent>;\r\n\r\n  @Input('resizableColumns') searchableTableResizableColumns?: QueryList<ResizableColumnComponent>;\r\n\r\n  /**\r\n   * Unlisten function returned by `renderer.listen()` that are called\r\n   * OnDestroy to clean up all listeners\r\n   */\r\n  private unlisteners: (() => void)[] = [];\r\n  \r\n  /** The current sort value */\r\n  private currentSort?: TableSort | TableSort[];\r\n  \r\n  /** A map of sortable `<th>` elements */\r\n  private sortableCols: {[field: string]: HTMLTableHeaderCellElement} = {};\r\n\r\n  private destroyed: Subject<void> = new Subject();\r\n  \r\n  constructor(private renderer: Renderer2, public el: ElementRef) {}\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (this.sortable && changes.sort && changes.sort.currentValue && !changes.sort.firstChange) {\r\n      const newValue: TableSort | TableSort[] = changes.sort.currentValue;\r\n      if (!this.currentSort || !isEqual(newValue, this.currentSort)){\r\n        this.onSort(newValue);\r\n      }\r\n    }\r\n  }\r\n\r\n  ngAfterContentInit(): void {\r\n    this.hasMasterDetailRows = !!this.masterRows.length;\r\n    \r\n    this.masterRows.changes.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe(() => {\r\n      this.hasMasterDetailRows = !!this.masterRows.length;\r\n    });\r\n\r\n    this.resizableColumns = this.searchableTableResizableColumns || this._resizableColumns;\r\n  }\r\n  \r\n  ngAfterViewInit(): void {\r\n    if (this.sortable) {\r\n      this.initSortable();\r\n    }\r\n\r\n    if (this.selectable) {\r\n      this.initSelectable();\r\n    }\r\n  }\r\n  \r\n  ngOnDestroy(): void {\r\n    if (this.unlisteners.length) {\r\n      this.unlisteners.forEach(unlisten => unlisten());\r\n    }\r\n    \r\n    this.destroyed.next();\r\n    this.destroyed.unsubscribe();\r\n  }\r\n\r\n  /**\r\n   * Get all elements inside the host element that match a selector \r\n   * @param selector - Any valid selector for [querySelectorAll](https://developer.mozilla.org/en-US/docs/Web/API/Document/querySelectorAll)\r\n   */\r\n  private queryElements<T>(selector: string): T[] {\r\n    const table = this.el.nativeElement;\r\n    return Array.from(table.querySelectorAll(selector));\r\n  }\r\n\r\n  /**\r\n   * Store a `<th>` for each sortable column and attach a click\r\n   * event listener. \r\n   */\r\n  private initSortable(): void {\r\n    this.queryElements<HTMLTableHeaderCellElement>('th').forEach(th => {\r\n      if (th.hasAttribute('data-sortfield')) {\r\n        \r\n        const field = th.getAttribute('data-sortfield');\r\n        if (field) {\r\n          this.sortableCols[field] = th;\r\n          const unlisten = this.renderer.listen(th, 'click', (event: MouseEvent) => {\r\n\r\n            // Do not sort if the event target was the resizable handle\r\n            if (th.classList.contains('is-resizable') && event.target && (<HTMLElement>event.target).classList.contains('handle')) {\r\n              return;\r\n            }\r\n\r\n            let direction: TableSortDirection;\r\n\r\n            if (th.classList.contains('is-sorted-desc')) {\r\n              direction = 'asc';\r\n            } else {\r\n              direction = 'desc';\r\n            }\r\n\r\n            this.onSort({ field, direction });\r\n          });\r\n  \r\n          // Store unlisten function to remove listener when component is destroyed\r\n          this.unlisteners.push(unlisten);\r\n        } else {\r\n          console.error(`data-sortfield attribute was not assigned a value.`, th);\r\n        }\r\n      }\r\n    });\r\n\r\n    // Set the initial sort\r\n    if (this.sort) {\r\n      this.onSort(this.sort);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Find the sorted column and apply the desc or asc class in order\r\n   * to show the correct sort icon. If another field was sorted, reset it.\r\n   * Emit the current field and direction of the sorted column\r\n   * @param field \r\n   */\r\n  private onSort(sort: TableSort | TableSort[]): void {\r\n    this.clearSort();\r\n\r\n    if (Array.isArray(sort)) {\r\n      sort.forEach(s => this.applySort(s));\r\n    } else {\r\n      this.applySort(sort);\r\n    }\r\n\r\n    this.currentSort = cloneDeep(sort);\r\n    this.sortChange.emit(this.currentSort);\r\n  }\r\n\r\n  private clearSort() {\r\n    Object.values(this.sortableCols).forEach(th => {\r\n      this.renderer.removeClass(th, `is-sorted-asc`);\r\n      this.renderer.removeClass(th, `is-sorted-desc`);\r\n    });\r\n  }\r\n\r\n  private applySort(sort: TableSort) {\r\n    const th = this.sortableCols[sort.field];\r\n    if (!th) { return; }\r\n\r\n    this.renderer.addClass(th, `is-sorted-${sort.direction}`);\r\n  }\r\n\r\n  private initSelectable(): void {\r\n    this.selectionContext!.rowCheckboxes.valueChanges.pipe(\r\n      takeUntil(this.destroyed)\r\n    ).subscribe((value: boolean[]) => {\r\n      this.hasSelection = value.some(v => v);\r\n    });\r\n  }\r\n}\r\n","<div class=\"selectable-table-toolbar\"\r\n     *ngIf=\"hasSelection && selectionToolbarTemplate\">\r\n  <ng-container *ngTemplateOutlet=\"selectionToolbarTemplate\"></ng-container>\r\n</div>\r\n<div id=\"{{id}}-scroll-container\" \r\n     class=\"table-scroll-container\">\r\n  <table class=\"main-table\"\r\n         id=\"{{id}}\"\r\n         [ecResizableTable]=\"resizable\"\r\n         [containerEl]=\"el\"\r\n         [sortableTable]=\"sortable\"\r\n         [resizableColumns]=\"resizableColumns\">\r\n    <ng-content></ng-content>\r\n  </table>\r\n</div>"]}