lib-portal-angular 0.0.87 → 0.0.89

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.
Files changed (35) hide show
  1. package/esm2022/lib/components/components.module.mjs +65 -40
  2. package/esm2022/lib/components/csv-importer/csv-importer.component.mjs +92 -0
  3. package/esm2022/lib/components/data-excel-exporter/IExcelExportConfig.mjs +2 -0
  4. package/esm2022/lib/components/data-excel-exporter/data-excel-exporter.component.mjs +126 -0
  5. package/esm2022/lib/components/data-excel-import/data-excel-importer.component.mjs +198 -0
  6. package/esm2022/lib/components/multi-select/multi-select.component.mjs +165 -115
  7. package/esm2022/lib/components/pdf-data-handler/argenta-pdf-data-handler.component.mjs +147 -0
  8. package/esm2022/lib/components/pdf-data-handler/function/formatToBrazilianNumber.mjs +7 -0
  9. package/esm2022/lib/components/pdf-data-handler/function/parseBrazilianNumber.mjs +9 -0
  10. package/esm2022/lib/components/pdf-data-handler/interface/DataRow.mjs +2 -0
  11. package/esm2022/lib/components/pdf-data-handler/interface/PdfExportConfig.mjs +2 -0
  12. package/esm2022/lib/components/pdf-data-handler/interface/PdfTableStyles.mjs +2 -0
  13. package/esm2022/lib/components/pdf-data-handler/interface/RowCalculation.mjs +2 -0
  14. package/esm2022/lib/components/pdf-data-handler/interface/TotalizerConfig.mjs +2 -0
  15. package/esm2022/lib/components/pdf-download/argenta-pdf-download.component.mjs +51 -0
  16. package/esm2022/public-api.mjs +36 -5
  17. package/fesm2022/lib-portal-angular.mjs +1095 -422
  18. package/fesm2022/lib-portal-angular.mjs.map +1 -1
  19. package/lib/components/components.module.d.ts +10 -5
  20. package/lib/components/csv-importer/csv-importer.component.d.ts +19 -0
  21. package/lib/components/data-excel-exporter/IExcelExportConfig.d.ts +21 -0
  22. package/lib/components/data-excel-exporter/data-excel-exporter.component.d.ts +9 -0
  23. package/lib/components/data-excel-import/data-excel-importer.component.d.ts +44 -0
  24. package/lib/components/multi-select/multi-select.component.d.ts +20 -9
  25. package/lib/components/pdf-data-handler/argenta-pdf-data-handler.component.d.ts +15 -0
  26. package/lib/components/pdf-data-handler/function/formatToBrazilianNumber.d.ts +1 -0
  27. package/lib/components/pdf-data-handler/function/parseBrazilianNumber.d.ts +1 -0
  28. package/lib/components/pdf-data-handler/interface/DataRow.d.ts +3 -0
  29. package/lib/components/pdf-data-handler/interface/PdfExportConfig.d.ts +12 -0
  30. package/lib/components/pdf-data-handler/interface/PdfTableStyles.d.ts +28 -0
  31. package/lib/components/pdf-data-handler/interface/RowCalculation.d.ts +4 -0
  32. package/lib/components/pdf-data-handler/interface/TotalizerConfig.d.ts +8 -0
  33. package/lib/components/pdf-download/argenta-pdf-download.component.d.ts +13 -0
  34. package/package.json +1 -1
  35. package/public-api.d.ts +34 -3
@@ -1,89 +1,130 @@
1
- import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, Input, Output } from '@angular/core';
2
- import { NG_VALUE_ACCESSOR } from '@angular/forms';
3
- import { of, Subject } from 'rxjs';
4
- import { catchError, debounceTime, map, startWith, switchMap } from 'rxjs/operators';
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, HostListener, Input, Output, ViewEncapsulation, } from "@angular/core";
2
+ import { NG_VALUE_ACCESSOR } from "@angular/forms";
3
+ import { of, Subject } from "rxjs";
4
+ import { catchError, debounceTime, distinctUntilChanged, map, switchMap, tap } 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";
8
8
  import * as i3 from "@angular/common";
9
- import * as i4 from "@angular/forms";
10
- import * as i5 from "@ng-select/ng-select";
9
+ import * as i4 from "lucide-angular";
11
10
  export class MultiSelectComponent {
12
- constructor(authService, http) {
11
+ onClickOutside(event) {
12
+ const target = event.target;
13
+ if (!target.closest(".custom-select-container")) {
14
+ this.closeDropdown();
15
+ }
16
+ }
17
+ constructor(authService, http, cdr) {
13
18
  this.authService = authService;
14
19
  this.http = http;
15
- this.label = 'Multi Select';
16
- this.data = []; // Accepts an array of generic objects
17
- this.placeholder = 'Select items';
18
- this.id = 'multiSelectId';
19
- this.bindLabel = ''; // Generic dynamic label
20
- this.bindValue = ''; // Generic dynamic value
21
- this.closeOnSelect = false; // New property to control dropdown close behavior
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
20
+ this.cdr = cdr;
21
+ this.label = "Multi Select";
22
+ this.data = [];
23
+ this.placeholder = "";
24
+ this.id = "multiSelectId";
25
+ this.bindLabel = "";
26
+ this.bindValue = "";
27
+ this.closeOnSelect = false;
28
+ this.searchUrl = "";
29
+ this.multiple = true;
30
+ this.searchParams = {};
25
31
  this.keyupEvent = new EventEmitter();
26
- this.backupData = []; // Backup of the initial data
27
- this.allItems = []; // Store the combined list
28
- this.items = of([]); // Initialization of the property
29
- this.filteredItems = of([]); // Filtered items
30
- this.searchTerms = new Subject(); // For search debounce
32
+ this.lastSearchTerm = "";
33
+ this.backupData = [];
34
+ this.allItems = [];
35
+ this.items = of([]);
36
+ this.filteredItems = of([]);
37
+ this.searchTerms = new Subject();
38
+ this.isOpen = false;
31
39
  this.onChangeCallback = () => { };
32
40
  this.onTouchedCallback = () => { };
33
41
  this.isCourseEntered = false;
34
42
  this.compareFn = (item1, item2) => {
35
- return item1 && item2 ? item1[this.bindValue] === item2[this.bindValue] : item1 === item2;
43
+ return item1 && item2
44
+ ? item1[this.bindValue] === item2[this.bindValue]
45
+ : item1 === item2;
36
46
  };
37
47
  }
38
48
  ngOnInit() {
39
- this.backupData = [...this.data]; // Backup initial data
40
- this.allItems = [...this.data]; // Initialize allItems with the initial data
41
- this.items = of(this.allItems);
42
- this.fetchInitialData().subscribe(data => {
49
+ this.filteredItems = this.searchTerms.pipe(debounceTime(700), // Reduz chamadas repetitivas
50
+ distinctUntilChanged(), // Ignora termos repetidos consecutivos
51
+ switchMap((term) => {
52
+ // Sempre busca no back-end
53
+ return this.search(term).pipe(tap((fetchedItems) => {
54
+ this.updateData(fetchedItems);
55
+ }), map((fetchedItems) => this.filterOutSelected(fetchedItems)));
56
+ }));
57
+ this.filteredItems.subscribe({
58
+ next: (items) => { },
59
+ error: (err) => console.error("Erro ao buscar no back-end:", err),
60
+ });
61
+ this.fetchInitialData().subscribe((data) => {
43
62
  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
63
  });
49
64
  }
50
- ngOnChanges(changes) {
51
- if (changes['selected'] && !changes['selected'].isFirstChange()) {
52
- this.addSelectedItemsToData();
65
+ onInputChange(event) {
66
+ const input = event.target.value.trim();
67
+ this.searchTerms.next(input);
68
+ if (input === "") {
69
+ this.fetchInitialData().subscribe((data) => {
70
+ this.allItems = data;
71
+ this.filteredItems = of(this.filterOutSelected(this.allItems));
72
+ });
53
73
  }
54
74
  }
75
+ search(term) {
76
+ return this.http.get(`${this.searchUrl}`, { params: { term } }).pipe(map((response) => {
77
+ const transformedItems = response.map((item) => ({
78
+ [this.bindValue]: item[this.bindValue],
79
+ [this.bindLabel]: item[this.bindLabel],
80
+ }));
81
+ this.allItems = transformedItems;
82
+ this.filteredItems = of(this.filterOutSelected(this.allItems));
83
+ this.cdr.markForCheck(); // Notifica o Angular sobre a alteração
84
+ return transformedItems;
85
+ }), catchError((error) => {
86
+ console.error("Erro ao buscar no back-end:", error);
87
+ this.filteredItems = of([]);
88
+ return of([]);
89
+ }));
90
+ }
55
91
  fetchInitialData() {
56
92
  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 [];
93
+ const transformedItems = response.map((item) => ({
94
+ [this.bindValue]: item[this.bindValue],
95
+ [this.bindLabel]: item[this.bindLabel],
96
+ }));
97
+ this.allItems = transformedItems;
98
+ return this.allItems;
64
99
  }), catchError((error) => {
65
- console.error('Error fetching initial data from backend:', error);
100
+ console.error("Erro ao carregar dados iniciais:", error);
66
101
  return of([]);
67
102
  }));
68
103
  }
104
+ ngOnChanges(changes) {
105
+ if (changes["selected"] && !changes["selected"].isFirstChange()) {
106
+ this.addSelectedItemsToData();
107
+ }
108
+ }
69
109
  updateData(newData) {
70
110
  newData.forEach((item) => {
71
- const existsInList = this.allItems.some(listItem => this.compareFn(listItem, item));
111
+ const existsInList = this.allItems.some((listItem) => this.compareFn(listItem, item));
72
112
  if (!existsInList) {
73
- this.allItems.push(item);
113
+ this.allItems = [...this.allItems, item]; // Cria uma nova referência
74
114
  }
75
115
  });
76
- this.items = of(this.allItems);
116
+ this.filteredItems = of([...this.allItems]); // Garante nova referência
117
+ this.cdr.markForCheck(); // Notifica o Angular sobre mudanças
77
118
  }
78
119
  addSelectedItemsToData() {
79
120
  if (this.selected) {
80
121
  const selectedItems = this.multiple ? this.selected : [this.selected];
81
122
  selectedItems.forEach((item) => {
82
- const existsInList = this.allItems.some(listItem => this.compareFn(listItem, item));
123
+ const existsInList = this.allItems.some((listItem) => this.compareFn(listItem, item));
83
124
  if (!existsInList) {
84
125
  const newItem = {
85
126
  [this.bindValue]: item[this.bindValue] || item,
86
- [this.bindLabel]: item[this.bindLabel] || item
127
+ [this.bindLabel]: item[this.bindLabel] || item,
87
128
  };
88
129
  this.data.push(newItem);
89
130
  this.allItems.push(newItem);
@@ -110,60 +151,15 @@ export class MultiSelectComponent {
110
151
  this.selected = event ? event : null;
111
152
  }
112
153
  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
154
+ if (previousSelected && Array.isArray(previousSelected)) {
155
+ const removedItems = previousSelected.filter((item) => !event.includes(item));
156
+ removedItems.forEach((item) => {
157
+ const existsInAllItems = this.allItems.some((listItem) => this.compareFn(listItem, item));
158
+ if (!existsInAllItems) {
159
+ this.allItems.push(item);
121
160
  }
122
161
  });
123
- }
124
- }
125
- onInputChange(event) {
126
- const input = event.target.value;
127
- this.searchTerms.next(input);
128
- }
129
- search(term) {
130
- if (!term.trim()) {
131
- // Se o termo de busca estiver vazio, retorna a lista completa
132
- return of(this.allItems);
133
- }
134
- // Filtra os itens localmente
135
- const filtered = this.allItems.filter((item) => item[this.bindLabel].toLowerCase().includes(term.toLowerCase()));
136
- if (filtered.length > 0) {
137
- console.log('Items filtered locally.');
138
- return of(filtered);
139
- }
140
- else if (this.searchUrl) {
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) => {
145
- if (response && response.length > 0) {
146
- // Transforma os itens do backend para o formato esperado pelo componente
147
- const transformedItems = response.map((item) => ({
148
- [this.bindValue]: item[this.bindValue],
149
- [this.bindLabel]: item[this.bindLabel]
150
- }));
151
- // Atualiza os dados com os novos itens buscados
152
- this.updateData(transformedItems);
153
- return this.allItems;
154
- }
155
- else {
156
- console.log('No items found in the backend search.');
157
- return this.allItems;
158
- }
159
- }), catchError((error) => {
160
- console.error('Error fetching from backend:', error);
161
- return of(this.allItems);
162
- }));
163
- }
164
- else {
165
- console.log('No search URL provided and no items found locally.');
166
- return of(this.allItems);
162
+ this.items = of(this.allItems);
167
163
  }
168
164
  }
169
165
  writeValue(value) {
@@ -174,6 +170,7 @@ export class MultiSelectComponent {
174
170
  this.selected = value || null;
175
171
  }
176
172
  this.addSelectedItemsToData();
173
+ this.cdr.detectChanges();
177
174
  }
178
175
  registerOnChange(fn) {
179
176
  this.onChangeCallback = fn;
@@ -181,9 +178,7 @@ export class MultiSelectComponent {
181
178
  registerOnTouched(fn) {
182
179
  this.onTouchedCallback = fn;
183
180
  }
184
- setDisabledState(isDisabled) {
185
- // No implementation needed for this example
186
- }
181
+ setDisabledState(isDisabled) { }
187
182
  hasPermission() {
188
183
  if (!this.permissions || this.permissions.length === 0) {
189
184
  return true;
@@ -193,33 +188,88 @@ export class MultiSelectComponent {
193
188
  }
194
189
  catch (error) {
195
190
  if (error instanceof Error) {
196
- console.error('Permission error:', error.message);
191
+ console.error("Permission error:", error.message);
197
192
  }
198
193
  else {
199
- console.error('Unknown error occurred during permission check');
194
+ console.error("Unknown error occurred during permission check");
200
195
  }
201
196
  return true;
202
197
  }
203
198
  }
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 }); }
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: [
199
+ toggleDropdown() {
200
+ this.isOpen = !this.isOpen;
201
+ }
202
+ closeDropdown() {
203
+ this.isOpen = false;
204
+ }
205
+ isSelected(item) {
206
+ if (this.multiple) {
207
+ return this.selected?.some((selectedItem) => this.compareFn(selectedItem, item));
208
+ }
209
+ else {
210
+ return this.compareFn(this.selected, item);
211
+ }
212
+ }
213
+ selectItem(item, event) {
214
+ event.stopPropagation();
215
+ if (!this.isSelected(item)) {
216
+ if (this.multiple) {
217
+ this.selected.push(item);
218
+ }
219
+ else {
220
+ this.selected = [item];
221
+ }
222
+ }
223
+ if (this.multiple) {
224
+ this.allItems = this.allItems.filter((listItem) => !this.compareFn(listItem, item));
225
+ }
226
+ this.filteredItems = of(this.filterOutSelected(this.allItems));
227
+ this.onSelectedChange(this.selected);
228
+ if (this.closeOnSelect) {
229
+ this.closeDropdown();
230
+ }
231
+ }
232
+ filterByTerm(term) {
233
+ return this.allItems.filter((item) => item[this.bindLabel]?.toLowerCase().includes(term.toLowerCase()));
234
+ }
235
+ filterOutSelected(items) {
236
+ if (!this.multiple) {
237
+ return items;
238
+ }
239
+ return items.filter((item) => !this.selected.some((selectedItem) => this.compareFn(selectedItem, item)));
240
+ }
241
+ removeSelectedItem(item, event) {
242
+ event.stopPropagation();
243
+ this.selected = this.selected.filter((selectedItem) => !this.compareFn(selectedItem, item));
244
+ const existsInAllItems = this.allItems.some((listItem) => this.compareFn(listItem, item));
245
+ if (!existsInAllItems) {
246
+ this.allItems.push(item);
247
+ }
248
+ this.filteredItems = of(this.filterByTerm(this.lastSearchTerm || ""));
249
+ this.onSelectedChange(this.selected);
250
+ }
251
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MultiSelectComponent, deps: [{ token: i1.AuthService }, { token: i2.HttpClient }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
252
+ 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" }, host: { listeners: { "document:click": "onClickOutside($event)" } }, providers: [
206
253
  {
207
254
  provide: NG_VALUE_ACCESSOR,
208
255
  useExisting: forwardRef(() => MultiSelectComponent),
209
- multi: true
210
- }
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\";.form-group{font-family:var(--font-family);font-size:1rem}.form-check-input{font-family:var(--font-family);color:#333;font-size:.9rem}.form-check-label{width:623px;height:19px;top:1608px;left:133px;gap:0px;opacity:0px;font-family:var(--font-family);font-size:16px;line-height:19.36px;text-align:left}.custom-select{font-family:var(--font-family);color:#333;font-size:.9rem;font-weight:400;border:1px solid #ccc;border-radius:4px;appearance:none;-webkit-appearance:none;-moz-appearance:none;padding-right:2rem;background-image:none;background-repeat:no-repeat;background-position:right .5rem center;height:46px}.custom-input{font-family:var(--font-family);color:#333;font-size:.9rem;height:46px}.custom-input::placeholder{font-size:14px;color:#d3d3d3}.form-label{font-family:var(--font-family);color:#333;font-size:1rem}.label-styles{font-weight:400;font-family:var(--font-family);font-size:16px;line-height:19.36px;text-align:left;margin-top:1rem;margin-bottom:.2rem}.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:var(--font-family);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 }); }
256
+ multi: true,
257
+ },
258
+ ], usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"hasPermission()\" class=\"form-group\">\n <label [for]=\"id\" class=\"form-label\" style=\"margin-top: 1rem\">\n {{ label }}\n </label>\n <div class=\"custom-select-container\" tabindex=\"0\">\n <!-- Input com itens selecionados -->\n <div class=\"custom-input\">\n <span class=\"selected-item\" *ngFor=\"let item of selected\">\n {{ item[bindLabel] }}\n <button\n class=\"remove-item-btn\"\n (click)=\"removeSelectedItem(item, $event)\"\n >\n \u00D7\n </button>\n </span>\n <input\n type=\"text\"\n class=\"input-field\"\n [id]=\"id\"\n [placeholder]=\"placeholder\"\n (focus)=\"toggleDropdown()\"\n (keyup)=\"onKeyUp($event)\"\n (input)=\"onInputChange($event)\"\n />\n <!-- \u00CDcone flutuante -->\n <lucide-icon\n [name]=\"isOpen ? 'align-justify' : 'chevron-down'\"\n class=\"dropdown-icon\"\n ></lucide-icon>\n </div>\n\n <!-- Dropdown suspenso -->\n <div *ngIf=\"isOpen\" class=\"dropdown\">\n <div\n *ngFor=\"let item of filteredItems | async\"\n class=\"dropdown-item\"\n [class.selected]=\"isSelected(item)\"\n (click)=\"selectItem(item, $event)\"\n >\n {{ item[bindLabel] }}\n </div>\n <div *ngIf=\"(filteredItems | async)?.length === 0\" class=\"empty-dropdown\">\n Nenhum item dispon\u00EDvel\n </div>\n </div>\n </div>\n</div>\n", styles: ["@charset \"UTF-8\";.custom-select-container{position:relative;display:flex;flex-direction:column;border:1px solid #ccc;border-radius:4px;padding:.5rem;background-color:#fff;color:#333;font-family:var(--font-family)}.custom-select-container .custom-input{display:flex;flex-wrap:wrap;align-items:center;gap:.5rem;width:100%;min-height:20px;cursor:text;background:none;border:none;outline:none;position:relative}.custom-select-container .custom-input .selected-item{display:flex;align-items:center;background-color:#f8f8f8;color:#333;border:1px solid #e0e0e0;border-radius:4px;padding:.15rem .4rem;font-size:.85rem}.custom-select-container .custom-input .selected-item .remove-item-btn{display:flex;align-items:center;justify-content:center;background-color:#c7c7c7;border:none;color:#4b4b4b;font-weight:700;cursor:pointer;font-size:.6rem;line-height:1;border-radius:50%;width:14px;height:14px;margin-left:.3rem;padding:0;position:relative}.custom-select-container .custom-input .selected-item .remove-item-btn span{transition:transform .2s ease,color .2s ease}.custom-select-container .custom-input .selected-item .remove-item-btn:hover{background-color:#8f8a8b;color:#fff}.custom-select-container .custom-input .selected-item .remove-item-btn:hover span{transform:scale(1.2);color:#fff}.custom-select-container .custom-input .selected-item .remove-item-btn:active{background-color:#767072}.custom-select-container .custom-input .input-field{flex:1;border:none;outline:none;font-size:.85rem;padding:.1rem 3rem .1rem .2rem;background:transparent;min-width:80px}.custom-select-container .custom-input .dropdown-icon{position:absolute;right:.1rem;top:50%;transform:translateY(-50%);font-size:1.2rem;color:#888;pointer-events:none}.dropdown{position:absolute;top:100%;left:0;right:0;max-height:200px;overflow-y:auto;background-color:#fff;border:1px solid #ccc;border-radius:4px;z-index:1000}.dropdown::-webkit-scrollbar{width:8px}.dropdown::-webkit-scrollbar-thumb{background-color:#ccc;border-radius:4px}.dropdown::-webkit-scrollbar-thumb:hover{background-color:#aaa}.dropdown::-webkit-scrollbar-track{background-color:#f1f1f1;border-radius:4px}.dropdown .dropdown-item{padding:.5rem;cursor:pointer}.dropdown .dropdown-item:hover{background-color:#f1f1f1}.dropdown .dropdown-item.selected{font-weight:700;background-color:#e9e9e9}.empty-dropdown{padding:1rem;text-align:center;color:#999;font-size:.9rem}@media (max-width: 768px){.custom-input .input-field{padding-right:4rem}.custom-input .dropdown-icon{right:.75rem}}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.Default, encapsulation: i0.ViewEncapsulation.None }); }
212
259
  }
213
260
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MultiSelectComponent, decorators: [{
214
261
  type: Component,
215
- args: [{ selector: 'argenta-custom-multi-select', providers: [
262
+ args: [{ selector: "argenta-custom-multi-select", encapsulation: ViewEncapsulation.None, providers: [
216
263
  {
217
264
  provide: NG_VALUE_ACCESSOR,
218
265
  useExisting: forwardRef(() => MultiSelectComponent),
219
- multi: true
220
- }
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\";.form-group{font-family:var(--font-family);font-size:1rem}.form-check-input{font-family:var(--font-family);color:#333;font-size:.9rem}.form-check-label{width:623px;height:19px;top:1608px;left:133px;gap:0px;opacity:0px;font-family:var(--font-family);font-size:16px;line-height:19.36px;text-align:left}.custom-select{font-family:var(--font-family);color:#333;font-size:.9rem;font-weight:400;border:1px solid #ccc;border-radius:4px;appearance:none;-webkit-appearance:none;-moz-appearance:none;padding-right:2rem;background-image:none;background-repeat:no-repeat;background-position:right .5rem center;height:46px}.custom-input{font-family:var(--font-family);color:#333;font-size:.9rem;height:46px}.custom-input::placeholder{font-size:14px;color:#d3d3d3}.form-label{font-family:var(--font-family);color:#333;font-size:1rem}.label-styles{font-weight:400;font-family:var(--font-family);font-size:16px;line-height:19.36px;text-align:left;margin-top:1rem;margin-bottom:.2rem}.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:var(--font-family);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"] }]
222
- }], ctorParameters: function () { return [{ type: i1.AuthService }, { type: i2.HttpClient }]; }, propDecorators: { label: [{
266
+ multi: true,
267
+ },
268
+ ], changeDetection: ChangeDetectionStrategy.Default, template: "<div *ngIf=\"hasPermission()\" class=\"form-group\">\n <label [for]=\"id\" class=\"form-label\" style=\"margin-top: 1rem\">\n {{ label }}\n </label>\n <div class=\"custom-select-container\" tabindex=\"0\">\n <!-- Input com itens selecionados -->\n <div class=\"custom-input\">\n <span class=\"selected-item\" *ngFor=\"let item of selected\">\n {{ item[bindLabel] }}\n <button\n class=\"remove-item-btn\"\n (click)=\"removeSelectedItem(item, $event)\"\n >\n \u00D7\n </button>\n </span>\n <input\n type=\"text\"\n class=\"input-field\"\n [id]=\"id\"\n [placeholder]=\"placeholder\"\n (focus)=\"toggleDropdown()\"\n (keyup)=\"onKeyUp($event)\"\n (input)=\"onInputChange($event)\"\n />\n <!-- \u00CDcone flutuante -->\n <lucide-icon\n [name]=\"isOpen ? 'align-justify' : 'chevron-down'\"\n class=\"dropdown-icon\"\n ></lucide-icon>\n </div>\n\n <!-- Dropdown suspenso -->\n <div *ngIf=\"isOpen\" class=\"dropdown\">\n <div\n *ngFor=\"let item of filteredItems | async\"\n class=\"dropdown-item\"\n [class.selected]=\"isSelected(item)\"\n (click)=\"selectItem(item, $event)\"\n >\n {{ item[bindLabel] }}\n </div>\n <div *ngIf=\"(filteredItems | async)?.length === 0\" class=\"empty-dropdown\">\n Nenhum item dispon\u00EDvel\n </div>\n </div>\n </div>\n</div>\n", styles: ["@charset \"UTF-8\";.custom-select-container{position:relative;display:flex;flex-direction:column;border:1px solid #ccc;border-radius:4px;padding:.5rem;background-color:#fff;color:#333;font-family:var(--font-family)}.custom-select-container .custom-input{display:flex;flex-wrap:wrap;align-items:center;gap:.5rem;width:100%;min-height:20px;cursor:text;background:none;border:none;outline:none;position:relative}.custom-select-container .custom-input .selected-item{display:flex;align-items:center;background-color:#f8f8f8;color:#333;border:1px solid #e0e0e0;border-radius:4px;padding:.15rem .4rem;font-size:.85rem}.custom-select-container .custom-input .selected-item .remove-item-btn{display:flex;align-items:center;justify-content:center;background-color:#c7c7c7;border:none;color:#4b4b4b;font-weight:700;cursor:pointer;font-size:.6rem;line-height:1;border-radius:50%;width:14px;height:14px;margin-left:.3rem;padding:0;position:relative}.custom-select-container .custom-input .selected-item .remove-item-btn span{transition:transform .2s ease,color .2s ease}.custom-select-container .custom-input .selected-item .remove-item-btn:hover{background-color:#8f8a8b;color:#fff}.custom-select-container .custom-input .selected-item .remove-item-btn:hover span{transform:scale(1.2);color:#fff}.custom-select-container .custom-input .selected-item .remove-item-btn:active{background-color:#767072}.custom-select-container .custom-input .input-field{flex:1;border:none;outline:none;font-size:.85rem;padding:.1rem 3rem .1rem .2rem;background:transparent;min-width:80px}.custom-select-container .custom-input .dropdown-icon{position:absolute;right:.1rem;top:50%;transform:translateY(-50%);font-size:1.2rem;color:#888;pointer-events:none}.dropdown{position:absolute;top:100%;left:0;right:0;max-height:200px;overflow-y:auto;background-color:#fff;border:1px solid #ccc;border-radius:4px;z-index:1000}.dropdown::-webkit-scrollbar{width:8px}.dropdown::-webkit-scrollbar-thumb{background-color:#ccc;border-radius:4px}.dropdown::-webkit-scrollbar-thumb:hover{background-color:#aaa}.dropdown::-webkit-scrollbar-track{background-color:#f1f1f1;border-radius:4px}.dropdown .dropdown-item{padding:.5rem;cursor:pointer}.dropdown .dropdown-item:hover{background-color:#f1f1f1}.dropdown .dropdown-item.selected{font-weight:700;background-color:#e9e9e9}.empty-dropdown{padding:1rem;text-align:center;color:#999;font-size:.9rem}@media (max-width: 768px){.custom-input .input-field{padding-right:4rem}.custom-input .dropdown-icon{right:.75rem}}\n"] }]
269
+ }], ctorParameters: function () { return [{ type: i1.AuthService }, { type: i2.HttpClient }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { onClickOutside: [{
270
+ type: HostListener,
271
+ args: ["document:click", ["$event"]]
272
+ }], label: [{
223
273
  type: Input
224
274
  }], data: [{
225
275
  type: Input
@@ -246,4 +296,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
246
296
  }], keyupEvent: [{
247
297
  type: Output
248
298
  }] } });
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"]}
299
+ //# 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 {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  forwardRef,\n  HostListener,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewEncapsulation,\n} from \"@angular/core\";\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from \"@angular/forms\";\nimport { Observable, of, Subject } from \"rxjs\";\nimport {\n  catchError,\n  debounceTime,\n  distinctUntilChanged,\n  map,\n  switchMap,\n  tap\n} 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  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectComponent),\n      multi: true,\n    },\n  ],\n  changeDetection: ChangeDetectionStrategy.Default,\n})\nexport class MultiSelectComponent\n  implements ControlValueAccessor, OnInit, OnChanges\n{\n  @HostListener(\"document:click\", [\"$event\"])\n  onClickOutside(event: Event) {\n    const target = event.target as HTMLElement;\n    if (!target.closest(\".custom-select-container\")) {\n      this.closeDropdown();\n    }\n  }\n\n  @Input() label: string = \"Multi Select\";\n  @Input() data: any[] = [];\n  @Input() placeholder: string = \"\";\n  @Input() selected: any;\n  @Input() id: string = \"multiSelectId\";\n  @Input() bindLabel: string = \"\";\n  @Input() bindValue: string = \"\";\n  @Input() permissions: string[] | undefined;\n  @Input() closeOnSelect: boolean = false;\n  @Input() searchUrl: string = \"\";\n  @Input() multiple: boolean = true;\n  @Input() searchParams: { [key: string]: string } = {};\n\n  @Output() keyupEvent: EventEmitter<any> = new EventEmitter<any>();\n\n  private lastSearchTerm: string = \"\";\n  private backupData: any[] = [];\n  private allItems: any[] = [];\n  items: Observable<any[]> = of([]);\n  filteredItems: Observable<any[]> = of([]);\n  private searchTerms = new Subject<string>();\n  isOpen = false;\n\n  constructor(\n    private authService: AuthService,\n    private http: HttpClient,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.filteredItems = this.searchTerms.pipe(\n      debounceTime(700), // Reduz chamadas repetitivas\n      distinctUntilChanged(), // Ignora termos repetidos consecutivos\n      switchMap((term) => {\n        // Sempre busca no back-end\n        return this.search(term).pipe(\n          tap((fetchedItems) => {\n            this.updateData(fetchedItems);\n          }),\n          map((fetchedItems) => this.filterOutSelected(fetchedItems))\n        );\n      })\n    );\n\n    this.filteredItems.subscribe({\n      next: (items: any) => {},\n      error: (err) => console.error(\"Erro ao buscar no back-end:\", err),\n    });\n\n    this.fetchInitialData().subscribe((data) => {\n      this.updateData(data);\n    });\n  }\n\n  onInputChange(event: any): void {\n    const input = event.target.value.trim();\n\n    this.searchTerms.next(input);\n\n    if (input === \"\") {\n      this.fetchInitialData().subscribe((data) => {\n        this.allItems = data;\n        this.filteredItems = of(this.filterOutSelected(this.allItems));\n      });\n    }\n  }\n\n  search(term: string): Observable<any[]> {\n    return this.http.get<any[]>(`${this.searchUrl}`, { params: { term } }).pipe(\n      map((response: any[]) => {\n        const transformedItems = response.map((item: any) => ({\n          [this.bindValue]: item[this.bindValue],\n          [this.bindLabel]: item[this.bindLabel],\n        }));\n\n        this.allItems = transformedItems;\n\n        this.filteredItems = of(this.filterOutSelected(this.allItems));\n        this.cdr.markForCheck(); // Notifica o Angular sobre a alteração\n\n        return transformedItems;\n      }),\n      catchError((error) => {\n        console.error(\"Erro ao buscar no back-end:\", error);\n        this.filteredItems = of([]);\n        return of([]);\n      })\n    );\n  }\n\n  private fetchInitialData(): Observable<any[]> {\n    return this.http.get<any[]>(this.searchUrl).pipe(\n      map((response: any[]) => {\n        const transformedItems = response.map((item: any) => ({\n          [this.bindValue]: item[this.bindValue],\n          [this.bindLabel]: item[this.bindLabel],\n        }));\n\n        this.allItems = transformedItems;\n        return this.allItems;\n      }),\n      catchError((error: any) => {\n        console.error(\"Erro ao carregar dados iniciais:\", error);\n        return of([]);\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[\"selected\"] && !changes[\"selected\"].isFirstChange()) {\n      this.addSelectedItemsToData();\n    }\n  }\n\n  private updateData(newData: any[]): void {\n    newData.forEach((item: any) => {\n      const existsInList = this.allItems.some((listItem) =>\n        this.compareFn(listItem, item)\n      );\n      if (!existsInList) {\n        this.allItems = [...this.allItems, item]; // Cria uma nova referência\n      }\n    });\n  \n    this.filteredItems = of([...this.allItems]); // Garante nova referência\n    this.cdr.markForCheck(); // Notifica o Angular sobre mudanças\n  }\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) =>\n          this.compareFn(listItem, item)\n        );\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    if (previousSelected && Array.isArray(previousSelected)) {\n      const removedItems = previousSelected.filter(\n        (item) => !event.includes(item)\n      );\n\n      removedItems.forEach((item) => {\n        const existsInAllItems = this.allItems.some((listItem) =>\n          this.compareFn(listItem, item)\n        );\n        if (!existsInAllItems) {\n          this.allItems.push(item);\n        }\n      });\n      this.items = 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    this.cdr.detectChanges();\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\n  compareFn = (item1: any, item2: any): boolean => {\n    return item1 && item2\n      ? item1[this.bindValue] === item2[this.bindValue]\n      : 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  toggleDropdown() {\n    this.isOpen = !this.isOpen;\n  }\n\n  closeDropdown() {\n    this.isOpen = false;\n  }\n\n  isSelected(item: any): boolean {\n    if (this.multiple) {\n      return this.selected?.some((selectedItem: any) =>\n        this.compareFn(selectedItem, item)\n      );\n    } else {\n      return this.compareFn(this.selected, item);\n    }\n  }\n\n  selectItem(item: any, event: Event): void {\n    event.stopPropagation();\n\n    if (!this.isSelected(item)) {\n      if (this.multiple) {\n        this.selected.push(item);\n      } else {\n        this.selected = [item];\n      }\n    }\n\n    if (this.multiple) {\n      this.allItems = this.allItems.filter(\n        (listItem) => !this.compareFn(listItem, item)\n      );\n    }\n\n    this.filteredItems = of(this.filterOutSelected(this.allItems));\n    this.onSelectedChange(this.selected);\n\n    if (this.closeOnSelect) {\n      this.closeDropdown();\n    }\n  }\n\n  private filterByTerm(term: string): any[] {\n    return this.allItems.filter((item) =>\n      item[this.bindLabel]?.toLowerCase().includes(term.toLowerCase())\n    );\n  }\n\n  private filterOutSelected(items: any[]): any[] {\n    if (!this.multiple) {\n      return items;\n    }\n\n    return items.filter(\n      (item) =>\n        !this.selected.some((selectedItem: any) =>\n          this.compareFn(selectedItem, item)\n        )\n    );\n  }\n\n  removeSelectedItem(item: any, event: Event): void {\n    event.stopPropagation();\n\n    this.selected = this.selected.filter(\n      (selectedItem: any) => !this.compareFn(selectedItem, item)\n    );\n\n    const existsInAllItems = this.allItems.some((listItem) =>\n      this.compareFn(listItem, item)\n    );\n\n    if (!existsInAllItems) {\n      this.allItems.push(item);\n    }\n\n    this.filteredItems = of(this.filterByTerm(this.lastSearchTerm || \"\"));\n    this.onSelectedChange(this.selected);\n  }\n}\n","<div *ngIf=\"hasPermission()\" class=\"form-group\">\n  <label [for]=\"id\" class=\"form-label\" style=\"margin-top: 1rem\">\n    {{ label }}\n  </label>\n  <div class=\"custom-select-container\" tabindex=\"0\">\n    <!-- Input com itens selecionados -->\n    <div class=\"custom-input\">\n      <span class=\"selected-item\" *ngFor=\"let item of selected\">\n        {{ item[bindLabel] }}\n        <button\n          class=\"remove-item-btn\"\n          (click)=\"removeSelectedItem(item, $event)\"\n        >\n          ×\n        </button>\n      </span>\n      <input\n        type=\"text\"\n        class=\"input-field\"\n        [id]=\"id\"\n        [placeholder]=\"placeholder\"\n        (focus)=\"toggleDropdown()\"\n        (keyup)=\"onKeyUp($event)\"\n        (input)=\"onInputChange($event)\"\n      />\n      <!-- Ícone flutuante -->\n      <lucide-icon\n        [name]=\"isOpen ? 'align-justify' : 'chevron-down'\"\n        class=\"dropdown-icon\"\n      ></lucide-icon>\n    </div>\n\n    <!-- Dropdown suspenso -->\n    <div *ngIf=\"isOpen\" class=\"dropdown\">\n      <div\n        *ngFor=\"let item of filteredItems | async\"\n        class=\"dropdown-item\"\n        [class.selected]=\"isSelected(item)\"\n        (click)=\"selectItem(item, $event)\"\n      >\n        {{ item[bindLabel] }}\n      </div>\n      <div *ngIf=\"(filteredItems | async)?.length === 0\" class=\"empty-dropdown\">\n        Nenhum item disponível\n      </div>\n    </div>\n  </div>\n</div>\n"]}