@energycap/components 0.39.4-ECAP-23220-bc-file-upload-dialog.20231220-1530 → 0.39.4-ECAP-23219-bills-tree-collapse.20231221-1308
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/controls/file-upload/file-upload.component.mjs +32 -93
- package/esm2020/lib/display/hierarchy/hierarchy-base.mjs +2 -2
- package/esm2020/lib/display/hierarchy/hierarchy-tree/hierarchy-tree.component.mjs +3 -3
- package/fesm2015/energycap-components.mjs +33 -97
- package/fesm2015/energycap-components.mjs.map +1 -1
- package/fesm2020/energycap-components.mjs +30 -89
- package/fesm2020/energycap-components.mjs.map +1 -1
- package/lib/controls/file-upload/file-upload.component.d.ts +2 -19
- package/lib/display/hierarchy/hierarchy-base.d.ts +2 -2
- package/package.json +1 -1
- package/src/assets/locales/en_US.json +1 -3
@@ -5,12 +5,10 @@ import { FormControlBase } from '../form-control-base';
|
|
5
5
|
import * as i0 from "@angular/core";
|
6
6
|
import * as i1 from "../../core/validation-message.service";
|
7
7
|
import * as i2 from "../../shared/form-group.helper";
|
8
|
-
import * as i3 from "@angular/
|
9
|
-
import * as i4 from "
|
10
|
-
import * as i5 from "../
|
11
|
-
import * as i6 from "../form-
|
12
|
-
import * as i7 from "../form-group/form-group.component";
|
13
|
-
import * as i8 from "@ngx-translate/core";
|
8
|
+
import * as i3 from "@angular/forms";
|
9
|
+
import * as i4 from "../button/button.component";
|
10
|
+
import * as i5 from "../form-control/form-control.component";
|
11
|
+
import * as i6 from "../form-group/form-group.component";
|
14
12
|
export const FileTypeExtensions = {
|
15
13
|
zip: ['.zip'],
|
16
14
|
excel: ['.xls', '.xlsx']
|
@@ -19,8 +17,8 @@ export class FileUploadComponent extends FormControlBase {
|
|
19
17
|
// static class to create the form group from a parent component
|
20
18
|
static getFormModel(validators, disabled = false) {
|
21
19
|
let formGroup = new UntypedFormGroup({
|
22
|
-
|
23
|
-
|
20
|
+
file: new UntypedFormControl({ value: null, disabled: disabled }, validators),
|
21
|
+
name: new UntypedFormControl({ value: null, disabled: disabled }, validators),
|
24
22
|
base64FileString: new UntypedFormControl(null),
|
25
23
|
uploadResult: new UntypedFormControl(null)
|
26
24
|
});
|
@@ -41,16 +39,6 @@ export class FileUploadComponent extends FormControlBase {
|
|
41
39
|
* File output, determines which properties are supplied on the formModel
|
42
40
|
*/
|
43
41
|
this.fileOutput = 'base64';
|
44
|
-
/**
|
45
|
-
* Optional display type that controls whether the file input textbox is displayed or
|
46
|
-
* simply a button the user clicks to launch the OS file storage dialog.
|
47
|
-
* Default: file
|
48
|
-
*/
|
49
|
-
this.displayType = 'file';
|
50
|
-
/**
|
51
|
-
* Optional property to control whether the user can select multiple files
|
52
|
-
*/
|
53
|
-
this.multiSelect = false;
|
54
42
|
}
|
55
43
|
ngOnChanges(changes) {
|
56
44
|
super.ngOnChanges(changes);
|
@@ -61,11 +49,10 @@ export class FileUploadComponent extends FormControlBase {
|
|
61
49
|
// Watch for name to change, if the value is cleared we will clear the other
|
62
50
|
// supporting model properties. The name can be cleared by the user manually or via
|
63
51
|
// the clear button
|
64
|
-
this.formModel?.get('
|
52
|
+
this.formModel?.get('name')?.valueChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(value => {
|
65
53
|
if (!value) {
|
66
54
|
this.formModel.patchValue({
|
67
|
-
|
68
|
-
names: null,
|
55
|
+
file: null,
|
69
56
|
base64FileString: null,
|
70
57
|
uploadResult: null
|
71
58
|
});
|
@@ -73,21 +60,25 @@ export class FileUploadComponent extends FormControlBase {
|
|
73
60
|
});
|
74
61
|
}
|
75
62
|
async fileChange(files) {
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
63
|
+
let file = files.item(0);
|
64
|
+
// If there is a file selected and then opened again and click cancel you get null so don't try and set anything
|
65
|
+
if (file) {
|
66
|
+
let reader = new FileReader();
|
67
|
+
reader.onloadend = async (e) => {
|
68
|
+
let base64FileString = reader?.result?.toString().split(",")[1];
|
69
|
+
this.processFile(file, base64FileString);
|
70
|
+
};
|
71
|
+
if (this.isBase64FileOutput()) {
|
72
|
+
reader.readAsDataURL(file);
|
73
|
+
}
|
74
|
+
else {
|
75
|
+
await this.processFile(file);
|
76
|
+
}
|
77
|
+
// Clear the file inputs value, this will allow the user to pick the same filename and cause
|
78
|
+
// the fileChange to trigger.
|
79
|
+
if (this.fileInput) {
|
80
|
+
this.fileInput.nativeElement.value = '';
|
85
81
|
}
|
86
|
-
}
|
87
|
-
// Clear the file inputs value, this will allow the user to pick the same filenames again
|
88
|
-
// and cause fileChange to re-trigger.
|
89
|
-
if (this.fileInput) {
|
90
|
-
this.fileInput.nativeElement.value = '';
|
91
82
|
}
|
92
83
|
}
|
93
84
|
/**
|
@@ -120,7 +111,7 @@ export class FileUploadComponent extends FormControlBase {
|
|
120
111
|
this.patchFileResult(file, base64FileString, result);
|
121
112
|
}
|
122
113
|
catch (e) {
|
123
|
-
// Bummer, we're not going to do anything about it though.
|
114
|
+
// Bummer, we're not going to do anything about it though.
|
124
115
|
// We are not patching any of the result so any existing information remains
|
125
116
|
}
|
126
117
|
}
|
@@ -143,27 +134,9 @@ export class FileUploadComponent extends FormControlBase {
|
|
143
134
|
* @param onFileSelectedResult
|
144
135
|
*/
|
145
136
|
patchFileResult(file, base64FileString, onFileSelectedResult) {
|
146
|
-
console.log('inside patchFileResult');
|
147
|
-
// Get the current value of files from formModel
|
148
|
-
let files = this.formModel?.get('files')?.value || [];
|
149
|
-
let names = this.formModel?.get('names')?.value || [];
|
150
|
-
// If multiSelect is true we need to append the file to the existing array
|
151
|
-
// otherwise we'll just set the file to the form model
|
152
|
-
if (this.multiSelect) {
|
153
|
-
if (file) {
|
154
|
-
files = [...files, file];
|
155
|
-
}
|
156
|
-
if (file?.name) {
|
157
|
-
names = [...names, file.name];
|
158
|
-
}
|
159
|
-
}
|
160
|
-
else {
|
161
|
-
files = [file];
|
162
|
-
names = [file?.name];
|
163
|
-
}
|
164
137
|
this.formModel?.patchValue({
|
165
|
-
|
166
|
-
|
138
|
+
file: file,
|
139
|
+
name: file?.name,
|
167
140
|
base64FileString: base64FileString ?? null
|
168
141
|
});
|
169
142
|
if (onFileSelectedResult) {
|
@@ -172,41 +145,13 @@ export class FileUploadComponent extends FormControlBase {
|
|
172
145
|
else {
|
173
146
|
this.formModel.patchValue({ uploadResult: null });
|
174
147
|
}
|
175
|
-
console.log('formModel.value after fileUploadComponent patch: ', this.formModel?.value);
|
176
|
-
}
|
177
|
-
/* ---------------------------- Helper Functions ---------------------------- */
|
178
|
-
clearFormModel() {
|
179
|
-
this.formModel?.patchValue({
|
180
|
-
files: null,
|
181
|
-
names: null,
|
182
|
-
base64FileString: null,
|
183
|
-
uploadResult: null
|
184
|
-
});
|
185
|
-
}
|
186
|
-
async handleMultipleFiles(files) {
|
187
|
-
const promises = Array.from(files).map((file) => this.readFile(file));
|
188
|
-
await Promise.all(promises);
|
189
|
-
}
|
190
|
-
;
|
191
|
-
async readFile(file) {
|
192
|
-
const reader = new FileReader();
|
193
|
-
reader.onloadend = async (e) => {
|
194
|
-
const base64FileString = reader?.result?.toString().split(",")[1];
|
195
|
-
await this.processFile(file, base64FileString);
|
196
|
-
};
|
197
|
-
if (this.isBase64FileOutput()) {
|
198
|
-
reader.readAsDataURL(file);
|
199
|
-
}
|
200
|
-
else {
|
201
|
-
await this.processFile(file);
|
202
|
-
}
|
203
148
|
}
|
204
149
|
}
|
205
150
|
FileUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, deps: [{ token: i1.ValidationMessageService }, { token: i2.FormGroupHelper }], target: i0.ɵɵFactoryTarget.Component });
|
206
|
-
FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected"
|
151
|
+
FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (click)=\"browseBtn.focus()\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\">\r\n <ec-form-control id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: i4.ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: i5.FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: i6.FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }] });
|
207
152
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, decorators: [{
|
208
153
|
type: Component,
|
209
|
-
args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"
|
154
|
+
args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (click)=\"browseBtn.focus()\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\">\r\n <ec-form-control id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
|
210
155
|
}], ctorParameters: function () { return [{ type: i1.ValidationMessageService }, { type: i2.FormGroupHelper }]; }, propDecorators: { placeholder: [{
|
211
156
|
type: Input
|
212
157
|
}], fileType: [{
|
@@ -217,14 +162,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
|
|
217
162
|
type: Input
|
218
163
|
}], onFileSelected: [{
|
219
164
|
type: Input
|
220
|
-
}], displayType: [{
|
221
|
-
type: Input
|
222
|
-
}], buttonLabel: [{
|
223
|
-
type: Input
|
224
|
-
}], multiSelect: [{
|
225
|
-
type: Input
|
226
165
|
}], fileInput: [{
|
227
166
|
type: ViewChild,
|
228
167
|
args: ["fileInput", { read: ElementRef, static: true }]
|
229
168
|
}] } });
|
230
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.ts","../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn } from '@angular/forms';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ValidationMessageService } from '../../core/validation-message.service';\r\nimport { FormGroupHelper } from '../../shared/form-group.helper';\r\nimport { FormControlBase } from '../form-control-base';\r\n\r\nexport type FileType = 'zip' | 'excel' | 'custom';\r\nexport type FileOutput = 'raw' | 'base64';\r\n\r\nexport const FileTypeExtensions = {\r\n  zip: ['.zip'],\r\n  excel: ['.xls', '.xlsx']\r\n};\r\n\r\n@Component({\r\n  selector: \"ec-file-upload\",\r\n  templateUrl: \"./file-upload.component.html\",\r\n  styleUrls: [\"./file-upload.component.scss\"]\r\n})\r\nexport class FileUploadComponent extends FormControlBase implements OnInit, OnChanges {\r\n\r\n  // static class to create the form group from a parent component\r\n  public static getFormModel(\r\n    validators: ValidatorFn[],\r\n    disabled: boolean = false\r\n  ): UntypedFormGroup {\r\n    let formGroup = new UntypedFormGroup({\r\n      files: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      names: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new UntypedFormControl(null),\r\n      uploadResult: new UntypedFormControl(null)\r\n    });\r\n    if (disabled) {\r\n      formGroup.disable();\r\n    }\r\n    return formGroup;\r\n  }\r\n\r\n  /**\r\n   * The value of the textbox input's placeholder\r\n   */\r\n  @Input() public placeholder?: string = \"Choose file...\";\r\n\r\n  /** Common extensions for a file browsing dialog\r\n   *  Note: Edge does not support the accept attribute on file inputs, therefor all file types\r\n   *        will be shown.  Firefox and Chrome both support this feature.\r\n   */\r\n  @Input() public fileType?: FileType | undefined;\r\n\r\n  /**\r\n   * File output, determines which properties are supplied on the formModel\r\n   */\r\n  @Input() public fileOutput?: FileOutput = 'base64';\r\n\r\n  /** If fileType is set to custom set the acceptable file types based on the custom array */\r\n  @Input() public customExtensions?: Array<string> | undefined;\r\n\r\n  /**\r\n   * Optional callback supported if the hosting page needs to process the file before\r\n   * setting the formModel with the file information. If the promise resolves it will continue\r\n   * and set the file name and contents to the formModel, otherwise on failure it'll do nothing.\r\n   */\r\n  @Input() public onFileSelected?: (files: File | File[]) => Promise<any>;\r\n\r\n  /** \r\n   * Optional display type that controls whether the file input textbox is displayed or\r\n   * simply a button the user clicks to launch the OS file storage dialog.\r\n   * Default: file\r\n   */\r\n  @Input() public displayType?: 'file' | 'button' = 'file';\r\n\r\n  /** \r\n   * When display type is set to button this property will control the button label\r\n   */\r\n  @Input() public buttonLabel?: string;\r\n\r\n  /**\r\n   * Optional property to control whether the user can select multiple files\r\n   */\r\n  @Input() public multiSelect?: boolean = false;\r\n\r\n  @ViewChild(\"fileInput\", { read: ElementRef, static: true }) public fileInput?: ElementRef;\r\n\r\n  /** Property bound to the file input to filter what file types are shown in the dialog */\r\n  public fileTypeAccept: string | undefined;\r\n\r\n  constructor(\r\n    protected validationMessageService: ValidationMessageService,\r\n    protected formGroupHelper: FormGroupHelper,\r\n  ) {\r\n    super(validationMessageService, formGroupHelper);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges) {\r\n    super.ngOnChanges(changes);\r\n\r\n    this.updateFileTypeAccept();\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    super.ngOnInit();\r\n\r\n    // Watch for name to change, if the value is cleared we will clear the other\r\n    // supporting model properties. The name can be cleared by the user manually or via\r\n    // the clear button\r\n    this.formModel?.get('names')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          files: null,\r\n          names: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n  }\r\n\r\n  public async fileChange(files: FileList): Promise<void> {\r\n\r\n    if (this.multiSelect) {\r\n      // Since we append the files to the form model when multiSelect=true, we need to clear the form model at the start\r\n      this.clearFormModel();\r\n      await this.handleMultipleFiles(files);\r\n    } else {\r\n      const file = files.item(0);\r\n      if (file) {\r\n        await this.readFile(file);\r\n      }\r\n    }\r\n\r\n    // Clear the file inputs value, this will allow the user to pick the same filenames again\r\n    // and cause fileChange to re-trigger.\r\n    if (this.fileInput) {\r\n      this.fileInput.nativeElement.value = '';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Checks the file type and updates the file type accept property. This is what determines the file\r\n   * type choices that the user will be limited to in the file browse dialog\r\n   */\r\n  private updateFileTypeAccept(): void {\r\n    if (this.fileType) {\r\n      if (this.fileType !== \"custom\") {\r\n        this.fileTypeAccept = FileTypeExtensions[this.fileType].join(\",\");\r\n      } else {\r\n        if (this.customExtensions) {\r\n          this.fileTypeAccept = this.customExtensions.join(\",\");\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Take a file that was selected by the user and process/patch our form model\r\n   * If the host component requires an action to occur with the file prior to the patch it will call\r\n   * and wait for it to return.\r\n   * @param file \r\n   * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64\r\n   */\r\n  private async processFile(file: File, base64FileString?: string | undefined): Promise<void> {\r\n    if (this.onFileSelected) {\r\n      try {\r\n        let result = await this.onFileSelected(file);\r\n        this.patchFileResult(file, base64FileString, result);\r\n      } catch (e) {\r\n        // Bummer, we're not going to do anything about it though.\r\n        // We are not patching any of the result so any existing information remains\r\n      }\r\n    } else {\r\n      this.patchFileResult(file, base64FileString);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Based on the fileOutput return whether this component is expected to deliver a base64 output\r\n   * @returns \r\n   */\r\n  private isBase64FileOutput(): boolean {\r\n    return this.fileOutput === 'base64';\r\n  }\r\n\r\n  /**\r\n   * When the file was selected and processed patch the file information that the hosting form will\r\n   * be looking for. \r\n   * @param file \r\n   * @param base64FileString \r\n   * @param onFileSelectedResult \r\n   */\r\n  private patchFileResult(file: File | null, base64FileString?: string, onFileSelectedResult?: any): void {    \r\n    console.log('inside patchFileResult');\r\n\r\n    // Get the current value of files from formModel\r\n    let files = this.formModel?.get('files')?.value || [];\r\n    let names = this.formModel?.get('names')?.value || [];\r\n\r\n    // If multiSelect is true we need to append the file to the existing array\r\n    // otherwise we'll just set the file to the form model\r\n    if (this.multiSelect) {\r\n      if (file) { files = [ ...files, file ]; }\r\n      if (file?.name) { names = [ ...names, file.name ]; }\r\n\r\n    } else {\r\n      files = [ file ];\r\n      names = [ file?.name ];\r\n    }\r\n\r\n    this.formModel?.patchValue({\r\n      files: files,\r\n      names: names,\r\n      base64FileString: base64FileString ?? null\r\n    });\r\n\r\n    if (onFileSelectedResult) {\r\n      this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n    } else {\r\n      this.formModel.patchValue({ uploadResult: null });\r\n    }\r\n\r\n    console.log('formModel.value after fileUploadComponent patch: ', this.formModel?.value);\r\n  }\r\n\r\n  /* ---------------------------- Helper Functions ---------------------------- */\r\n  private clearFormModel(): void {\r\n    this.formModel?.patchValue({\r\n      files: null,\r\n      names: null,\r\n      base64FileString: null,\r\n      uploadResult: null\r\n    });\r\n  }\r\n\r\n  private async handleMultipleFiles(files: FileList): Promise<void> {\r\n    const promises = Array.from(files).map((file) => this.readFile(file));\r\n    await Promise.all(promises);\r\n  };\r\n\r\n  private async readFile(file: File): Promise<void> {\r\n    const reader: FileReader = new FileReader();\r\n\r\n    reader.onloadend = async e => {\r\n      const base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n      await this.processFile(file, base64FileString);\r\n    };\r\n\r\n    if (this.isBase64FileOutput()) {\r\n      reader.readAsDataURL(file);\r\n    } else {\r\n      await this.processFile(file);\r\n    }\r\n  }\r\n}\r\n","<ec-form-group [label]=\"label\"\r\n               [formGroup]=\"formModel\"\r\n               class=\"mb-0\">\r\n  <div class=\"d-flex control-group\">\r\n    <div class=\"d-flex flex-grow position-relative\">\r\n      <input #fileInput\r\n             id=\"{{inputId}}_input\"\r\n             type=\"file\"\r\n             tabindex=\"-1\"\r\n             [attr.accept]=\"fileTypeAccept\"\r\n             (change)=\"fileChange($event.target.files)\"\r\n             [class.has-value]=\"displayType === 'file' ? formModel?.get('names').value : undefined\"\r\n             [attr.multiple]=\"multiSelect ? 'multiple' : undefined\">\r\n            <ec-form-control *ngIf=\"displayType === 'file'\"\r\n                       id=\"{{inputId}}_formControl\"\r\n                       class=\"text-truncate\"\r\n                       [required]=\"required\"\r\n                       [pending]=\"pending\">\r\n        <input id=\"{{inputId}}_name\"\r\n               [formControl]=\"formModel?.get('names')\"\r\n               type=\"text\"\r\n               [placeholder]=\"placeholder\"\r\n               [tabindex]=\"-1\">\r\n      </ec-form-control>\r\n    </div>\r\n    <ec-button *ngIf=\"displayType === 'file'\"\r\n               #browseBtn\r\n               id=\"{{inputId}}_browseBtn\"\r\n               (clicked)=\"fileInput.click()\"\r\n               type=\"secondary\"\r\n               [tabindex]=\"tabindex\"\r\n               [disabled]=\"formModel?.get('names').disabled\"\r\n               label=\"Browse\"\r\n               [autofocus]=\"autofocus\">\r\n    </ec-button>\r\n  </div>\r\n  <ec-button *ngIf=\"displayType === 'button'\"\r\n             #selectZipFilesBtn\r\n             id=\"{{inputId}}_btn\"\r\n             [pending]=\"pending\"\r\n             type=\"primary\"\r\n             [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n             (clicked)=\"fileInput.click()\"\r\n             style=\"width: 100%;\">\r\n  </ec-button>\r\n</ec-form-group>"]}
|
169
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.ts","../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn } from '@angular/forms';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ValidationMessageService } from '../../core/validation-message.service';\r\nimport { FormGroupHelper } from '../../shared/form-group.helper';\r\nimport { FormControlBase } from '../form-control-base';\r\n\r\nexport type FileType = 'zip' | 'excel' | 'custom';\r\nexport type FileOutput = 'raw' | 'base64';\r\n\r\nexport const FileTypeExtensions = {\r\n  zip: ['.zip'],\r\n  excel: ['.xls', '.xlsx']\r\n};\r\n\r\n@Component({\r\n  selector: \"ec-file-upload\",\r\n  templateUrl: \"./file-upload.component.html\",\r\n  styleUrls: [\"./file-upload.component.scss\"]\r\n})\r\nexport class FileUploadComponent extends FormControlBase implements OnInit, OnChanges {\r\n\r\n  // static class to create the form group from a parent component\r\n  public static getFormModel(\r\n    validators: ValidatorFn[],\r\n    disabled: boolean = false\r\n  ): UntypedFormGroup {\r\n    let formGroup = new UntypedFormGroup({\r\n      file: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      name: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new UntypedFormControl(null),\r\n      uploadResult: new UntypedFormControl(null)\r\n    });\r\n    if (disabled) {\r\n      formGroup.disable();\r\n    }\r\n    return formGroup;\r\n  }\r\n\r\n  /**\r\n   * The value of the textbox input's placeholder\r\n   */\r\n  @Input() public placeholder?: string = \"Choose file...\";\r\n\r\n  /** Common extensions for a file browsing dialog\r\n   *  Note: Edge does not support the accept attribute on file inputs, therefor all file types\r\n   *        will be shown.  Firefox and Chrome both support this feature.\r\n   */\r\n  @Input() public fileType?: FileType | undefined;\r\n\r\n  /**\r\n   * File output, determines which properties are supplied on the formModel\r\n   */\r\n  @Input() public fileOutput?: FileOutput = 'base64';\r\n\r\n  /** If fileType is set to custom set the acceptable file types based on the custom array */\r\n  @Input() public customExtensions?: Array<string> | undefined;\r\n\r\n  /**\r\n   * Optional callback supported if the hosting page needs to process the file before\r\n   * setting the formModel with the file information. If the promise resolves it will continue\r\n   * and set the file name and contents to the formModel, otherwise on failure it'll do nothing.\r\n   */\r\n  @Input() public onFileSelected?: (file: File) => Promise<any>;\r\n\r\n  @ViewChild(\"fileInput\", { read: ElementRef, static: true }) public fileInput?: ElementRef;\r\n\r\n  /** Property bound to the file input to filter what file types are shown in the dialog */\r\n  public fileTypeAccept: string | undefined;\r\n\r\n  constructor(\r\n    protected validationMessageService: ValidationMessageService,\r\n    protected formGroupHelper: FormGroupHelper\r\n  ) {\r\n    super(validationMessageService, formGroupHelper);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges) {\r\n    super.ngOnChanges(changes);\r\n\r\n    this.updateFileTypeAccept();\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    super.ngOnInit();\r\n\r\n    // Watch for name to change, if the value is cleared we will clear the other\r\n    // supporting model properties. The name can be cleared by the user manually or via\r\n    // the clear button\r\n    this.formModel?.get('name')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          file: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n  }\r\n\r\n  public async fileChange(files: FileList): Promise<void> {\r\n    let file: File | null = files.item(0);\r\n\r\n    // If there is a file selected and then opened again and click cancel you get null so don't try and set anything\r\n    if (file) {\r\n      let reader: FileReader = new FileReader();\r\n\r\n      reader.onloadend = async e => {\r\n        let base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n        this.processFile(file!, base64FileString);\r\n      };\r\n\r\n      if (this.isBase64FileOutput()) {\r\n        reader.readAsDataURL(file);\r\n      } else {\r\n        await this.processFile(file!);\r\n      }\r\n\r\n      // Clear the file inputs value, this will allow the user to pick the same filename and cause\r\n      // the fileChange to trigger.\r\n      if (this.fileInput) {\r\n        this.fileInput.nativeElement.value = '';\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Checks the file type and updates the file type accept property. This is what determines the file\r\n   * type choices that the user will be limited to in the file browse dialog\r\n   */\r\n  private updateFileTypeAccept(): void {\r\n    if (this.fileType) {\r\n      if (this.fileType !== \"custom\") {\r\n        this.fileTypeAccept = FileTypeExtensions[this.fileType].join(\",\");\r\n      } else {\r\n        if (this.customExtensions) {\r\n          this.fileTypeAccept = this.customExtensions.join(\",\");\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Take a file that was selected by the user and process/patch our form model\r\n   * If the host component requires an action to occur with the file prior to the patch it will call\r\n   * and wait for it to return.\r\n   * @param file \r\n   * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64\r\n   */\r\n  private async processFile(file: File, base64FileString?: string | undefined): Promise<void> {\r\n    if (this.onFileSelected) {\r\n      try {\r\n        let result = await this.onFileSelected(file);\r\n        this.patchFileResult(file, base64FileString, result);\r\n      } catch (e) {\r\n        // Bummer, we're not going to do anything about it though. \r\n        // We are not patching any of the result so any existing information remains\r\n      }\r\n    } else {\r\n      this.patchFileResult(file, base64FileString);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Based on the fileOutput return whether this component is expected to deliver a base64 output\r\n   * @returns \r\n   */\r\n  private isBase64FileOutput(): boolean {\r\n    return this.fileOutput === 'base64';\r\n  }\r\n\r\n  /**\r\n   * When the file was selected and processed patch the file information that the hosting form will\r\n   * be looking for. \r\n   * @param file \r\n   * @param base64FileString \r\n   * @param onFileSelectedResult \r\n   */\r\n  private patchFileResult(file: File | null, base64FileString?: string, onFileSelectedResult?: any): void {\r\n    this.formModel?.patchValue({ \r\n      file: file, \r\n      name: file?.name, \r\n      base64FileString: base64FileString ?? null\r\n    });\r\n\r\n    if (onFileSelectedResult) {\r\n      this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n    } else {\r\n      this.formModel.patchValue({ uploadResult: null });\r\n    }\r\n  }\r\n}\r\n","<ec-form-group [label]=\"label\"\r\n               [formGroup]=\"formModel\"\r\n               class=\"mb-0\">\r\n  <div class=\"d-flex control-group\">\r\n    <div class=\"d-flex flex-grow position-relative\">\r\n      <input #fileInput\r\n             id=\"{{inputId}}_input\"\r\n             type=\"file\"\r\n             tabindex=\"-1\"\r\n             [attr.accept]=\"fileTypeAccept\"\r\n             (click)=\"browseBtn.focus()\"\r\n             (change)=\"fileChange($event.target.files)\"\r\n             [class.has-value]=\"formModel?.get('name')?.value\">\r\n      <ec-form-control id=\"{{inputId}}_formControl\"\r\n                       class=\"text-truncate\"\r\n                       [required]=\"required\"\r\n                       [pending]=\"pending\">\r\n        <input id=\"{{inputId}}_name\"\r\n               [formControl]=\"formModel?.get('name')\"\r\n               type=\"text\"\r\n               [placeholder]=\"placeholder\"\r\n               [tabindex]=\"-1\">\r\n      </ec-form-control>\r\n    </div>\r\n    <ec-button #browseBtn\r\n               id=\"{{inputId}}_browseBtn\"\r\n               (clicked)=\"fileInput.click()\"\r\n               type=\"secondary\"\r\n               [tabindex]=\"tabindex\"\r\n               [disabled]=\"formModel?.get('name')?.disabled\"\r\n               label=\"Browse\"\r\n               [autofocus]=\"autofocus\">\r\n    </ec-button>\r\n  </div>\r\n</ec-form-group>"]}
|
@@ -9,7 +9,7 @@ export class HierarchyItem {
|
|
9
9
|
this.label = '';
|
10
10
|
/** Indicates how many levels deep the item is in the parent structure */
|
11
11
|
this.level = 0;
|
12
|
-
/** Display item as a link or a
|
12
|
+
/** Display item as a link, a heading, or with a divider line below it */
|
13
13
|
this.display = 'default';
|
14
14
|
/** Set to hide toggle, even if item has children */
|
15
15
|
this.hideToggle = false;
|
@@ -103,4 +103,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
|
|
103
103
|
}], getItemChildren: [{
|
104
104
|
type: Output
|
105
105
|
}] } });
|
106
|
-
//# sourceMappingURL=data:application/json;base64,
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchy-base.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/display/hierarchy/hierarchy-base.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAY,KAAK,EAAwB,MAAM,EAAiB,MAAM,eAAe,CAAC;AAEtH,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAE/B,OAAO,EAAE,gBAAgB,EAAE,MAAM,8BAA8B,CAAC;;AAGhE,MAAM,OAAO,aAAa;IAA1B;QACE,OAAE,GAAW,EAAE,CAAC;QAEhB,oCAAoC;QACpC,UAAK,GAAW,EAAE,CAAC;QAKnB,yEAAyE;QACzE,UAAK,GAAW,CAAC,CAAC;QAUlB,yEAAyE;QACzE,YAAO,GAAuC,SAAS,CAAC;QAExD,oDAAoD;QACpD,eAAU,GAAa,KAAK,CAAC;QAE7B;;WAEG;QACH,gBAAW,GAAY,KAAK,CAAC;QAE7B,4CAA4C;QAC5C,aAAQ,GAAyB,EAAE,CAAC;QAEpC,qDAAqD;QACrD,aAAQ,GAAa,KAAK,CAAC;QAE3B,+CAA+C;QAC/C,aAAQ,GAAa,KAAK,CAAC;QAE3B,mDAAmD;QACnD,eAAU,GAAa,KAAK,CAAC;QAE7B,aAAQ,GAAa,KAAK,CAAC;QAE3B;;WAEG;QACH,WAAM,GAAgB,SAAS,CAAC;QAEhC,oFAAoF;QACpF,aAAQ,GAAa,KAAK,CAAC;QAE3B;;;WAGG;QACH,uBAAkB,GAAa,IAAI,CAAC;IAOtC,CAAC;CAAA;AAGD,MAAM,OAAgB,aAAa;IAYjC,IAAW,SAAS;QAClB,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAID,YAAY,QAAkB;QAb9B,yDAAyD;QACxC,oBAAe,GAAgC,IAAI,YAAY,EAAiB,CAAC;QAElG;;UAEE;QACM,eAAU,GAAiB,IAAI,OAAO,EAAO,CAAC;QAQpD,IAAI,CAAC,gBAAgB,GAAG,QAAQ,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;IACzD,CAAC;IAEM,WAAW,CAAC,OAAsB;QACvC,IAAI,OAAO,IAAI,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,QAAQ,CAAC,YAAY,EAAE;YAChE;;cAEE;YACF,IAAI,IAAI,CAAC,QAAQ,CAAC,WAAW,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE;gBACpE,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC1C;SACF;IACH,CAAC;IAED;;MAEE;IACK,WAAW;QAChB,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;QACtB,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;IAC/B,CAAC;IAED,oCAAoC;IAC7B,iBAAiB,CAAC,IAAmB,EAAE,QAAiB;QAC7D,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,kBAAkB,EAAE;YACnD,cAAc,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,WAAW;YACxD,SAAS,EAAE,GAAG,IAAI,CAAC,KAAK,EAAE;SAC3B,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE;YACnE,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;YACxB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;SACjC;IACH,CAAC;IAED;OACG;IACI,eAAe,CAAC,UAAyB,EAAE,KAA2B;QAC3E,UAAU,CAAC,QAAQ,GAAG,KAAK,CAAC;QAC5B,UAAU,CAAC,MAAM,GAAG,SAAS,CAAC;IAChC,CAAC;IAED;;MAEE;IACK,YAAY,CAAC,UAAyB;QAC3C,UAAU,CAAC,MAAM,GAAG,OAAO,CAAC;IAC9B,CAAC;;0GAnEmB,aAAa;8FAAb,aAAa;2FAAb,aAAa;kBADlC,SAAS;+FAIQ,QAAQ;sBAAvB,KAAK;gBAGW,eAAe;sBAA/B,MAAM","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@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  /**\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    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 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 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 .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 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"] }] });
|
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 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 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 .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 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"] }]
|
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,mxHAkFc;2FDxED,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        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              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,o1HAmFc;2FDzED,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              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>"]}
|