lib-portal-angular 0.0.86 → 0.0.88
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/components.module.mjs +65 -40
- package/esm2022/lib/components/csv-importer/csv-importer.component.mjs +92 -0
- package/esm2022/lib/components/data-excel-exporter/IExcelExportConfig.mjs +2 -0
- package/esm2022/lib/components/data-excel-exporter/data-excel-exporter.component.mjs +126 -0
- package/esm2022/lib/components/data-excel-import/data-excel-importer.component.mjs +198 -0
- package/esm2022/lib/components/multi-select/multi-select.component.mjs +162 -114
- package/esm2022/lib/components/pdf-data-handler/argenta-pdf-data-handler.component.mjs +147 -0
- package/esm2022/lib/components/pdf-data-handler/function/formatToBrazilianNumber.mjs +7 -0
- package/esm2022/lib/components/pdf-data-handler/function/parseBrazilianNumber.mjs +9 -0
- package/esm2022/lib/components/pdf-data-handler/interface/DataRow.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/PdfExportConfig.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/PdfTableStyles.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/RowCalculation.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/TotalizerConfig.mjs +2 -0
- package/esm2022/lib/components/pdf-download/argenta-pdf-download.component.mjs +51 -0
- package/esm2022/public-api.mjs +36 -5
- package/fesm2022/lib-portal-angular.mjs +1093 -422
- package/fesm2022/lib-portal-angular.mjs.map +1 -1
- package/lib/components/components.module.d.ts +10 -5
- package/lib/components/csv-importer/csv-importer.component.d.ts +19 -0
- package/lib/components/data-excel-exporter/IExcelExportConfig.d.ts +21 -0
- package/lib/components/data-excel-exporter/data-excel-exporter.component.d.ts +9 -0
- package/lib/components/data-excel-import/data-excel-importer.component.d.ts +44 -0
- package/lib/components/multi-select/multi-select.component.d.ts +20 -9
- package/lib/components/pdf-data-handler/argenta-pdf-data-handler.component.d.ts +15 -0
- package/lib/components/pdf-data-handler/function/formatToBrazilianNumber.d.ts +1 -0
- package/lib/components/pdf-data-handler/function/parseBrazilianNumber.d.ts +1 -0
- package/lib/components/pdf-data-handler/interface/DataRow.d.ts +3 -0
- package/lib/components/pdf-data-handler/interface/PdfExportConfig.d.ts +12 -0
- package/lib/components/pdf-data-handler/interface/PdfTableStyles.d.ts +28 -0
- package/lib/components/pdf-data-handler/interface/RowCalculation.d.ts +4 -0
- package/lib/components/pdf-data-handler/interface/TotalizerConfig.d.ts +8 -0
- package/lib/components/pdf-download/argenta-pdf-download.component.d.ts +13 -0
- package/package.json +1 -1
- package/public-api.d.ts +34 -3
@@ -1,89 +1,129 @@
|
|
1
|
-
import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, Input, Output } from
|
2
|
-
import { NG_VALUE_ACCESSOR } from
|
3
|
-
import { of, Subject } from
|
4
|
-
import { catchError, debounceTime,
|
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 "
|
10
|
-
import * as i5 from "@ng-select/ng-select";
|
9
|
+
import * as i4 from "lucide-angular";
|
11
10
|
export class MultiSelectComponent {
|
12
|
-
|
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.
|
16
|
-
this.
|
17
|
-
this.
|
18
|
-
this.
|
19
|
-
this.
|
20
|
-
this.
|
21
|
-
this.
|
22
|
-
this.
|
23
|
-
this.
|
24
|
-
this.
|
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.
|
27
|
-
this.
|
28
|
-
this.
|
29
|
-
this.
|
30
|
-
this.
|
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
|
43
|
+
return item1 && item2
|
44
|
+
? item1[this.bindValue] === item2[this.bindValue]
|
45
|
+
: item1 === item2;
|
36
46
|
};
|
37
47
|
}
|
38
48
|
ngOnInit() {
|
39
|
-
this.
|
40
|
-
|
41
|
-
|
42
|
-
|
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
|
-
|
51
|
-
|
52
|
-
|
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
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
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(
|
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
113
|
this.allItems.push(item);
|
74
114
|
}
|
75
115
|
});
|
76
|
-
this.
|
116
|
+
this.filteredItems = of(this.allItems);
|
77
117
|
}
|
78
118
|
addSelectedItemsToData() {
|
79
119
|
if (this.selected) {
|
80
120
|
const selectedItems = this.multiple ? this.selected : [this.selected];
|
81
121
|
selectedItems.forEach((item) => {
|
82
|
-
const existsInList = this.allItems.some(listItem => this.compareFn(listItem, item));
|
122
|
+
const existsInList = this.allItems.some((listItem) => this.compareFn(listItem, item));
|
83
123
|
if (!existsInList) {
|
84
124
|
const newItem = {
|
85
125
|
[this.bindValue]: item[this.bindValue] || item,
|
86
|
-
[this.bindLabel]: item[this.bindLabel] || item
|
126
|
+
[this.bindLabel]: item[this.bindLabel] || item,
|
87
127
|
};
|
88
128
|
this.data.push(newItem);
|
89
129
|
this.allItems.push(newItem);
|
@@ -110,60 +150,15 @@ export class MultiSelectComponent {
|
|
110
150
|
this.selected = event ? event : null;
|
111
151
|
}
|
112
152
|
this.onChangeCallback(this.selected);
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
this.data.push(item);
|
120
|
-
this.allItems.push(item); // Adicionar de volta aos itens disponíveis
|
153
|
+
if (previousSelected && Array.isArray(previousSelected)) {
|
154
|
+
const removedItems = previousSelected.filter((item) => !event.includes(item));
|
155
|
+
removedItems.forEach((item) => {
|
156
|
+
const existsInAllItems = this.allItems.some((listItem) => this.compareFn(listItem, item));
|
157
|
+
if (!existsInAllItems) {
|
158
|
+
this.allItems.push(item);
|
121
159
|
}
|
122
160
|
});
|
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);
|
161
|
+
this.items = of(this.allItems);
|
167
162
|
}
|
168
163
|
}
|
169
164
|
writeValue(value) {
|
@@ -181,9 +176,7 @@ export class MultiSelectComponent {
|
|
181
176
|
registerOnTouched(fn) {
|
182
177
|
this.onTouchedCallback = fn;
|
183
178
|
}
|
184
|
-
setDisabledState(isDisabled) {
|
185
|
-
// No implementation needed for this example
|
186
|
-
}
|
179
|
+
setDisabledState(isDisabled) { }
|
187
180
|
hasPermission() {
|
188
181
|
if (!this.permissions || this.permissions.length === 0) {
|
189
182
|
return true;
|
@@ -193,33 +186,88 @@ export class MultiSelectComponent {
|
|
193
186
|
}
|
194
187
|
catch (error) {
|
195
188
|
if (error instanceof Error) {
|
196
|
-
console.error(
|
189
|
+
console.error("Permission error:", error.message);
|
197
190
|
}
|
198
191
|
else {
|
199
|
-
console.error(
|
192
|
+
console.error("Unknown error occurred during permission check");
|
200
193
|
}
|
201
194
|
return true;
|
202
195
|
}
|
203
196
|
}
|
204
|
-
|
205
|
-
|
197
|
+
toggleDropdown() {
|
198
|
+
this.isOpen = !this.isOpen;
|
199
|
+
}
|
200
|
+
closeDropdown() {
|
201
|
+
this.isOpen = false;
|
202
|
+
}
|
203
|
+
isSelected(item) {
|
204
|
+
if (this.multiple) {
|
205
|
+
return this.selected?.some((selectedItem) => this.compareFn(selectedItem, item));
|
206
|
+
}
|
207
|
+
else {
|
208
|
+
return this.compareFn(this.selected, item);
|
209
|
+
}
|
210
|
+
}
|
211
|
+
selectItem(item, event) {
|
212
|
+
event.stopPropagation();
|
213
|
+
if (!this.isSelected(item)) {
|
214
|
+
if (this.multiple) {
|
215
|
+
this.selected.push(item);
|
216
|
+
}
|
217
|
+
else {
|
218
|
+
this.selected = [item];
|
219
|
+
}
|
220
|
+
}
|
221
|
+
if (this.multiple) {
|
222
|
+
this.allItems = this.allItems.filter((listItem) => !this.compareFn(listItem, item));
|
223
|
+
}
|
224
|
+
this.filteredItems = of(this.filterOutSelected(this.allItems));
|
225
|
+
this.onSelectedChange(this.selected);
|
226
|
+
if (this.closeOnSelect) {
|
227
|
+
this.closeDropdown();
|
228
|
+
}
|
229
|
+
}
|
230
|
+
filterByTerm(term) {
|
231
|
+
return this.allItems.filter((item) => item[this.bindLabel]?.toLowerCase().includes(term.toLowerCase()));
|
232
|
+
}
|
233
|
+
filterOutSelected(items) {
|
234
|
+
if (!this.multiple) {
|
235
|
+
return items;
|
236
|
+
}
|
237
|
+
return items.filter((item) => !this.selected.some((selectedItem) => this.compareFn(selectedItem, item)));
|
238
|
+
}
|
239
|
+
removeSelectedItem(item, event) {
|
240
|
+
event.stopPropagation();
|
241
|
+
this.selected = this.selected.filter((selectedItem) => !this.compareFn(selectedItem, item));
|
242
|
+
const existsInAllItems = this.allItems.some((listItem) => this.compareFn(listItem, item));
|
243
|
+
if (!existsInAllItems) {
|
244
|
+
this.allItems.push(item);
|
245
|
+
}
|
246
|
+
this.filteredItems = of(this.filterByTerm(this.lastSearchTerm || ""));
|
247
|
+
this.onSelectedChange(this.selected);
|
248
|
+
}
|
249
|
+
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 }); }
|
250
|
+
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
251
|
{
|
207
252
|
provide: NG_VALUE_ACCESSOR,
|
208
253
|
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
|
254
|
+
multi: true,
|
255
|
+
},
|
256
|
+
], 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.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
212
257
|
}
|
213
258
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MultiSelectComponent, decorators: [{
|
214
259
|
type: Component,
|
215
|
-
args: [{ selector:
|
260
|
+
args: [{ selector: "argenta-custom-multi-select", encapsulation: ViewEncapsulation.None, providers: [
|
216
261
|
{
|
217
262
|
provide: NG_VALUE_ACCESSOR,
|
218
263
|
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
|
222
|
-
}], ctorParameters: function () { return [{ type: i1.AuthService }, { type: i2.HttpClient }]; }, propDecorators: {
|
264
|
+
multi: true,
|
265
|
+
},
|
266
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, 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"] }]
|
267
|
+
}], ctorParameters: function () { return [{ type: i1.AuthService }, { type: i2.HttpClient }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { onClickOutside: [{
|
268
|
+
type: HostListener,
|
269
|
+
args: ["document:click", ["$event"]]
|
270
|
+
}], label: [{
|
223
271
|
type: Input
|
224
272
|
}], data: [{
|
225
273
|
type: Input
|
@@ -246,4 +294,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
246
294
|
}], keyupEvent: [{
|
247
295
|
type: Output
|
248
296
|
}] } });
|
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"]}
|
297
|
+
//# 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  startWith,\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.OnPush,\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.push(item);\n      }\n    });\n\n    this.filteredItems = 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) =>\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  }\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"]}
|