lib-portal-angular 0.0.87 → 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.
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 +162 -114
  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 +1093 -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,129 @@
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
113
  this.allItems.push(item);
74
114
  }
75
115
  });
76
- this.items = of(this.allItems);
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
- // 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
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('Permission error:', error.message);
189
+ console.error("Permission error:", error.message);
197
190
  }
198
191
  else {
199
- console.error('Unknown error occurred during permission check');
192
+ console.error("Unknown error occurred during permission check");
200
193
  }
201
194
  return true;
202
195
  }
203
196
  }
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: [
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;\">{{ 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 }); }
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: 'argenta-custom-multi-select', providers: [
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;\">{{ 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: [{
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"]}