lib-portal-angular 0.0.53 → 0.0.55

Sign up to get free protection for your applications and to get access to all the features.
@@ -32,11 +32,11 @@ export class CustomSwitchComponent {
32
32
  }
33
33
  }
34
34
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CustomSwitchComponent, deps: [{ token: i1.AuthService }], target: i0.ɵɵFactoryTarget.Component }); }
35
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: CustomSwitchComponent, selector: "argenta-custom-switch", inputs: { checked: "checked", label: "label", permissions: "permissions" }, outputs: { switchChange: "switchChange" }, ngImport: i0, template: "<ng-container *ngIf=\"hasPermission()\">\n <div class=\"form-check form-switch\">\n <input \n class=\"form-check-input\" \n type=\"checkbox\" \n [checked]=\"checked\" \n (change)=\"toggleSwitch()\" \n id=\"flexSwitchCheckDefault\" \n />\n <label class=\"form-check-label\" for=\"flexSwitchCheckDefault\">\n {{ label }}\n </label>\n </div>\n</ng-container>\n", styles: ["@charset \"UTF-8\";.form-check{display:flex;align-items:center}.form-check-input{width:50px;height:24px;margin-right:10px;cursor:pointer}.form-check-input:checked{background-color:#00444c;border-color:#00444c}.form-check-input:not(:checked){border-color:#3ec9d6;box-shadow:0 0 5px #e5e5e5}.form-check-label{color:#00444c;font-family:Inter,sans-serif;font-size:16px;font-weight:600;height:24px;line-height:24px}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
35
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: CustomSwitchComponent, selector: "argenta-custom-switch", inputs: { checked: "checked", label: "label", permissions: "permissions" }, outputs: { switchChange: "switchChange" }, ngImport: i0, template: "<ng-container *ngIf=\"hasPermission()\">\n <div class=\"form-check form-switch\">\n <input \n class=\"form-check-input\" \n type=\"checkbox\" \n [checked]=\"checked\" \n (change)=\"toggleSwitch()\" \n id=\"flexSwitchCheckDefault\" \n />\n <label class=\"form-check-label\" for=\"flexSwitchCheckDefault\">\n {{ label }}\n </label>\n </div>\n</ng-container>\n", styles: ["@charset \"UTF-8\";.form-check{display:flex;align-items:center}.form-check-input{width:60px;height:34px;margin-right:10px;cursor:pointer;position:relative;appearance:none;background-color:#e5e5e5;border:2px solid #00444C;border-radius:34px;outline:none;transition:background-color .3s,border-color .3s}.form-check-input:focus{outline:none;box-shadow:none}.form-check-input:checked{background-color:#00444c;border-color:#00444c}.form-check-input:before{content:\"\";position:absolute;width:26px;height:26px;background-color:#00444c;border-radius:50%;top:3px;left:3px;transition:transform .3s,background-color .3s}.form-check-input:checked:before{transform:translate(26px);background-color:#fff}.form-check-label{color:#00444c;font-family:Inter,sans-serif;font-size:16px;font-weight:600;height:24px;line-height:24px}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
36
36
  }
37
37
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CustomSwitchComponent, decorators: [{
38
38
  type: Component,
39
- args: [{ selector: 'argenta-custom-switch', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"hasPermission()\">\n <div class=\"form-check form-switch\">\n <input \n class=\"form-check-input\" \n type=\"checkbox\" \n [checked]=\"checked\" \n (change)=\"toggleSwitch()\" \n id=\"flexSwitchCheckDefault\" \n />\n <label class=\"form-check-label\" for=\"flexSwitchCheckDefault\">\n {{ label }}\n </label>\n </div>\n</ng-container>\n", styles: ["@charset \"UTF-8\";.form-check{display:flex;align-items:center}.form-check-input{width:50px;height:24px;margin-right:10px;cursor:pointer}.form-check-input:checked{background-color:#00444c;border-color:#00444c}.form-check-input:not(:checked){border-color:#3ec9d6;box-shadow:0 0 5px #e5e5e5}.form-check-label{color:#00444c;font-family:Inter,sans-serif;font-size:16px;font-weight:600;height:24px;line-height:24px}\n"] }]
39
+ args: [{ selector: 'argenta-custom-switch', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"hasPermission()\">\n <div class=\"form-check form-switch\">\n <input \n class=\"form-check-input\" \n type=\"checkbox\" \n [checked]=\"checked\" \n (change)=\"toggleSwitch()\" \n id=\"flexSwitchCheckDefault\" \n />\n <label class=\"form-check-label\" for=\"flexSwitchCheckDefault\">\n {{ label }}\n </label>\n </div>\n</ng-container>\n", styles: ["@charset \"UTF-8\";.form-check{display:flex;align-items:center}.form-check-input{width:60px;height:34px;margin-right:10px;cursor:pointer;position:relative;appearance:none;background-color:#e5e5e5;border:2px solid #00444C;border-radius:34px;outline:none;transition:background-color .3s,border-color .3s}.form-check-input:focus{outline:none;box-shadow:none}.form-check-input:checked{background-color:#00444c;border-color:#00444c}.form-check-input:before{content:\"\";position:absolute;width:26px;height:26px;background-color:#00444c;border-radius:50%;top:3px;left:3px;transition:transform .3s,background-color .3s}.form-check-input:checked:before{transform:translate(26px);background-color:#fff}.form-check-label{color:#00444c;font-family:Inter,sans-serif;font-size:16px;font-weight:600;height:24px;line-height:24px}\n"] }]
40
40
  }], ctorParameters: function () { return [{ type: i1.AuthService }]; }, propDecorators: { checked: [{
41
41
  type: Input
42
42
  }], label: [{
@@ -1,7 +1,7 @@
1
- import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, forwardRef } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, Input, Output } from '@angular/core';
2
2
  import { NG_VALUE_ACCESSOR } from '@angular/forms';
3
3
  import { of, Subject } from 'rxjs';
4
- import { debounceTime, switchMap, catchError, startWith, map } from 'rxjs/operators';
4
+ import { catchError, debounceTime, map, startWith, switchMap } from 'rxjs/operators';
5
5
  import * as i0 from "@angular/core";
6
6
  import * as i1 from "../../service/auth-service.service";
7
7
  import * as i2 from "@angular/common/http";
@@ -15,19 +15,19 @@ export class MultiSelectComponent {
15
15
  this.label = 'Multi Select';
16
16
  this.data = []; // Accepts an array of generic objects
17
17
  this.placeholder = 'Select items';
18
- this.selected = []; // The selected variable should be an array
19
18
  this.id = 'multiSelectId';
20
19
  this.bindLabel = ''; // Generic dynamic label
21
20
  this.bindValue = ''; // Generic dynamic value
22
21
  this.closeOnSelect = false; // New property to control dropdown close behavior
23
22
  this.searchUrl = ''; // URL for backend search
23
+ this.multiple = true; // New property to control single or multiple selection
24
+ this.searchParams = {}; // Parâmetros de busca dinâmicos
24
25
  this.keyupEvent = new EventEmitter();
25
- this.originalData = []; // Store the original list
26
+ this.backupData = []; // Backup of the initial data
26
27
  this.allItems = []; // Store the combined list
27
28
  this.items = of([]); // Initialization of the property
28
29
  this.filteredItems = of([]); // Filtered items
29
30
  this.searchTerms = new Subject(); // For search debounce
30
- this.backendItems = []; // Items found from backend
31
31
  this.onChangeCallback = () => { };
32
32
  this.onTouchedCallback = () => { };
33
33
  this.isCourseEntered = false;
@@ -36,11 +36,61 @@ export class MultiSelectComponent {
36
36
  };
37
37
  }
38
38
  ngOnInit() {
39
- this.originalData = [...this.data]; // Preserve the original data
40
- this.allItems = [...this.data]; // Initialize allItems with the original data
39
+ this.backupData = [...this.data]; // Backup initial data
40
+ this.allItems = [...this.data]; // Initialize allItems with the initial data
41
41
  this.items = of(this.allItems);
42
- this.filteredItems = this.searchTerms.pipe(debounceTime(700), startWith(''), // Start with an empty search to load the original list
43
- switchMap(term => this.search(term)));
42
+ this.fetchInitialData().subscribe(data => {
43
+ this.updateData(data);
44
+ this.filteredItems = this.searchTerms.pipe(debounceTime(700), startWith(''), // Start with an empty search to load the original list
45
+ switchMap(term => this.search(term)));
46
+ // Adicionar itens selecionados à lista após buscar dados iniciais
47
+ this.addSelectedItemsToData();
48
+ });
49
+ }
50
+ ngOnChanges(changes) {
51
+ if (changes['selected'] && !changes['selected'].isFirstChange()) {
52
+ this.addSelectedItemsToData();
53
+ }
54
+ }
55
+ fetchInitialData() {
56
+ return this.http.get(this.searchUrl).pipe(map((response) => {
57
+ if (response && response.length > 0) {
58
+ return response.map((item) => ({
59
+ [this.bindValue]: item[this.bindValue],
60
+ [this.bindLabel]: item[this.bindLabel]
61
+ }));
62
+ }
63
+ return [];
64
+ }), catchError((error) => {
65
+ console.error('Error fetching initial data from backend:', error);
66
+ return of([]);
67
+ }));
68
+ }
69
+ updateData(newData) {
70
+ newData.forEach((item) => {
71
+ const existsInList = this.allItems.some(listItem => this.compareFn(listItem, item));
72
+ if (!existsInList) {
73
+ this.allItems.push(item);
74
+ }
75
+ });
76
+ this.items = of(this.allItems);
77
+ }
78
+ addSelectedItemsToData() {
79
+ if (this.selected) {
80
+ const selectedItems = this.multiple ? this.selected : [this.selected];
81
+ selectedItems.forEach((item) => {
82
+ const existsInList = this.allItems.some(listItem => this.compareFn(listItem, item));
83
+ if (!existsInList) {
84
+ const newItem = {
85
+ [this.bindValue]: item[this.bindValue] || item,
86
+ [this.bindLabel]: item[this.bindLabel] || item
87
+ };
88
+ this.data.push(newItem);
89
+ this.allItems.push(newItem);
90
+ }
91
+ });
92
+ this.items = of(this.allItems);
93
+ }
44
94
  }
45
95
  onFocus() {
46
96
  this.isCourseEntered = true;
@@ -52,25 +102,25 @@ export class MultiSelectComponent {
52
102
  this.keyupEvent.emit(event);
53
103
  }
54
104
  onSelectedChange(event) {
55
- const newlySelectedItems = event.filter((item) => !this.selected.includes(item));
56
- this.selected = event;
57
- // Check if any newly selected item is from backendItems
58
- const addedFromBackend = this.backendItems.filter((item) => newlySelectedItems.includes(item));
59
- if (addedFromBackend.length > 0) {
60
- // Transform the backend items to match the format expected by the component
61
- const transformedItems = addedFromBackend.map((item) => ({
62
- [this.bindValue]: item[this.bindValue],
63
- [this.bindLabel]: item[this.bindLabel]
64
- }));
65
- // Add backend items to the allItems list and update the observable
66
- this.originalData = [...this.originalData, ...transformedItems];
67
- this.allItems = [...this.originalData];
68
- this.items = of(this.allItems); // Update the observable with new data
69
- this.backendItems = []; // Clear backend items after processing
70
- this.searchTerms.next(''); // Reset the search term to update the list
71
- console.log('Added items from backend to the list and reset the dropdown.');
105
+ const previousSelected = this.selected;
106
+ if (this.multiple) {
107
+ this.selected = event;
108
+ }
109
+ else {
110
+ this.selected = event ? event : null;
72
111
  }
73
112
  this.onChangeCallback(this.selected);
113
+ // Verificar se um item foi removido
114
+ if (previousSelected && Array.isArray(previousSelected) && previousSelected.length > event.length) {
115
+ const removedItems = previousSelected.filter(item => !event.includes(item));
116
+ removedItems.forEach(item => {
117
+ const existsInData = this.data.some(dataItem => this.compareFn(dataItem, item));
118
+ if (!existsInData) {
119
+ this.data.push(item);
120
+ this.allItems.push(item); // Adicionar de volta aos itens disponíveis
121
+ }
122
+ });
123
+ }
74
124
  }
75
125
  onInputChange(event) {
76
126
  const input = event.target.value;
@@ -78,41 +128,52 @@ export class MultiSelectComponent {
78
128
  }
79
129
  search(term) {
80
130
  if (!term.trim()) {
81
- // If the search term is empty, show the complete list
131
+ // Se o termo de busca estiver vazio, retorna a lista completa
82
132
  return of(this.allItems);
83
133
  }
134
+ // Filtra os itens localmente
84
135
  const filtered = this.allItems.filter((item) => item[this.bindLabel].toLowerCase().includes(term.toLowerCase()));
85
136
  if (filtered.length > 0) {
86
137
  console.log('Items filtered locally.');
87
138
  return of(filtered);
88
139
  }
89
140
  else if (this.searchUrl) {
90
- return this.http.get(`${this.searchUrl}?term=${term}`).pipe(map((response) => {
141
+ // Construir os parâmetros de busca dinamicamente
142
+ const params = new URLSearchParams(this.searchParams);
143
+ params.append('term', term);
144
+ return this.http.get(`${this.searchUrl}?${params.toString()}`).pipe(map((response) => {
91
145
  if (response && response.length > 0) {
92
- // Transform the backend items to match the format expected by the component
146
+ // Transforma os itens do backend para o formato esperado pelo componente
93
147
  const transformedItems = response.map((item) => ({
94
148
  [this.bindValue]: item[this.bindValue],
95
149
  [this.bindLabel]: item[this.bindLabel]
96
150
  }));
97
- this.backendItems = transformedItems;
98
- return [...filtered, ...transformedItems];
151
+ // Atualiza os dados com os novos itens buscados
152
+ this.updateData(transformedItems);
153
+ return this.allItems;
99
154
  }
100
155
  else {
101
156
  console.log('No items found in the backend search.');
102
- return filtered;
157
+ return this.allItems;
103
158
  }
104
159
  }), catchError((error) => {
105
160
  console.error('Error fetching from backend:', error);
106
- return of(filtered);
161
+ return of(this.allItems);
107
162
  }));
108
163
  }
109
164
  else {
110
165
  console.log('No search URL provided and no items found locally.');
111
- return of(filtered);
166
+ return of(this.allItems);
112
167
  }
113
168
  }
114
169
  writeValue(value) {
115
- this.selected = value || [];
170
+ if (this.multiple) {
171
+ this.selected = value || [];
172
+ }
173
+ else {
174
+ this.selected = value || null;
175
+ }
176
+ this.addSelectedItemsToData();
116
177
  }
117
178
  registerOnChange(fn) {
118
179
  this.onChangeCallback = fn;
@@ -141,77 +202,23 @@ export class MultiSelectComponent {
141
202
  }
142
203
  }
143
204
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MultiSelectComponent, deps: [{ token: i1.AuthService }, { token: i2.HttpClient }], target: i0.ɵɵFactoryTarget.Component }); }
144
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: MultiSelectComponent, selector: "argenta-custom-multi-select", inputs: { label: "label", data: "data", placeholder: "placeholder", selected: "selected", id: "id", bindLabel: "bindLabel", bindValue: "bindValue", permissions: "permissions", closeOnSelect: "closeOnSelect", searchUrl: "searchUrl" }, outputs: { keyupEvent: "keyupEvent" }, providers: [
205
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: MultiSelectComponent, selector: "argenta-custom-multi-select", inputs: { label: "label", data: "data", placeholder: "placeholder", selected: "selected", id: "id", bindLabel: "bindLabel", bindValue: "bindValue", permissions: "permissions", closeOnSelect: "closeOnSelect", searchUrl: "searchUrl", multiple: "multiple", searchParams: "searchParams" }, outputs: { keyupEvent: "keyupEvent" }, providers: [
145
206
  {
146
207
  provide: NG_VALUE_ACCESSOR,
147
208
  useExisting: forwardRef(() => MultiSelectComponent),
148
209
  multi: true
149
210
  }
150
- ], ngImport: i0, template: `
151
- <div *ngIf="hasPermission()" class="form-group">
152
- <label [for]="id" class="form-label">{{ label }}</label>
153
- <ng-select
154
- [class.course-entry]="isCourseEntered"
155
- class="ng-select"
156
- [items]="filteredItems | async"
157
- [multiple]="true"
158
- [closeOnSelect]="closeOnSelect"
159
- [hideSelected]="true"
160
- [bindLabel]="bindLabel"
161
- [bindValue]="bindValue"
162
- [(ngModel)]="selected"
163
- [compareWith]="compareFn"
164
- (change)="onSelectedChange($event)"
165
- (keyup)="onKeyUp($event)"
166
- (input)="onInputChange($event)"
167
- [id]="id"
168
- [placeholder]="selected && selected.length === 0 ? placeholder : ''"
169
- (focus)="onFocus()"
170
- (blur)="onBlur()"
171
- >
172
- <ng-template ng-option-tmp let-item="item">
173
- {{ item[bindLabel] }}
174
- </ng-template>
175
- </ng-select>
176
- </div>
177
- `, isInline: true, styles: ["@charset \"UTF-8\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@400;700&display=swap\";body{font-family:Inter,sans-serif}.form-group{font-family:Inter,Arial,sans-serif;font-size:1rem}.form-check-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:.9rem}.form-check-label{width:623px;height:19px;top:1608px;left:133px;gap:0px;opacity:0px;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left}.custom-select{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem 2rem .5rem .5rem;appearance:none;-webkit-appearance:none;-moz-appearance:none;background-image:none;background-repeat:no-repeat;background-position:right .5rem center}.custom-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem}.form-label{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem}.label-styles{font-weight:400;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left;margin-top:1rem;margin-bottom:.5rem}.select-container{position:relative;display:inline-block;width:100%}.select-container lucide-icon{position:absolute;right:.75rem;top:50%;transform:translateY(-50%);pointer-events:none;color:#5e6366}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i5.NgSelectComponent, selector: "ng-select", inputs: ["bindLabel", "bindValue", "markFirst", "placeholder", "notFoundText", "typeToSearchText", "addTagText", "loadingText", "clearAllText", "appearance", "dropdownPosition", "appendTo", "loading", "closeOnSelect", "hideSelected", "selectOnTab", "openOnEnter", "maxSelectedItems", "groupBy", "groupValue", "bufferAmount", "virtualScroll", "selectableGroup", "selectableGroupAsModel", "searchFn", "trackByFn", "clearOnBackspace", "labelForId", "inputAttrs", "tabIndex", "readonly", "searchWhileComposing", "minTermLength", "editableSearchTerm", "keyDownFn", "typeahead", "multiple", "addTag", "searchable", "clearable", "isOpen", "items", "compareWith", "clearSearchOnAdd", "deselectOnClick"], outputs: ["blur", "focus", "change", "open", "close", "search", "clear", "add", "remove", "scroll", "scrollToEnd"] }, { kind: "directive", type: i5.NgOptionTemplateDirective, selector: "[ng-option-tmp]" }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
211
+ ], usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"hasPermission()\" class=\"form-group\">\n <label [for]=\"id\" class=\"form-label\" style=\"margin-top: 1rem;\">{{ label }}</label>\n <ng-select\n [class.course-entry]=\"isCourseEntered\"\n class=\"ng-select custom-ng-select\"\n [items]=\"filteredItems | async\"\n [multiple]=\"multiple\"\n [closeOnSelect]=\"closeOnSelect\"\n [hideSelected]=\"true\"\n [bindLabel]=\"bindLabel\"\n [bindValue]=\"bindValue\"\n [(ngModel)]=\"selected\"\n [compareWith]=\"compareFn\"\n (change)=\"onSelectedChange($event)\"\n (keyup)=\"onKeyUp($event)\"\n (input)=\"onInputChange($event)\"\n [id]=\"id\"\n [placeholder]=\"selected && (multiple ? selected.length === 0 : !selected) ? placeholder : ''\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\">\n <ng-template ng-option-tmp let-item=\"item\">\n {{ item[bindLabel] }}\n </ng-template>\n </ng-select>\n</div>\n", styles: ["@charset \"UTF-8\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@400;700&display=swap\";body{font-family:Inter,sans-serif}.form-group{font-family:Inter,Arial,sans-serif;font-size:1rem}.form-check-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:.9rem}.form-check-label{width:623px;height:19px;top:1608px;left:133px;gap:0px;opacity:0px;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left}.custom-select{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem 2rem .5rem .5rem;appearance:none;-webkit-appearance:none;-moz-appearance:none;background-image:none;background-repeat:no-repeat;background-position:right .5rem center}.custom-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem}.form-label{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem}.label-styles{font-weight:400;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left;margin-top:1rem;margin-bottom:.5rem}.select-container{position:relative;display:inline-block;width:100%}.select-container lucide-icon{position:absolute;right:.75rem;top:50%;transform:translateY(-50%);pointer-events:none;color:#5e6366}.ng-select{display:block;width:100%;font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem}.ng-select .ng-select-container{display:flex;align-items:center;border:1px solid #ccc;border-radius:4px;padding:.5rem;background-color:#fff;color:#333}.ng-select .ng-select-container .ng-value-container{display:flex;align-items:center;flex-grow:1}.ng-select .ng-select-container .ng-clear{display:none}.ng-select .ng-select-container .ng-input{flex-grow:1;border:none;outline:none}.custom-ng-select.ng-select .ng-select-container .ng-input>input{box-sizing:border-box;background:none transparent;border:0 none;box-shadow:none;outline:none;padding:.5rem!important;cursor:default;width:100%}.ng-select .ng-dropdown-panel{border:1px solid #ccc;border-radius:4px;background-color:#fff;color:#333;box-shadow:0 2px 4px #0000001a}.ng-select .ng-dropdown-panel .ng-option{padding:8px;cursor:pointer}.ng-select .ng-dropdown-panel .ng-option:hover{background-color:#f1f1f1}.ng-select .ng-dropdown-panel .ng-option.selected{background-color:#007bff;color:#fff}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i5.NgSelectComponent, selector: "ng-select", inputs: ["bindLabel", "bindValue", "markFirst", "placeholder", "notFoundText", "typeToSearchText", "addTagText", "loadingText", "clearAllText", "appearance", "dropdownPosition", "appendTo", "loading", "closeOnSelect", "hideSelected", "selectOnTab", "openOnEnter", "maxSelectedItems", "groupBy", "groupValue", "bufferAmount", "virtualScroll", "selectableGroup", "selectableGroupAsModel", "searchFn", "trackByFn", "clearOnBackspace", "labelForId", "inputAttrs", "tabIndex", "readonly", "searchWhileComposing", "minTermLength", "editableSearchTerm", "keyDownFn", "typeahead", "multiple", "addTag", "searchable", "clearable", "isOpen", "items", "compareWith", "clearSearchOnAdd", "deselectOnClick"], outputs: ["blur", "focus", "change", "open", "close", "search", "clear", "add", "remove", "scroll", "scrollToEnd"] }, { kind: "directive", type: i5.NgOptionTemplateDirective, selector: "[ng-option-tmp]" }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
178
212
  }
179
213
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MultiSelectComponent, decorators: [{
180
214
  type: Component,
181
- args: [{ selector: 'argenta-custom-multi-select', template: `
182
- <div *ngIf="hasPermission()" class="form-group">
183
- <label [for]="id" class="form-label">{{ label }}</label>
184
- <ng-select
185
- [class.course-entry]="isCourseEntered"
186
- class="ng-select"
187
- [items]="filteredItems | async"
188
- [multiple]="true"
189
- [closeOnSelect]="closeOnSelect"
190
- [hideSelected]="true"
191
- [bindLabel]="bindLabel"
192
- [bindValue]="bindValue"
193
- [(ngModel)]="selected"
194
- [compareWith]="compareFn"
195
- (change)="onSelectedChange($event)"
196
- (keyup)="onKeyUp($event)"
197
- (input)="onInputChange($event)"
198
- [id]="id"
199
- [placeholder]="selected && selected.length === 0 ? placeholder : ''"
200
- (focus)="onFocus()"
201
- (blur)="onBlur()"
202
- >
203
- <ng-template ng-option-tmp let-item="item">
204
- {{ item[bindLabel] }}
205
- </ng-template>
206
- </ng-select>
207
- </div>
208
- `, providers: [
215
+ args: [{ selector: 'argenta-custom-multi-select', providers: [
209
216
  {
210
217
  provide: NG_VALUE_ACCESSOR,
211
218
  useExisting: forwardRef(() => MultiSelectComponent),
212
219
  multi: true
213
220
  }
214
- ], changeDetection: ChangeDetectionStrategy.OnPush, styles: ["@charset \"UTF-8\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@400;700&display=swap\";body{font-family:Inter,sans-serif}.form-group{font-family:Inter,Arial,sans-serif;font-size:1rem}.form-check-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:.9rem}.form-check-label{width:623px;height:19px;top:1608px;left:133px;gap:0px;opacity:0px;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left}.custom-select{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem 2rem .5rem .5rem;appearance:none;-webkit-appearance:none;-moz-appearance:none;background-image:none;background-repeat:no-repeat;background-position:right .5rem center}.custom-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem}.form-label{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem}.label-styles{font-weight:400;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left;margin-top:1rem;margin-bottom:.5rem}.select-container{position:relative;display:inline-block;width:100%}.select-container lucide-icon{position:absolute;right:.75rem;top:50%;transform:translateY(-50%);pointer-events:none;color:#5e6366}\n"] }]
221
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div *ngIf=\"hasPermission()\" class=\"form-group\">\n <label [for]=\"id\" class=\"form-label\" style=\"margin-top: 1rem;\">{{ label }}</label>\n <ng-select\n [class.course-entry]=\"isCourseEntered\"\n class=\"ng-select custom-ng-select\"\n [items]=\"filteredItems | async\"\n [multiple]=\"multiple\"\n [closeOnSelect]=\"closeOnSelect\"\n [hideSelected]=\"true\"\n [bindLabel]=\"bindLabel\"\n [bindValue]=\"bindValue\"\n [(ngModel)]=\"selected\"\n [compareWith]=\"compareFn\"\n (change)=\"onSelectedChange($event)\"\n (keyup)=\"onKeyUp($event)\"\n (input)=\"onInputChange($event)\"\n [id]=\"id\"\n [placeholder]=\"selected && (multiple ? selected.length === 0 : !selected) ? placeholder : ''\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\">\n <ng-template ng-option-tmp let-item=\"item\">\n {{ item[bindLabel] }}\n </ng-template>\n </ng-select>\n</div>\n", styles: ["@charset \"UTF-8\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@400;700&display=swap\";body{font-family:Inter,sans-serif}.form-group{font-family:Inter,Arial,sans-serif;font-size:1rem}.form-check-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:.9rem}.form-check-label{width:623px;height:19px;top:1608px;left:133px;gap:0px;opacity:0px;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left}.custom-select{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem 2rem .5rem .5rem;appearance:none;-webkit-appearance:none;-moz-appearance:none;background-image:none;background-repeat:no-repeat;background-position:right .5rem center}.custom-input{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem;font-weight:400;border:1px solid #ccc;border-radius:4px;padding:.5rem}.form-label{font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem}.label-styles{font-weight:400;font-family:Inter,Arial,sans-serif;font-size:16px;line-height:19.36px;text-align:left;margin-top:1rem;margin-bottom:.5rem}.select-container{position:relative;display:inline-block;width:100%}.select-container lucide-icon{position:absolute;right:.75rem;top:50%;transform:translateY(-50%);pointer-events:none;color:#5e6366}.ng-select{display:block;width:100%;font-family:Inter,Arial,sans-serif;color:#333;font-size:1rem}.ng-select .ng-select-container{display:flex;align-items:center;border:1px solid #ccc;border-radius:4px;padding:.5rem;background-color:#fff;color:#333}.ng-select .ng-select-container .ng-value-container{display:flex;align-items:center;flex-grow:1}.ng-select .ng-select-container .ng-clear{display:none}.ng-select .ng-select-container .ng-input{flex-grow:1;border:none;outline:none}.custom-ng-select.ng-select .ng-select-container .ng-input>input{box-sizing:border-box;background:none transparent;border:0 none;box-shadow:none;outline:none;padding:.5rem!important;cursor:default;width:100%}.ng-select .ng-dropdown-panel{border:1px solid #ccc;border-radius:4px;background-color:#fff;color:#333;box-shadow:0 2px 4px #0000001a}.ng-select .ng-dropdown-panel .ng-option{padding:8px;cursor:pointer}.ng-select .ng-dropdown-panel .ng-option:hover{background-color:#f1f1f1}.ng-select .ng-dropdown-panel .ng-option.selected{background-color:#007bff;color:#fff}\n"] }]
215
222
  }], ctorParameters: function () { return [{ type: i1.AuthService }, { type: i2.HttpClient }]; }, propDecorators: { label: [{
216
223
  type: Input
217
224
  }], data: [{
@@ -232,7 +239,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
232
239
  type: Input
233
240
  }], searchUrl: [{
234
241
  type: Input
242
+ }], multiple: [{
243
+ type: Input
244
+ }], searchParams: [{
245
+ type: Input
235
246
  }], keyupEvent: [{
236
247
  type: Output
237
248
  }] } });
238
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select.component.js","sourceRoot":"","sources":["../../../../../../projects/lib-portal-angular/src/lib/components/multi-select/multi-select.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AACpH,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAc,EAAE,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/C,OAAO,EAAE,YAAY,EAAE,SAAS,EAAE,UAAU,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;;;;;;;AA4CrF,MAAM,OAAO,oBAAoB;IAqB/B,YAAoB,WAAwB,EAAU,IAAgB;QAAlD,gBAAW,GAAX,WAAW,CAAa;QAAU,SAAI,GAAJ,IAAI,CAAY;QApB7D,UAAK,GAAW,cAAc,CAAC;QAC/B,SAAI,GAAU,EAAE,CAAC,CAAC,sCAAsC;QACxD,gBAAW,GAAW,cAAc,CAAC;QACrC,aAAQ,GAAU,EAAE,CAAC,CAAC,2CAA2C;QACjE,OAAE,GAAW,eAAe,CAAC;QAC7B,cAAS,GAAW,EAAE,CAAC,CAAC,wBAAwB;QAChD,cAAS,GAAW,EAAE,CAAC,CAAC,wBAAwB;QAEhD,kBAAa,GAAY,KAAK,CAAC,CAAC,kDAAkD;QAClF,cAAS,GAAW,EAAE,CAAC,CAAC,yBAAyB;QAEhD,eAAU,GAAsB,IAAI,YAAY,EAAO,CAAC;QAE1D,iBAAY,GAAU,EAAE,CAAC,CAAC,0BAA0B;QACpD,aAAQ,GAAU,EAAE,CAAC,CAAC,0BAA0B;QACxD,UAAK,GAAsB,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,iCAAiC;QACpE,kBAAa,GAAsB,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,iBAAiB;QACpD,gBAAW,GAAG,IAAI,OAAO,EAAU,CAAC,CAAC,sBAAsB;QAC3D,iBAAY,GAAU,EAAE,CAAC,CAAC,2BAA2B;QAerD,qBAAgB,GAAqB,GAAG,EAAE,GAAG,CAAC,CAAC;QAC/C,sBAAiB,GAAe,GAAG,EAAE,GAAG,CAAC,CAAC;QAElD,oBAAe,GAAG,KAAK,CAAC;QAqGxB,cAAS,GAAG,CAAC,KAAU,EAAE,KAAU,EAAW,EAAE;YAC9C,OAAO,KAAK,IAAI,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,KAAK,KAAK,KAAK,CAAC;QAC5F,CAAC,CAAC;IAvHwE,CAAC;IAE3E,QAAQ;QACN,IAAI,CAAC,YAAY,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,6BAA6B;QACjE,IAAI,CAAC,QAAQ,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,6CAA6C;QAC7E,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC/B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CACxC,YAAY,CAAC,GAAG,CAAC,EACjB,SAAS,CAAC,EAAE,CAAC,EAAE,uDAAuD;QACtE,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CACrC,CAAC;IACJ,CAAC;IAOD,OAAO;QACL,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;IAC9B,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;IAC/B,CAAC;IAED,OAAO,CAAC,KAAoB;QAC1B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC9B,CAAC;IAED,gBAAgB,CAAC,KAAU;QACzB,MAAM,kBAAkB,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,IAAS,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;QACtF,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QAEtB,wDAAwD;QACxD,MAAM,gBAAgB,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,IAAS,EAAE,EAAE,CAAC,kBAAkB,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;QAEpG,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE;YAC/B,4EAA4E;YAC5E,MAAM,gBAAgB,GAAG,gBAAgB,CAAC,GAAG,CAAC,CAAC,IAAS,EAAE,EAAE,CAAC,CAAC;gBAC5D,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC;gBACtC,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC;aACvC,CAAC,CAAC,CAAC;YAEJ,mEAAmE;YACnE,IAAI,CAAC,YAAY,GAAG,CAAC,GAAG,IAAI,CAAC,YAAY,EAAE,GAAG,gBAAgB,CAAC,CAAC;YAChE,IAAI,CAAC,QAAQ,GAAG,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC;YACvC,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,sCAAsC;YACtE,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC,CAAC,uCAAuC;YAC/D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,2CAA2C;YACtE,OAAO,CAAC,GAAG,CAAC,8DAA8D,CAAC,CAAC;SAC7E;QAED,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACvC,CAAC;IAED,aAAa,CAAC,KAAU;QACtB,MAAM,KAAK,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC;QACjC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,MAAM,CAAC,IAAY;QACjB,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE;YAChB,sDAAsD;YACtD,OAAO,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAC1B;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,IAAS,EAAE,EAAE,CAClD,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAChE,CAAC;QAEF,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;YACvB,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;YACvC,OAAO,EAAE,CAAC,QAAQ,CAAC,CAAC;SACrB;aAAM,IAAI,IAAI,CAAC,SAAS,EAAE;YACzB,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAM,GAAG,IAAI,CAAC,SAAS,SAAS,IAAI,EAAE,CAAC,CAAC,IAAI,CAC9D,GAAG,CAAC,CAAC,QAAa,EAAE,EAAE;gBACpB,IAAI,QAAQ,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;oBACnC,4EAA4E;oBAC5E,MAAM,gBAAgB,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,IAAS,EAAE,EAAE,CAAC,CAAC;wBACpD,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC;wBACtC,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC;qBACvC,CAAC,CAAC,CAAC;oBACJ,IAAI,CAAC,YAAY,GAAG,gBAAgB,CAAC;oBACrC,OAAO,CAAC,GAAG,QAAQ,EAAE,GAAG,gBAAgB,CAAC,CAAC;iBAC3C;qBAAM;oBACL,OAAO,CAAC,GAAG,CAAC,uCAAuC,CAAC,CAAC;oBACrD,OAAO,QAAQ,CAAC;iBACjB;YACH,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,KAAU,EAAE,EAAE;gBACxB,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;gBACrD,OAAO,EAAE,CAAC,QAAQ,CAAC,CAAC;YACtB,CAAC,CAAC,CACH,CAAC;SACH;aAAM;YACL,OAAO,CAAC,GAAG,CAAC,oDAAoD,CAAC,CAAC;YAClE,OAAO,EAAE,CAAC,QAAQ,CAAC,CAAC;SACrB;IACH,CAAC;IAED,UAAU,CAAC,KAAU;QACnB,IAAI,CAAC,QAAQ,GAAG,KAAK,IAAI,EAAE,CAAC;IAC9B,CAAC;IAED,gBAAgB,CAAC,EAAO;QACtB,IAAI,CAAC,gBAAgB,GAAG,EAAE,CAAC;IAC7B,CAAC;IAED,iBAAiB,CAAC,EAAO;QACvB,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;IAC9B,CAAC;IAED,gBAAgB,CAAE,UAAmB;QACnC,4CAA4C;IAC9C,CAAC;IAMD,aAAa;QACX,IAAI,CAAC,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE;YACtD,OAAO,IAAI,CAAC;SACb;QAED,IAAI;YACF,OAAO,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;SACzD;QAAC,OAAO,KAAc,EAAE;YACvB,IAAI,KAAK,YAAY,KAAK,EAAE;gBAC1B,OAAO,CAAC,KAAK,CAAC,mBAAmB,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;aACnD;iBAAM;gBACL,OAAO,CAAC,KAAK,CAAC,gDAAgD,CAAC,CAAC;aACjE;YACD,OAAO,IAAI,CAAC;SACb;IACH,CAAC;+GA7JU,oBAAoB;mGAApB,oBAAoB,uUATpB;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,oBAAoB,CAAC;gBACnD,KAAK,EAAE,IAAI;aACZ;SACF,0BAlCS;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BT;;4FAUU,oBAAoB;kBAxChC,SAAS;+BACE,6BAA6B,YAE7B;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BT,aACU;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,qBAAqB,CAAC;4BACnD,KAAK,EAAE,IAAI;yBACZ;qBACF,mBACgB,uBAAuB,CAAC,MAAM;2HAGtC,KAAK;sBAAb,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,EAAE;sBAAV,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBAEI,UAAU;sBAAnB,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, OnInit, Output, forwardRef } from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { Observable, of, Subject } from 'rxjs';\nimport { debounceTime, switchMap, catchError, startWith, map } from 'rxjs/operators';\nimport { AuthService } from '../../service/auth-service.service';\nimport { HttpClient } from '@angular/common/http';\n\n@Component({\n  selector: 'argenta-custom-multi-select',\n  styleUrls: ['./multi-select.component.scss'],\n  template: `\n    <div *ngIf=\"hasPermission()\" class=\"form-group\">\n      <label [for]=\"id\" class=\"form-label\">{{ label }}</label>\n      <ng-select\n        [class.course-entry]=\"isCourseEntered\"\n        class=\"ng-select\"\n        [items]=\"filteredItems | async\"\n        [multiple]=\"true\"\n        [closeOnSelect]=\"closeOnSelect\"\n        [hideSelected]=\"true\"\n        [bindLabel]=\"bindLabel\"\n        [bindValue]=\"bindValue\"\n        [(ngModel)]=\"selected\"\n        [compareWith]=\"compareFn\"\n        (change)=\"onSelectedChange($event)\"\n        (keyup)=\"onKeyUp($event)\"\n        (input)=\"onInputChange($event)\"\n        [id]=\"id\"\n        [placeholder]=\"selected && selected.length === 0 ? placeholder : ''\"\n        (focus)=\"onFocus()\"\n        (blur)=\"onBlur()\"\n      >\n        <ng-template ng-option-tmp let-item=\"item\">\n          {{ item[bindLabel] }}\n        </ng-template>\n      </ng-select>\n    </div>\n  `,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectComponent),\n      multi: true\n    }\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class MultiSelectComponent implements ControlValueAccessor, OnInit {\n  @Input() label: string = 'Multi Select';\n  @Input() data: any[] = []; // Accepts an array of generic objects\n  @Input() placeholder: string = 'Select items';\n  @Input() selected: any[] = []; // The selected variable should be an array\n  @Input() id: string = 'multiSelectId';\n  @Input() bindLabel: string = ''; // Generic dynamic label\n  @Input() bindValue: string = ''; // Generic dynamic value\n  @Input() permissions: string[] | undefined; // Adding permissions as input\n  @Input() closeOnSelect: boolean = false; // New property to control dropdown close behavior\n  @Input() searchUrl: string = ''; // URL for backend search\n\n  @Output() keyupEvent: EventEmitter<any> = new EventEmitter<any>();\n\n  private originalData: any[] = []; // Store the original list\n  private allItems: any[] = []; // Store the combined list\n  items: Observable<any[]> = of([]); // Initialization of the property\n  filteredItems: Observable<any[]> = of([]); // Filtered items\n  private searchTerms = new Subject<string>(); // For search debounce\n  private backendItems: any[] = []; // Items found from backend\n\n  constructor(private authService: AuthService, private http: HttpClient) { }\n\n  ngOnInit(): void {\n    this.originalData = [...this.data]; // Preserve the original data\n    this.allItems = [...this.data]; // Initialize allItems with the original data\n    this.items = of(this.allItems);\n    this.filteredItems = this.searchTerms.pipe(\n      debounceTime(700),\n      startWith(''), // Start with an empty search to load the original list\n      switchMap(term => this.search(term))\n    );\n  }\n\n  private onChangeCallback: (_: any) => void = () => { };\n  private onTouchedCallback: () => void = () => { };\n\n  isCourseEntered = false;\n\n  onFocus() {\n    this.isCourseEntered = true;\n  }\n\n  onBlur() {\n    this.isCourseEntered = false;\n  }\n\n  onKeyUp(event: KeyboardEvent): void {\n    this.keyupEvent.emit(event);\n  }\n\n  onSelectedChange(event: any): void {\n    const newlySelectedItems = event.filter((item: any) => !this.selected.includes(item));\n    this.selected = event;\n\n    // Check if any newly selected item is from backendItems\n    const addedFromBackend = this.backendItems.filter((item: any) => newlySelectedItems.includes(item));\n\n    if (addedFromBackend.length > 0) {\n      // Transform the backend items to match the format expected by the component\n      const transformedItems = addedFromBackend.map((item: any) => ({\n        [this.bindValue]: item[this.bindValue],\n        [this.bindLabel]: item[this.bindLabel]\n      }));\n\n      // Add backend items to the allItems list and update the observable\n      this.originalData = [...this.originalData, ...transformedItems];\n      this.allItems = [...this.originalData];\n      this.items = of(this.allItems); // Update the observable with new data\n      this.backendItems = []; // Clear backend items after processing\n      this.searchTerms.next(''); // Reset the search term to update the list\n      console.log('Added items from backend to the list and reset the dropdown.');\n    }\n\n    this.onChangeCallback(this.selected);\n  }\n\n  onInputChange(event: any): void {\n    const input = event.target.value;\n    this.searchTerms.next(input);\n  }\n\n  search(term: string): Observable<any[]> {\n    if (!term.trim()) {\n      // If the search term is empty, show the complete list\n      return of(this.allItems);\n    }\n\n    const filtered = this.allItems.filter((item: any) =>\n      item[this.bindLabel].toLowerCase().includes(term.toLowerCase())\n    );\n\n    if (filtered.length > 0) {\n      console.log('Items filtered locally.');\n      return of(filtered);\n    } else if (this.searchUrl) {\n      return this.http.get<any>(`${this.searchUrl}?term=${term}`).pipe(\n        map((response: any) => {\n          if (response && response.length > 0) {\n            // Transform the backend items to match the format expected by the component\n            const transformedItems = response.map((item: any) => ({\n              [this.bindValue]: item[this.bindValue],\n              [this.bindLabel]: item[this.bindLabel]\n            }));\n            this.backendItems = transformedItems;\n            return [...filtered, ...transformedItems];\n          } else {\n            console.log('No items found in the backend search.');\n            return filtered;\n          }\n        }),\n        catchError((error: any) => {\n          console.error('Error fetching from backend:', error);\n          return of(filtered);\n        })\n      );\n    } else {\n      console.log('No search URL provided and no items found locally.');\n      return of(filtered);\n    }\n  }\n\n  writeValue(value: any): void {\n    this.selected = value || [];\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChangeCallback = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouchedCallback = fn;\n  }\n\n  setDisabledState?(isDisabled: boolean): void {\n    // No implementation needed for this example\n  }\n\n  compareFn = (item1: any, item2: any): boolean => {\n    return item1 && item2 ? item1[this.bindValue] === item2[this.bindValue] : item1 === item2;\n  };\n\n  hasPermission(): boolean {\n    if (!this.permissions || this.permissions.length === 0) {\n      return true;\n    }\n\n    try {\n      return this.authService.hasPermission(this.permissions);\n    } catch (error: unknown) {\n      if (error instanceof Error) {\n        console.error('Permission error:', error.message);\n      } else {\n        console.error('Unknown error occurred during permission check');\n      }\n      return true;\n    }\n  }\n}\n"]}
249
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select.component.js","sourceRoot":"","sources":["../../../../../../projects/lib-portal-angular/src/lib/components/multi-select/multi-select.component.ts","../../../../../../projects/lib-portal-angular/src/lib/components/multi-select/multi-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, Input, OnChanges, OnInit, Output, SimpleChanges } from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { Observable, of, Subject } from 'rxjs';\nimport { catchError, debounceTime, map, startWith, switchMap } from 'rxjs/operators';\nimport { AuthService } from '../../service/auth-service.service';\n\n@Component({\n  selector: 'argenta-custom-multi-select',\n  templateUrl: './multi-select.component.html',\n  styleUrls: ['./multi-select.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectComponent),\n      multi: true\n    }\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class MultiSelectComponent implements ControlValueAccessor, OnInit, OnChanges {\n  @Input() label: string = 'Multi Select';\n  @Input() data: any[] = []; // Accepts an array of generic objects\n  @Input() placeholder: string = 'Select items';\n  @Input() selected: any; // The selected variable can be a single value or an array\n  @Input() id: string = 'multiSelectId';\n  @Input() bindLabel: string = ''; // Generic dynamic label\n  @Input() bindValue: string = ''; // Generic dynamic value\n  @Input() permissions: string[] | undefined; // Adding permissions as input\n  @Input() closeOnSelect: boolean = false; // New property to control dropdown close behavior\n  @Input() searchUrl: string = ''; // URL for backend search\n  @Input() multiple: boolean = true; // New property to control single or multiple selection\n  @Input() searchParams: { [key: string]: string } = {}; // Parâmetros de busca dinâmicos\n\n  @Output() keyupEvent: EventEmitter<any> = new EventEmitter<any>();\n\n  private backupData: any[] = []; // Backup of the initial data\n  private allItems: any[] = []; // Store the combined list\n  items: Observable<any[]> = of([]); // Initialization of the property\n  filteredItems: Observable<any[]> = of([]); // Filtered items\n  private searchTerms = new Subject<string>(); // For search debounce\n\n  constructor(private authService: AuthService, private http: HttpClient) { }\n\n  ngOnInit(): void {\n    this.backupData = [...this.data]; // Backup initial data\n    this.allItems = [...this.data]; // Initialize allItems with the initial data\n    this.items = of(this.allItems);\n\n    this.fetchInitialData().subscribe(data => {\n      this.updateData(data);\n      this.filteredItems = this.searchTerms.pipe(\n        debounceTime(700),\n        startWith(''), // Start with an empty search to load the original list\n        switchMap(term => this.search(term))\n      );\n\n      // Adicionar itens selecionados à lista após buscar dados iniciais\n      this.addSelectedItemsToData();\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['selected'] && !changes['selected'].isFirstChange()) {\n      this.addSelectedItemsToData();\n    }\n  }\n\n  private fetchInitialData(): Observable<any[]> {\n    return this.http.get<any>(this.searchUrl).pipe(\n      map((response: any) => {\n        if (response && response.length > 0) {\n          return response.map((item: any) => ({\n            [this.bindValue]: item[this.bindValue],\n            [this.bindLabel]: item[this.bindLabel]\n          }));\n        }\n        return [];\n      }),\n      catchError((error: any) => {\n        console.error('Error fetching initial data from backend:', error);\n        return of([]);\n      })\n    );\n  }\n\n  private updateData(newData: any[]): void {\n    newData.forEach((item: any) => {\n      const existsInList = this.allItems.some(listItem => this.compareFn(listItem, item));\n      if (!existsInList) {\n        this.allItems.push(item);\n      }\n    });\n    this.items = of(this.allItems);\n  }\n\n  private addSelectedItemsToData(): void {\n    if (this.selected) {\n      const selectedItems = this.multiple ? this.selected : [this.selected];\n      selectedItems.forEach((item: any) => {\n        const existsInList = this.allItems.some(listItem => this.compareFn(listItem, item));\n        if (!existsInList) {\n          const newItem = {\n            [this.bindValue]: item[this.bindValue] || item,\n            [this.bindLabel]: item[this.bindLabel] || item\n          };\n          this.data.push(newItem);\n          this.allItems.push(newItem);\n        }\n      });\n\n      this.items = of(this.allItems);\n    }\n  }\n\n  private onChangeCallback: (_: any) => void = () => { };\n  private onTouchedCallback: () => void = () => { };\n\n  isCourseEntered = false;\n\n  onFocus() {\n    this.isCourseEntered = true;\n  }\n\n  onBlur() {\n    this.isCourseEntered = false;\n  }\n\n  onKeyUp(event: KeyboardEvent): void {\n    this.keyupEvent.emit(event);\n  }\n\n  onSelectedChange(event: any): void {\n    const previousSelected = this.selected;\n\n    if (this.multiple) {\n      this.selected = event;\n    } else {\n      this.selected = event ? event : null;\n    }\n\n    this.onChangeCallback(this.selected);\n\n    // Verificar se um item foi removido\n    if (previousSelected && Array.isArray(previousSelected) && previousSelected.length > event.length) {\n      const removedItems = previousSelected.filter(item => !event.includes(item));\n      removedItems.forEach(item => {\n        const existsInData = this.data.some(dataItem => this.compareFn(dataItem, item));\n        if (!existsInData) {\n          this.data.push(item);\n          this.allItems.push(item); // Adicionar de volta aos itens disponíveis\n        }\n      });\n    }\n  }\n\n  onInputChange(event: any): void {\n    const input = event.target.value;\n    this.searchTerms.next(input);\n  }\n\n  search(term: string): Observable<any[]> {\n    if (!term.trim()) {\n      // Se o termo de busca estiver vazio, retorna a lista completa\n      return of(this.allItems);\n    }\n\n    // Filtra os itens localmente\n    const filtered = this.allItems.filter((item: any) =>\n      item[this.bindLabel].toLowerCase().includes(term.toLowerCase())\n    );\n\n    if (filtered.length > 0) {\n      console.log('Items filtered locally.');\n      return of(filtered);\n    } else if (this.searchUrl) {\n      // Construir os parâmetros de busca dinamicamente\n      const params = new URLSearchParams(this.searchParams);\n      params.append('term', term);\n\n      return this.http.get<any[]>(`${this.searchUrl}?${params.toString()}`).pipe(\n        map((response: any[]) => {\n          if (response && response.length > 0) {\n            // Transforma os itens do backend para o formato esperado pelo componente\n            const transformedItems = response.map((item: any) => ({\n              [this.bindValue]: item[this.bindValue],\n              [this.bindLabel]: item[this.bindLabel]\n            }));\n\n            // Atualiza os dados com os novos itens buscados\n            this.updateData(transformedItems);\n\n            return this.allItems;\n          } else {\n            console.log('No items found in the backend search.');\n            return this.allItems;\n          }\n        }),\n        catchError((error: any) => {\n          console.error('Error fetching from backend:', error);\n          return of(this.allItems);\n        })\n      );\n    } else {\n      console.log('No search URL provided and no items found locally.');\n      return of(this.allItems);\n    }\n  }\n\n  writeValue(value: any): void {\n    if (this.multiple) {\n      this.selected = value || [];\n    } else {\n      this.selected = value || null;\n    }\n\n    this.addSelectedItemsToData();\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChangeCallback = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouchedCallback = fn;\n  }\n\n  setDisabledState?(isDisabled: boolean): void {\n    // No implementation needed for this example\n  }\n\n  compareFn = (item1: any, item2: any): boolean => {\n    return item1 && item2 ? item1[this.bindValue] === item2[this.bindValue] : item1 === item2;\n  };\n\n  hasPermission(): boolean {\n    if (!this.permissions || this.permissions.length === 0) {\n      return true;\n    }\n\n    try {\n      return this.authService.hasPermission(this.permissions);\n    } catch (error: unknown) {\n      if (error instanceof Error) {\n        console.error('Permission error:', error.message);\n      } else {\n        console.error('Unknown error occurred during permission check');\n      }\n      return true;\n    }\n  }\n}\n","<div *ngIf=\"hasPermission()\" class=\"form-group\">\n  <label [for]=\"id\" class=\"form-label\" style=\"margin-top: 1rem;\">{{ label }}</label>\n  <ng-select\n    [class.course-entry]=\"isCourseEntered\"\n    class=\"ng-select custom-ng-select\"\n    [items]=\"filteredItems | async\"\n    [multiple]=\"multiple\"\n    [closeOnSelect]=\"closeOnSelect\"\n    [hideSelected]=\"true\"\n    [bindLabel]=\"bindLabel\"\n    [bindValue]=\"bindValue\"\n    [(ngModel)]=\"selected\"\n    [compareWith]=\"compareFn\"\n    (change)=\"onSelectedChange($event)\"\n    (keyup)=\"onKeyUp($event)\"\n    (input)=\"onInputChange($event)\"\n    [id]=\"id\"\n    [placeholder]=\"selected && (multiple ? selected.length === 0 : !selected) ? placeholder : ''\"\n    (focus)=\"onFocus()\"\n    (blur)=\"onBlur()\">\n    <ng-template ng-option-tmp let-item=\"item\">\n      {{ item[bindLabel] }}\n    </ng-template>\n  </ng-select>\n</div>\n"]}