turbogui-angular 15.1.2 → 15.2.0

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.
@@ -20,10 +20,10 @@ export class FadeAnimationClass {
20
20
  * @param enter The time and easing that we want to use for the enter state
21
21
  * @param leave The time and easing that we want to use for the leave state
22
22
  */
23
- static getTrigger(triggerName, enter = '1s ease', leave = '0s ease') {
23
+ static getTrigger(triggerName, enter = '1s ease', leave = '300ms ease') {
24
24
  return trigger(triggerName, [
25
- transition(':enter', [style({ opacity: 0 }), animate(enter)]),
26
- transition(':leave', [style({ opacity: 1 }), animate(leave, style({ opacity: 0 }))])
25
+ transition('void => *', [style({ opacity: 0 }), animate(enter, style({ opacity: 1 }))]),
26
+ transition('* => void', [animate(leave, style({ opacity: 0 }))])
27
27
  ]);
28
28
  }
29
29
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: FadeAnimationClass, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
@@ -32,4 +32,4 @@ export class FadeAnimationClass {
32
32
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: FadeAnimationClass, decorators: [{
33
33
  type: Injectable
34
34
  }] });
35
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFkZS5hbmltYXRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90dXJib2d1aS1hbmd1bGFyL3NyYy9tYWluL3ZpZXcvYW5pbWF0aW9ucy9mYWRlLmFuaW1hdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7OztHQU9HO0FBRUgsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzQyxPQUFPLEVBQUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsS0FBSyxFQUFFLE1BQU0scUJBQXFCLENBQUM7O0FBRzFFOztHQUVHO0FBRUgsTUFBTSxPQUFPLGtCQUFrQjtJQUUzQjs7Ozs7O09BTUc7SUFDSCxNQUFNLENBQUMsVUFBVSxDQUFDLFdBQW1CLEVBQUUsS0FBSyxHQUFHLFNBQVMsRUFBRSxLQUFLLEdBQUcsU0FBUztRQUV2RSxPQUFPLE9BQU8sQ0FBQyxXQUFXLEVBQUU7WUFDQyxVQUFVLENBQUMsUUFBUSxFQUFFLENBQUMsS0FBSyxDQUFDLEVBQUMsT0FBTyxFQUFFLENBQUMsRUFBQyxDQUFDLEVBQUUsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7WUFDM0QsVUFBVSxDQUFDLFFBQVEsRUFBRSxDQUFDLEtBQUssQ0FBQyxFQUFDLE9BQU8sRUFBRSxDQUFDLEVBQUMsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxLQUFLLEVBQUUsS0FBSyxDQUFDLEVBQUMsT0FBTyxFQUFFLENBQUMsRUFBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1NBQ2hGLENBQUMsQ0FBQztJQUNuQyxDQUFDOzhHQWZRLGtCQUFrQjtrSEFBbEIsa0JBQWtCOzsyRkFBbEIsa0JBQWtCO2tCQUQ5QixVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXHJcbiAqIFR1cmJvR1VJIGlzIEEgbGlicmFyeSB0aGF0IGhlbHBzIHdpdGggdGhlIG1vc3QgY29tbW9uIGFuZCBnZW5lcmljIFVJIGVsZW1lbnRzIGFuZCBmdW5jdGlvbmFsaXRpZXNcclxuICpcclxuICogV2Vic2l0ZSA6IC0+IGh0dHA6Ly93d3cudHVyYm9ndWkub3JnXHJcbiAqIExpY2Vuc2UgOiAtPiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wLiBZb3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuXHJcbiAqIExpY2Vuc2UgVXJsIDogLT4gaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wXHJcbiAqIENvcHlSaWdodCA6IC0+IENvcHlyaWdodCAyMDE4IEVkZXJ0b25lIEFkdmFuZGVkIFNvbHV0aW9ucy4gaHR0cHM6Ly93d3cuZWRlcnRvbmUuY29tXHJcbiAqL1xyXG5cclxuaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyB0cmlnZ2VyLCBhbmltYXRlLCB0cmFuc2l0aW9uLCBzdHlsZSB9IGZyb20gJ0Bhbmd1bGFyL2FuaW1hdGlvbnMnO1xyXG5cclxuXHJcbi8qKlxyXG4gKiBGYWRlIGFuaW1hdGlvbnNcclxuICovXHJcbkBJbmplY3RhYmxlKClcclxuZXhwb3J0IGNsYXNzIEZhZGVBbmltYXRpb25DbGFzcyB7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBHZXQgYSBjdXN0b20gdHJpZ2dlciB0byBjcmVhdGUgZmFkZSBhbmltYXRpb25zIHdoZW4gY29tcG9uZW50cyBhcmUgYWRkZWQgb3IgcmVtb3ZlZCBmcm9tIHRoZSBhcHBsaWNhdGlvblxyXG4gICAgICpcclxuICAgICAqIEBwYXJhbSB0cmlnZ2VyTmFtZSBUaGUgbmFtZSBmb3IgdGhlIHRyaWdnZXIgd2Ugd2FudCB0byBjcmVhdGVcclxuICAgICAqIEBwYXJhbSBlbnRlciBUaGUgdGltZSBhbmQgZWFzaW5nIHRoYXQgd2Ugd2FudCB0byB1c2UgZm9yIHRoZSBlbnRlciBzdGF0ZVxyXG4gICAgICogQHBhcmFtIGxlYXZlIFRoZSB0aW1lIGFuZCBlYXNpbmcgdGhhdCB3ZSB3YW50IHRvIHVzZSBmb3IgdGhlIGxlYXZlIHN0YXRlXHJcbiAgICAgKi9cclxuICAgIHN0YXRpYyBnZXRUcmlnZ2VyKHRyaWdnZXJOYW1lOiBzdHJpbmcsIGVudGVyID0gJzFzIGVhc2UnLCBsZWF2ZSA9ICcwcyBlYXNlJykge1xyXG5cclxuICAgICAgICByZXR1cm4gdHJpZ2dlcih0cmlnZ2VyTmFtZSwgW1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdHJhbnNpdGlvbignOmVudGVyJywgW3N0eWxlKHtvcGFjaXR5OiAwfSksIGFuaW1hdGUoZW50ZXIpXSksXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0cmFuc2l0aW9uKCc6bGVhdmUnLCBbc3R5bGUoe29wYWNpdHk6IDF9KSwgYW5pbWF0ZShsZWF2ZSwgc3R5bGUoe29wYWNpdHk6IDB9KSldKVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBdKTtcclxuICAgIH1cclxufVxyXG4iXX0=
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFkZS5hbmltYXRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90dXJib2d1aS1hbmd1bGFyL3NyYy9tYWluL3ZpZXcvYW5pbWF0aW9ucy9mYWRlLmFuaW1hdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7OztHQU9HO0FBRUgsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzQyxPQUFPLEVBQUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsS0FBSyxFQUFFLE1BQU0scUJBQXFCLENBQUM7O0FBRzFFOztHQUVHO0FBRUgsTUFBTSxPQUFPLGtCQUFrQjtJQUUzQjs7Ozs7O09BTUc7SUFDSCxNQUFNLENBQUMsVUFBVSxDQUFDLFdBQW1CLEVBQUUsS0FBSyxHQUFHLFNBQVMsRUFBRSxLQUFLLEdBQUcsWUFBWTtRQUUxRSxPQUFPLE9BQU8sQ0FBQyxXQUFXLEVBQUU7WUFDQyxVQUFVLENBQUMsV0FBVyxFQUFFLENBQUMsS0FBSyxDQUFDLEVBQUMsT0FBTyxFQUFFLENBQUMsRUFBQyxDQUFDLEVBQUUsT0FBTyxDQUFDLEtBQUssRUFBRSxLQUFLLENBQUMsRUFBRSxPQUFPLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDckYsVUFBVSxDQUFDLFdBQVcsRUFBRSxDQUFDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsS0FBSyxDQUFDLEVBQUMsT0FBTyxFQUFFLENBQUMsRUFBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1NBQzlELENBQUMsQ0FBQztJQUNuQyxDQUFDOzhHQWZRLGtCQUFrQjtrSEFBbEIsa0JBQWtCOzsyRkFBbEIsa0JBQWtCO2tCQUQ5QixVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXHJcbiAqIFR1cmJvR1VJIGlzIEEgbGlicmFyeSB0aGF0IGhlbHBzIHdpdGggdGhlIG1vc3QgY29tbW9uIGFuZCBnZW5lcmljIFVJIGVsZW1lbnRzIGFuZCBmdW5jdGlvbmFsaXRpZXNcclxuICpcclxuICogV2Vic2l0ZSA6IC0+IGh0dHA6Ly93d3cudHVyYm9ndWkub3JnXHJcbiAqIExpY2Vuc2UgOiAtPiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wLiBZb3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuXHJcbiAqIExpY2Vuc2UgVXJsIDogLT4gaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wXHJcbiAqIENvcHlSaWdodCA6IC0+IENvcHlyaWdodCAyMDE4IEVkZXJ0b25lIEFkdmFuZGVkIFNvbHV0aW9ucy4gaHR0cHM6Ly93d3cuZWRlcnRvbmUuY29tXHJcbiAqL1xyXG5cclxuaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyB0cmlnZ2VyLCBhbmltYXRlLCB0cmFuc2l0aW9uLCBzdHlsZSB9IGZyb20gJ0Bhbmd1bGFyL2FuaW1hdGlvbnMnO1xyXG5cclxuXHJcbi8qKlxyXG4gKiBGYWRlIGFuaW1hdGlvbnNcclxuICovXHJcbkBJbmplY3RhYmxlKClcclxuZXhwb3J0IGNsYXNzIEZhZGVBbmltYXRpb25DbGFzcyB7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBHZXQgYSBjdXN0b20gdHJpZ2dlciB0byBjcmVhdGUgZmFkZSBhbmltYXRpb25zIHdoZW4gY29tcG9uZW50cyBhcmUgYWRkZWQgb3IgcmVtb3ZlZCBmcm9tIHRoZSBhcHBsaWNhdGlvblxyXG4gICAgICpcclxuICAgICAqIEBwYXJhbSB0cmlnZ2VyTmFtZSBUaGUgbmFtZSBmb3IgdGhlIHRyaWdnZXIgd2Ugd2FudCB0byBjcmVhdGVcclxuICAgICAqIEBwYXJhbSBlbnRlciBUaGUgdGltZSBhbmQgZWFzaW5nIHRoYXQgd2Ugd2FudCB0byB1c2UgZm9yIHRoZSBlbnRlciBzdGF0ZVxyXG4gICAgICogQHBhcmFtIGxlYXZlIFRoZSB0aW1lIGFuZCBlYXNpbmcgdGhhdCB3ZSB3YW50IHRvIHVzZSBmb3IgdGhlIGxlYXZlIHN0YXRlXHJcbiAgICAgKi9cclxuICAgIHN0YXRpYyBnZXRUcmlnZ2VyKHRyaWdnZXJOYW1lOiBzdHJpbmcsIGVudGVyID0gJzFzIGVhc2UnLCBsZWF2ZSA9ICczMDBtcyBlYXNlJykge1xyXG5cclxuICAgICAgICByZXR1cm4gdHJpZ2dlcih0cmlnZ2VyTmFtZSwgW1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdHJhbnNpdGlvbigndm9pZCA9PiAqJywgW3N0eWxlKHtvcGFjaXR5OiAwfSksIGFuaW1hdGUoZW50ZXIsIHN0eWxlKHsgb3BhY2l0eTogMSB9KSldKSxcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zaXRpb24oJyogPT4gdm9pZCcsIFthbmltYXRlKGxlYXZlLCBzdHlsZSh7b3BhY2l0eTogMH0pKV0pXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIF0pO1xyXG4gICAgfVxyXG59XHJcbiJdfQ==
@@ -50,6 +50,10 @@ export class DialogSingleSelectionListComponent extends DialogBaseComponent {
50
50
  * It may be filtered or not depending on this component setup and the user input
51
51
  */
52
52
  this.filteredOptions = [];
53
+ /**
54
+ * Stores the number of options that are currently visible to the user
55
+ */
56
+ this.filteredOptionsCount = 0;
53
57
  /**
54
58
  * Contains the original list of elements that are provided to be listed on this component before
55
59
  * being possibly filtered. It is only used as a backup, not shown to the user
@@ -76,6 +80,7 @@ export class DialogSingleSelectionListComponent extends DialogBaseComponent {
76
80
  throw new Error('DialogSingleSelectionListComponent expects one or more options');
77
81
  }
78
82
  this.originalOptions = data.options;
83
+ this.filteredOptionsCount = this.originalOptions.length;
79
84
  for (let option of this.originalOptions) {
80
85
  this.filteredOptions.push(option);
81
86
  this.originalOptionsFullTextSearch.push(StringUtils.formatForFullTextSearch(option));
@@ -87,17 +92,33 @@ export class DialogSingleSelectionListComponent extends DialogBaseComponent {
87
92
  getListItemsContainerMaxheight() {
88
93
  return (this.browserService.getWindowHeight() * 0.6) + 'px';
89
94
  }
95
+ /**
96
+ * If the user presses enter key and there's only one element filtered in the list, we will close this dialog
97
+ * setting that element as the selected.
98
+ */
99
+ onIntroKeyPress() {
100
+ if (this.filteredOptionsCount === 1) {
101
+ for (let i = 0; i < this.originalOptionsFullTextSearch.length; i++) {
102
+ if (this.filteredOptions[i] !== '') {
103
+ this.closeDialog(i);
104
+ return;
105
+ }
106
+ }
107
+ }
108
+ }
90
109
  /**
91
110
  * When the user types a value on the input element to filter the list, this method will perform
92
111
  * that filtering and refresh the list
93
112
  */
94
113
  onSearchChange(input) {
95
114
  this.selectedItemIndex = -1;
115
+ this.filteredOptionsCount = 0;
96
116
  let inputValue = StringUtils.formatForFullTextSearch(input.value);
97
117
  for (let i = 0; i < this.originalOptionsFullTextSearch.length; i++) {
98
118
  if (inputValue === '' ||
99
119
  this.originalOptionsFullTextSearch[i].indexOf(inputValue) >= 0) {
100
120
  this.filteredOptions[i] = this.originalOptions[i];
121
+ this.filteredOptionsCount++;
101
122
  }
102
123
  else {
103
124
  this.filteredOptions[i] = '';
@@ -105,13 +126,13 @@ export class DialogSingleSelectionListComponent extends DialogBaseComponent {
105
126
  }
106
127
  }
107
128
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: DialogSingleSelectionListComponent, deps: [{ token: i0.ElementRef }, { token: i1.MatDialogRef }, { token: i2.BrowserService }, { token: MAT_DIALOG_DATA }], target: i0.ɵɵFactoryTarget.Component }); }
108
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.8", type: DialogSingleSelectionListComponent, isStandalone: true, selector: "tg-dialog-single-selection-list", providers: [], usesInheritance: true, ngImport: i0, template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i5.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: TurboGuiAngularModule }, { kind: "directive", type: i6.AutoFocusOnDisplayDirective, selector: "[autoFocusOnDisplay]" }] }); }
129
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.8", type: DialogSingleSelectionListComponent, isStandalone: true, selector: "tg-dialog-single-selection-list", providers: [], usesInheritance: true, ngImport: i0, template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (keyup.enter)=\"onIntroKeyPress()\"\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i5.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: TurboGuiAngularModule }, { kind: "directive", type: i6.AutoFocusOnDisplayDirective, selector: "[autoFocusOnDisplay]" }] }); }
109
130
  }
110
131
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: DialogSingleSelectionListComponent, decorators: [{
111
132
  type: Component,
112
- args: [{ selector: 'tg-dialog-single-selection-list', standalone: true, imports: [CommonModule, MatInputModule, MatFormFieldModule, TurboGuiAngularModule], providers: [], template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"] }]
133
+ args: [{ selector: 'tg-dialog-single-selection-list', standalone: true, imports: [CommonModule, MatInputModule, MatFormFieldModule, TurboGuiAngularModule], providers: [], template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (keyup.enter)=\"onIntroKeyPress()\"\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"] }]
113
134
  }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i1.MatDialogRef }, { type: i2.BrowserService }, { type: undefined, decorators: [{
114
135
  type: Inject,
115
136
  args: [MAT_DIALOG_DATA]
116
137
  }] }] });
117
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog-single-selection-list.component.js","sourceRoot":"","sources":["../../../../../../../projects/turbogui-angular/src/main/view/components/dialog-single-selection-list/dialog-single-selection-list.component.ts","../../../../../../../projects/turbogui-angular/src/main/view/components/dialog-single-selection-list/dialog-single-selection-list.component.html"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { Component, ElementRef, Inject } from '@angular/core';\r\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\r\nimport { DialogBaseComponent } from '../dialog-base/dialog-base.component';\r\nimport { BrowserService } from '../../../controller/browser.service';\r\nimport { StringUtils } from 'turbocommons-ts';\r\nimport { CommonModule } from '@angular/common';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { TurboGuiAngularModule } from '../../../model/modules/turbogui-angular.module';\n\r\n\r\n/**\r\n * A dialog component which allows us to select one single item from a list. The elements on that list are displayed on a table\r\n * which may show a scroll if necessary when there are many elements on the list.\r\n * \r\n * It also allows us to filter inside the list with a custom text that we can type into a search input, which is optional.\r\n *\r\n * texts parameter must contain the title, the description (optional), the filter input title (optional), and the submit button caption.\r\n * If that caption is not provided, the selection will be automatically performed once user clicks on an element on the list. Otherwise, \r\n * the element will be selected on the list and the selection will be performed once the user clicks on the submit button.\r\n *\r\n * options parameter must contain the list of elements that will be displayed to the user  \r\n */\r\n@Component({\r\n  \tselector: 'tg-dialog-single-selection-list',\r\n  \tstandalone: true,\r\n\timports: [CommonModule, MatInputModule, MatFormFieldModule, TurboGuiAngularModule],\r\n\tproviders: [],\r\n  \ttemplateUrl: './dialog-single-selection-list.component.html',\r\n  \tstyleUrls: ['./dialog-single-selection-list.component.scss']\r\n})\r\n\r\n\r\nexport class DialogSingleSelectionListComponent extends DialogBaseComponent {\r\n    \r\n    \r\n    static readonly DIALOG_CLASS_NAME = 'DialogSingleSelectionListComponent';\r\n    \r\n    \r\n    /**\r\n     * Declare a reference to the string utils class so it can be used on the html part\r\n     */\r\n    stringUtils = StringUtils;\r\n    \r\n    /** \r\n     * Contains the list of elements that will be directly shown to the user at the component list.\r\n     * It may be filtered or not depending on this component setup and the user input\r\n     */\r\n    filteredOptions:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * Contains the original list of elements that are provided to be listed on this component before\r\n     * being possibly filtered. It is only used as a backup, not shown to the user\r\n     */\r\n    private originalOptions:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * The same list as the originally provided but processed for a better text search.\r\n     * It will be used to perform the search, but not shown to the user.\r\n     */\r\n    private originalOptionsFullTextSearch:string[] = [];\r\n       \r\n    \r\n    /**\r\n     * Stores the index for the element that's been selected by the user\r\n     */\r\n    selectedItemIndex = -1;\r\n    \r\n\r\n    constructor(public elementRef: ElementRef,\r\n    \t\t\tpublic dialogRef: MatDialogRef<DialogBaseComponent>,\r\n                public browserService: BrowserService,\r\n                @Inject(MAT_DIALOG_DATA) public data: any) {\r\n\r\n        super(elementRef, dialogRef);\r\n\r\n        if (data.texts.length < 1) {\r\n\r\n            throw new Error('DialogSingleSelectionListComponent expects 4 texts: The title, the description (optional), the filter input title (optional), and the submit button caption');\r\n        }\r\n        \r\n        if (data.options.length < 1) {\r\n\r\n            throw new Error('DialogSingleSelectionListComponent expects one or more options');\r\n        }\r\n        \r\n        this.originalOptions = data.options;\r\n        \r\n        for(let option of this.originalOptions){\r\n            \r\n            this.filteredOptions.push(option);\r\n            this.originalOptionsFullTextSearch.push(StringUtils.formatForFullTextSearch(option));\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Method to calculate the max possible height that the list items container is allowed to have\r\n     */\r\n    getListItemsContainerMaxheight(){\r\n        \r\n        return (this.browserService.getWindowHeight() * 0.6) + 'px';\r\n    }\r\n\r\n\r\n    /**\r\n     * When the user types a value on the input element to filter the list, this method will perform\r\n     * that filtering and refresh the list\r\n     */\r\n    onSearchChange(input:HTMLInputElement){\r\n\r\n        this.selectedItemIndex = -1;\r\n       \r\n        let inputValue = StringUtils.formatForFullTextSearch(input.value);\r\n\r\n        for (let i = 0; i < this.originalOptionsFullTextSearch.length; i++){\r\n\r\n            if(inputValue === '' ||\r\n               this.originalOptionsFullTextSearch[i].indexOf(inputValue) >= 0){\r\n\r\n                this.filteredOptions[i] = this.originalOptions[i];\r\n\r\n            }else{\r\n\r\n                this.filteredOptions[i] = '';\r\n            }\r\n        }\r\n    }\r\n    \r\n    /**\r\n     * This method is used to greatly improve ngFor performance with arrays of primitive values. It tells the refresh\r\n     * function to work by index instead of by value. The change in performance when refreshing the list is massive. \r\n     */\r\n    trackByFn = (index: number, _value: string) => index;\r\n}\r\n","<h3>\r\n    {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n    {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n    class=\"searchItemInputContainer\">\r\n    \r\n    <mat-label>{{data.texts[2]}}</mat-label>\r\n    <input matInput autoFocusOnDisplay\r\n        (input)=\"onSearchChange($event.target)\">\r\n        \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n    [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n    <div class=\"listItemContainer\"\r\n        [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n        *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n        (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n    \r\n        <p *ngIf=\"item !== ''\">\r\n            {{item}}\r\n        </p>\r\n    \r\n    </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n    [disabled]=\"selectedItemIndex < 0\"\r\n    (click)=\"closeDialog(selectedItemIndex)\"\r\n    *ngIf=\"data.texts.length > 3\">\r\n\r\n    {{data.texts[3]}}\r\n    \r\n</button>"]}
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog-single-selection-list.component.js","sourceRoot":"","sources":["../../../../../../../projects/turbogui-angular/src/main/view/components/dialog-single-selection-list/dialog-single-selection-list.component.ts","../../../../../../../projects/turbogui-angular/src/main/view/components/dialog-single-selection-list/dialog-single-selection-list.component.html"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { Component, ElementRef, Inject } from '@angular/core';\r\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\r\nimport { DialogBaseComponent } from '../dialog-base/dialog-base.component';\r\nimport { BrowserService } from '../../../controller/browser.service';\r\nimport { StringUtils } from 'turbocommons-ts';\r\nimport { CommonModule } from '@angular/common';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { TurboGuiAngularModule } from '../../../model/modules/turbogui-angular.module';\n\r\n\r\n/**\r\n * A dialog component which allows us to select one single item from a list. The elements on that list are displayed on a table\r\n * which may show a scroll if necessary when there are many elements on the list.\r\n * \r\n * It also allows us to filter inside the list with a custom text that we can type into a search input, which is optional.\r\n *\r\n * texts parameter must contain the title, the description (optional), the filter input title (optional), and the submit button caption.\r\n * If that caption is not provided, the selection will be automatically performed once user clicks on an element on the list. Otherwise, \r\n * the element will be selected on the list and the selection will be performed once the user clicks on the submit button.\r\n *\r\n * options parameter must contain the list of elements that will be displayed to the user  \r\n */\r\n@Component({\r\n  \tselector: 'tg-dialog-single-selection-list',\r\n  \tstandalone: true,\r\n\timports: [CommonModule, MatInputModule, MatFormFieldModule, TurboGuiAngularModule],\r\n\tproviders: [],\r\n  \ttemplateUrl: './dialog-single-selection-list.component.html',\r\n  \tstyleUrls: ['./dialog-single-selection-list.component.scss']\r\n})\r\n\r\n\r\nexport class DialogSingleSelectionListComponent extends DialogBaseComponent {\r\n    \r\n    \r\n    static readonly DIALOG_CLASS_NAME = 'DialogSingleSelectionListComponent';\r\n    \r\n    \r\n    /**\r\n     * Declare a reference to the string utils class so it can be used on the html part\r\n     */\r\n    stringUtils = StringUtils;\r\n    \r\n    /** \r\n     * Contains the list of elements that will be directly shown to the user at the component list.\r\n     * It may be filtered or not depending on this component setup and the user input\r\n     */\r\n    filteredOptions:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * Stores the number of options that are currently visible to the user\r\n     */\r\n    filteredOptionsCount = 0;\r\n    \r\n    \r\n    /**\r\n     * Contains the original list of elements that are provided to be listed on this component before\r\n     * being possibly filtered. It is only used as a backup, not shown to the user\r\n     */\r\n    private originalOptions:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * The same list as the originally provided but processed for a better text search.\r\n     * It will be used to perform the search, but not shown to the user.\r\n     */\r\n    private originalOptionsFullTextSearch:string[] = [];\r\n       \r\n    \r\n    /**\r\n     * Stores the index for the element that's been selected by the user\r\n     */\r\n    selectedItemIndex = -1;\r\n    \r\n    \r\n    constructor(public elementRef: ElementRef,\r\n    \t\t\tpublic dialogRef: MatDialogRef<DialogBaseComponent>,\r\n                public browserService: BrowserService,\r\n                @Inject(MAT_DIALOG_DATA) public data: any) {\r\n\r\n        super(elementRef, dialogRef);\r\n\r\n        if (data.texts.length < 1) {\r\n\r\n            throw new Error('DialogSingleSelectionListComponent expects 4 texts: The title, the description (optional), the filter input title (optional), and the submit button caption');\r\n        }\r\n        \r\n        if (data.options.length < 1) {\r\n\r\n            throw new Error('DialogSingleSelectionListComponent expects one or more options');\r\n        }\r\n        \r\n        this.originalOptions = data.options;\r\n        this.filteredOptionsCount = this.originalOptions.length;\r\n        \r\n        for(let option of this.originalOptions){\r\n            \r\n            this.filteredOptions.push(option);\r\n            this.originalOptionsFullTextSearch.push(StringUtils.formatForFullTextSearch(option));\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Method to calculate the max possible height that the list items container is allowed to have\r\n     */\r\n    getListItemsContainerMaxheight(){\r\n        \r\n        return (this.browserService.getWindowHeight() * 0.6) + 'px';\r\n    }\r\n    \r\n    \r\n    /**\r\n     * If the user presses enter key and there's only one element filtered in the list, we will close this dialog\r\n     * setting that element as the selected.\r\n     */\r\n    onIntroKeyPress(){\r\n\r\n        if(this.filteredOptionsCount === 1){\r\n            \r\n            for (let i = 0; i < this.originalOptionsFullTextSearch.length; i++){\r\n                \r\n                if(this.filteredOptions[i] !== ''){\r\n                    \r\n                    this.closeDialog(i);\r\n                    \r\n                    return;\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * When the user types a value on the input element to filter the list, this method will perform\r\n     * that filtering and refresh the list\r\n     */\r\n    onSearchChange(input:HTMLInputElement){\r\n\r\n        this.selectedItemIndex = -1;\r\n        this.filteredOptionsCount = 0;\r\n       \r\n        let inputValue = StringUtils.formatForFullTextSearch(input.value);\r\n\r\n        for (let i = 0; i < this.originalOptionsFullTextSearch.length; i++){\r\n\r\n            if(inputValue === '' ||\r\n               this.originalOptionsFullTextSearch[i].indexOf(inputValue) >= 0){\r\n\r\n                this.filteredOptions[i] = this.originalOptions[i];\r\n                this.filteredOptionsCount ++;\r\n\r\n            }else{\r\n\r\n                this.filteredOptions[i] = '';\r\n            }\r\n        }\r\n    }\r\n    \r\n    /**\r\n     * This method is used to greatly improve ngFor performance with arrays of primitive values. It tells the refresh\r\n     * function to work by index instead of by value. The change in performance when refreshing the list is massive. \r\n     */\r\n    trackByFn = (index: number, _value: string) => index;\r\n}\r\n","<h3>\r\n    {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n    {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n    class=\"searchItemInputContainer\">\r\n    \r\n    <mat-label>{{data.texts[2]}}</mat-label>\r\n    <input matInput autoFocusOnDisplay\r\n        (keyup.enter)=\"onIntroKeyPress()\"\r\n        (input)=\"onSearchChange($event.target)\">\r\n        \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n    [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n    <div class=\"listItemContainer\"\r\n        [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n        *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n        (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n    \r\n        <p *ngIf=\"item !== ''\">\r\n            {{item}}\r\n        </p>\r\n    \r\n    </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n    [disabled]=\"selectedItemIndex < 0\"\r\n    (click)=\"closeDialog(selectedItemIndex)\"\r\n    *ngIf=\"data.texts.length > 3\">\r\n\r\n    {{data.texts[3]}}\r\n    \r\n</button>"]}
@@ -8,6 +8,7 @@ import * as i3 from '@angular/material/button';
8
8
  import { MatButtonModule } from '@angular/material/button';
9
9
  import * as i2 from '@angular/common';
10
10
  import { CommonModule } from '@angular/common';
11
+ import * as i1$2 from '@angular/animations';
11
12
  import { style, animate, transition, trigger } from '@angular/animations';
12
13
  import { ComponentPortal, DomPortalOutlet } from '@angular/cdk/portal';
13
14
  import * as i2$1 from '@angular/material/datepicker';
@@ -500,10 +501,10 @@ class FadeAnimationClass {
500
501
  * @param enter The time and easing that we want to use for the enter state
501
502
  * @param leave The time and easing that we want to use for the leave state
502
503
  */
503
- static getTrigger(triggerName, enter = '1s ease', leave = '0s ease') {
504
+ static getTrigger(triggerName, enter = '1s ease', leave = '300ms ease') {
504
505
  return trigger(triggerName, [
505
- transition(':enter', [style({ opacity: 0 }), animate(enter)]),
506
- transition(':leave', [style({ opacity: 1 }), animate(leave, style({ opacity: 0 }))])
506
+ transition('void => *', [style({ opacity: 0 }), animate(enter, style({ opacity: 1 }))]),
507
+ transition('* => void', [animate(leave, style({ opacity: 0 }))])
507
508
  ]);
508
509
  }
509
510
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: FadeAnimationClass, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
@@ -693,9 +694,10 @@ class DialogService extends SingletoneStrictClass {
693
694
  * Change the application visual appearance so the user perceives that the application is
694
695
  * currently busy. While modal busy state is enabled, no user input is possible neither via
695
696
  * keyboard, mouse or touch. Use this state when performing server requests or operations that
696
- * must block the user interaction with the application.
697
+ * must block the user interaction with the application. To allow user interaction again, you must
698
+ * call removeModalBusyState()
697
699
  *
698
- * Note: We can modify the busy state component that is shown by this method. To do it, we must
700
+ * Notice: We can modify the busy state visual component that is shown by this method. To do it, we must
699
701
  * set this.customBusyStateComponentClass property with our own custom busy state component class. (We can do it at
700
702
  * our main application component constructor for example). Our custom component must extend the
701
703
  * BusyStateBaseComponent one to add its own visual appearance.
@@ -723,7 +725,7 @@ class DialogService extends SingletoneStrictClass {
723
725
  return this._isShowingBusyState;
724
726
  }
725
727
  /**
726
- * Remove the application busy state and restore it back to normal user interaction
728
+ * Cancel the application busy state and restore it back to normal so user interaction is allowed again
727
729
  */
728
730
  removeModalBusyState() {
729
731
  if (!this._isEnabled || !this._isShowingBusyState) {
@@ -1512,9 +1514,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImpor
1512
1514
  * Manages adding, removing and manipulating the application views
1513
1515
  */
1514
1516
  class ViewsService extends SingletoneStrictClass {
1515
- constructor(componentFactoryResolver) {
1517
+ constructor(animationBuilder) {
1516
1518
  super(ViewsService);
1517
- this.componentFactoryResolver = componentFactoryResolver;
1519
+ this.animationBuilder = animationBuilder;
1518
1520
  /**
1519
1521
  * See getter method for docs
1520
1522
  */
@@ -1523,6 +1525,10 @@ class ViewsService extends SingletoneStrictClass {
1523
1525
  * See setter method for docs
1524
1526
  */
1525
1527
  this._viewContainerRef = null;
1528
+ /**
1529
+ * Contains the reference to the currently loaded view component
1530
+ */
1531
+ this._currentComponentRef = null;
1526
1532
  }
1527
1533
  /**
1528
1534
  * Tells if there's any view currently loaded on the application views container
@@ -1552,32 +1558,71 @@ class ViewsService extends SingletoneStrictClass {
1552
1558
  *
1553
1559
  * @return The instance of the newly added and created view.
1554
1560
  */
1555
- pushView(view) {
1561
+ async pushView(view) {
1562
+ this.verifyViewsContainerExist();
1563
+ // If the loaded view is the same as the specified one, we will do nothing
1564
+ if (this._loadedViewClass === view) {
1565
+ return this._currentComponentRef;
1566
+ }
1556
1567
  // If a view is already loaded, we will unload it first
1557
1568
  if (this._loadedViewClass !== null) {
1558
- // If the loaded view is the same as the specified one, we will do nothing
1559
- if (this._loadedViewClass === view) {
1560
- return;
1561
- }
1562
- this.popView();
1563
- }
1564
- this.verifyViewsContainerExist();
1565
- const factory = this.componentFactoryResolver.resolveComponentFactory(view);
1566
- const componentRef = this._viewContainerRef.createComponent(factory);
1567
- componentRef.changeDetectorRef.detectChanges();
1568
- this._loadedViewClass = view;
1569
- return componentRef;
1569
+ await this.removeCurrentView();
1570
+ }
1571
+ // Create the new view
1572
+ const newComponentRef = this._viewContainerRef.createComponent(view);
1573
+ newComponentRef.changeDetectorRef.detectChanges();
1574
+ // Set initial opacity to 0
1575
+ newComponentRef.location.nativeElement.style.opacity = '0';
1576
+ // Fade in the new view
1577
+ const fadeInPlayer = this.animationBuilder.build([
1578
+ style({ opacity: 0 }),
1579
+ animate('300ms ease', style({ opacity: 1 }))
1580
+ ]).create(newComponentRef.location.nativeElement);
1581
+ await new Promise((resolve) => {
1582
+ fadeInPlayer.onDone(() => {
1583
+ this._currentComponentRef = newComponentRef;
1584
+ this._loadedViewClass = view;
1585
+ resolve();
1586
+ });
1587
+ fadeInPlayer.play();
1588
+ });
1589
+ return this._currentComponentRef;
1570
1590
  }
1571
1591
  /**
1572
1592
  * Delete the currently loaded view from the views container, and leave it empty.
1573
1593
  * If no view is currently loaded, this method will do nothing
1574
1594
  */
1575
- popView() {
1595
+ async popView() {
1576
1596
  this.verifyViewsContainerExist();
1577
- if (this._loadedViewClass !== null) {
1578
- this._viewContainerRef.clear();
1597
+ if (this._loadedViewClass !== null && this._currentComponentRef) {
1598
+ await this.removeCurrentView();
1579
1599
  }
1580
- this._loadedViewClass = null;
1600
+ }
1601
+ /**
1602
+ * Aux method to remove the currently loaded view from the views container using a promise
1603
+ */
1604
+ removeCurrentView() {
1605
+ return new Promise((resolve) => {
1606
+ if (this._currentComponentRef) {
1607
+ const element = this._currentComponentRef.location.nativeElement;
1608
+ const fadeOutPlayer = this.animationBuilder.build([
1609
+ style({ opacity: 1 }),
1610
+ animate('300ms ease', style({ opacity: 0 }))
1611
+ ]).create(element);
1612
+ fadeOutPlayer.onDone(() => {
1613
+ if (this._viewContainerRef !== null) {
1614
+ this._viewContainerRef.clear();
1615
+ }
1616
+ this._currentComponentRef = null;
1617
+ this._loadedViewClass = null;
1618
+ resolve();
1619
+ });
1620
+ fadeOutPlayer.play();
1621
+ }
1622
+ else {
1623
+ resolve();
1624
+ }
1625
+ });
1581
1626
  }
1582
1627
  /**
1583
1628
  * Auxiliary method to test if the views container instance is available on the application
@@ -1587,7 +1632,7 @@ class ViewsService extends SingletoneStrictClass {
1587
1632
  throw new Error('Views container not defined. Please declare a <views-container> element in your application');
1588
1633
  }
1589
1634
  }
1590
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: ViewsService, deps: [{ token: i0.ComponentFactoryResolver }], target: i0.ɵɵFactoryTarget.Injectable }); }
1635
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: ViewsService, deps: [{ token: i1$2.AnimationBuilder }], target: i0.ɵɵFactoryTarget.Injectable }); }
1591
1636
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: ViewsService, providedIn: 'root' }); }
1592
1637
  }
1593
1638
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: ViewsService, decorators: [{
@@ -1595,7 +1640,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImpor
1595
1640
  args: [{
1596
1641
  providedIn: 'root',
1597
1642
  }]
1598
- }], ctorParameters: () => [{ type: i0.ComponentFactoryResolver }] });
1643
+ }], ctorParameters: () => [{ type: i1$2.AnimationBuilder }] });
1599
1644
 
1600
1645
  /**
1601
1646
  * TurboGUI is A library that helps with the most common and generic UI elements and functionalities
@@ -1740,6 +1785,10 @@ class DialogSingleSelectionListComponent extends DialogBaseComponent {
1740
1785
  * It may be filtered or not depending on this component setup and the user input
1741
1786
  */
1742
1787
  this.filteredOptions = [];
1788
+ /**
1789
+ * Stores the number of options that are currently visible to the user
1790
+ */
1791
+ this.filteredOptionsCount = 0;
1743
1792
  /**
1744
1793
  * Contains the original list of elements that are provided to be listed on this component before
1745
1794
  * being possibly filtered. It is only used as a backup, not shown to the user
@@ -1766,6 +1815,7 @@ class DialogSingleSelectionListComponent extends DialogBaseComponent {
1766
1815
  throw new Error('DialogSingleSelectionListComponent expects one or more options');
1767
1816
  }
1768
1817
  this.originalOptions = data.options;
1818
+ this.filteredOptionsCount = this.originalOptions.length;
1769
1819
  for (let option of this.originalOptions) {
1770
1820
  this.filteredOptions.push(option);
1771
1821
  this.originalOptionsFullTextSearch.push(StringUtils.formatForFullTextSearch(option));
@@ -1777,17 +1827,33 @@ class DialogSingleSelectionListComponent extends DialogBaseComponent {
1777
1827
  getListItemsContainerMaxheight() {
1778
1828
  return (this.browserService.getWindowHeight() * 0.6) + 'px';
1779
1829
  }
1830
+ /**
1831
+ * If the user presses enter key and there's only one element filtered in the list, we will close this dialog
1832
+ * setting that element as the selected.
1833
+ */
1834
+ onIntroKeyPress() {
1835
+ if (this.filteredOptionsCount === 1) {
1836
+ for (let i = 0; i < this.originalOptionsFullTextSearch.length; i++) {
1837
+ if (this.filteredOptions[i] !== '') {
1838
+ this.closeDialog(i);
1839
+ return;
1840
+ }
1841
+ }
1842
+ }
1843
+ }
1780
1844
  /**
1781
1845
  * When the user types a value on the input element to filter the list, this method will perform
1782
1846
  * that filtering and refresh the list
1783
1847
  */
1784
1848
  onSearchChange(input) {
1785
1849
  this.selectedItemIndex = -1;
1850
+ this.filteredOptionsCount = 0;
1786
1851
  let inputValue = StringUtils.formatForFullTextSearch(input.value);
1787
1852
  for (let i = 0; i < this.originalOptionsFullTextSearch.length; i++) {
1788
1853
  if (inputValue === '' ||
1789
1854
  this.originalOptionsFullTextSearch[i].indexOf(inputValue) >= 0) {
1790
1855
  this.filteredOptions[i] = this.originalOptions[i];
1856
+ this.filteredOptionsCount++;
1791
1857
  }
1792
1858
  else {
1793
1859
  this.filteredOptions[i] = '';
@@ -1795,11 +1861,11 @@ class DialogSingleSelectionListComponent extends DialogBaseComponent {
1795
1861
  }
1796
1862
  }
1797
1863
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: DialogSingleSelectionListComponent, deps: [{ token: i0.ElementRef }, { token: i1.MatDialogRef }, { token: BrowserService }, { token: MAT_DIALOG_DATA }], target: i0.ɵɵFactoryTarget.Component }); }
1798
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.8", type: DialogSingleSelectionListComponent, isStandalone: true, selector: "tg-dialog-single-selection-list", providers: [], usesInheritance: true, ngImport: i0, template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i5.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: TurboGuiAngularModule }, { kind: "directive", type: AutoFocusOnDisplayDirective, selector: "[autoFocusOnDisplay]" }] }); }
1864
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.8", type: DialogSingleSelectionListComponent, isStandalone: true, selector: "tg-dialog-single-selection-list", providers: [], usesInheritance: true, ngImport: i0, template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (keyup.enter)=\"onIntroKeyPress()\"\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i5.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: TurboGuiAngularModule }, { kind: "directive", type: AutoFocusOnDisplayDirective, selector: "[autoFocusOnDisplay]" }] }); }
1799
1865
  }
1800
1866
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: DialogSingleSelectionListComponent, decorators: [{
1801
1867
  type: Component,
1802
- args: [{ selector: 'tg-dialog-single-selection-list', standalone: true, imports: [CommonModule, MatInputModule, MatFormFieldModule, TurboGuiAngularModule], providers: [], template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"] }]
1868
+ args: [{ selector: 'tg-dialog-single-selection-list', standalone: true, imports: [CommonModule, MatInputModule, MatFormFieldModule, TurboGuiAngularModule], providers: [], template: "<h3>\r\n {{data.texts[0]}}\r\n</h3>\r\n\r\n<!-- Here goes the dialog subtitle. Leave it blank if you don't need it -->\r\n<p *ngIf=\"data.texts.length > 1 &amp;&amp; !stringUtils.isEmpty(data.texts[1])\">\r\n {{data.texts[1]}}\r\n</p>\r\n\r\n<mat-form-field *ngIf=\"data.texts.length > 2 &amp;&amp; !stringUtils.isEmpty(data.texts[2])\"\r\n class=\"searchItemInputContainer\">\r\n \r\n <mat-label>{{data.texts[2]}}</mat-label>\r\n <input matInput autoFocusOnDisplay\r\n (keyup.enter)=\"onIntroKeyPress()\"\r\n (input)=\"onSearchChange($event.target)\">\r\n \r\n</mat-form-field>\r\n\r\n<!-- here goes the list of elements that will be shown to the user -->\r\n<div class=\"listItemsContainer\"\r\n [style.max-height]=\"getListItemsContainerMaxheight()\">\r\n\r\n <div class=\"listItemContainer\"\r\n [style.background-color]=\"selectedItemIndex === i ? '#90d1ffad' : (i % 2 === 0 ? 'initial' : '#00000009')\"\r\n *ngFor=\"let item of filteredOptions; let i = index; trackBy: trackByFn\"\r\n (click)=\"data.texts.length < 4 ? closeDialog(i) : selectedItemIndex = i\">\r\n \r\n <p *ngIf=\"item !== ''\">\r\n {{item}}\r\n </p>\r\n \r\n </div>\r\n\r\n</div>\r\n\r\n<button mat-raised-button color=\"primary\"\r\n [disabled]=\"selectedItemIndex < 0\"\r\n (click)=\"closeDialog(selectedItemIndex)\"\r\n *ngIf=\"data.texts.length > 3\">\r\n\r\n {{data.texts[3]}}\r\n \r\n</button>", styles: [":host{min-height:300px}h3{margin-top:0;margin-bottom:10px}p{margin-top:0;margin-bottom:5px}.searchItemInputContainer{width:100%;margin-top:0;margin-bottom:0}.listItemsContainer{overflow-y:auto;border:1px solid rgba(0,0,0,.2);border-radius:4px;margin-bottom:5px}.listItemContainer p{line-height:36px;width:calc(100% - 12px);margin:0 0 0 6px}button{float:right}\n"] }]
1803
1869
  }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i1.MatDialogRef }, { type: BrowserService }, { type: undefined, decorators: [{
1804
1870
  type: Inject,
1805
1871
  args: [MAT_DIALOG_DATA]