ng-firebase-table-kxp 1.2.0 → 1.2.2

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 (48) hide show
  1. package/{fesm2020 → fesm2022}/ng-firebase-table-kxp.mjs +2762 -2744
  2. package/fesm2022/ng-firebase-table-kxp.mjs.map +1 -0
  3. package/index.d.ts +665 -5
  4. package/package.json +3 -11
  5. package/esm2020/lib/components/table/table.component.mjs +0 -710
  6. package/esm2020/lib/components/table-tabs/table-tabs.component.mjs +0 -73
  7. package/esm2020/lib/components/table-tooltip/table-tooltip.component.mjs +0 -34
  8. package/esm2020/lib/ng-firebase-table-kxp.component.mjs +0 -11
  9. package/esm2020/lib/ng-firebase-table-kxp.module.mjs +0 -98
  10. package/esm2020/lib/ng-firebase-table-kxp.service.mjs +0 -14
  11. package/esm2020/lib/services/filter.service.mjs +0 -416
  12. package/esm2020/lib/services/pagination.service.mjs +0 -115
  13. package/esm2020/lib/services/table.service.mjs +0 -1140
  14. package/esm2020/lib/services/tooltip.service.mjs +0 -141
  15. package/esm2020/lib/types/Table.mjs +0 -9
  16. package/esm2020/lib/utils/table.utils.mjs +0 -75
  17. package/esm2020/ng-firebase-table-kxp.mjs +0 -5
  18. package/esm2020/public-api.mjs +0 -22
  19. package/fesm2015/ng-firebase-table-kxp.mjs +0 -2869
  20. package/fesm2015/ng-firebase-table-kxp.mjs.map +0 -1
  21. package/fesm2020/ng-firebase-table-kxp.mjs.map +0 -1
  22. package/lib/components/table/table.component.d.ts +0 -132
  23. package/lib/components/table/table.component.d.ts.map +0 -1
  24. package/lib/components/table-tabs/table-tabs.component.d.ts +0 -34
  25. package/lib/components/table-tabs/table-tabs.component.d.ts.map +0 -1
  26. package/lib/components/table-tooltip/table-tooltip.component.d.ts +0 -18
  27. package/lib/components/table-tooltip/table-tooltip.component.d.ts.map +0 -1
  28. package/lib/ng-firebase-table-kxp.component.d.ts +0 -5
  29. package/lib/ng-firebase-table-kxp.component.d.ts.map +0 -1
  30. package/lib/ng-firebase-table-kxp.module.d.ts +0 -23
  31. package/lib/ng-firebase-table-kxp.module.d.ts.map +0 -1
  32. package/lib/ng-firebase-table-kxp.service.d.ts +0 -6
  33. package/lib/ng-firebase-table-kxp.service.d.ts.map +0 -1
  34. package/lib/services/filter.service.d.ts +0 -88
  35. package/lib/services/filter.service.d.ts.map +0 -1
  36. package/lib/services/pagination.service.d.ts +0 -34
  37. package/lib/services/pagination.service.d.ts.map +0 -1
  38. package/lib/services/table.service.d.ts +0 -80
  39. package/lib/services/table.service.d.ts.map +0 -1
  40. package/lib/services/tooltip.service.d.ts +0 -73
  41. package/lib/services/tooltip.service.d.ts.map +0 -1
  42. package/lib/types/Table.d.ts +0 -162
  43. package/lib/types/Table.d.ts.map +0 -1
  44. package/lib/utils/table.utils.d.ts +0 -25
  45. package/lib/utils/table.utils.d.ts.map +0 -1
  46. package/ng-firebase-table-kxp.d.ts.map +0 -1
  47. package/public-api.d.ts +0 -12
  48. package/public-api.d.ts.map +0 -1
@@ -1,710 +0,0 @@
1
- import { TABLE_DEFAULTS, } from '../../types/Table';
2
- import { Component, Input, ViewChild, ChangeDetectionStrategy, } from '@angular/core';
3
- import { MatPaginator } from '@angular/material/paginator';
4
- import { MatSort } from '@angular/material/sort';
5
- import { MatTableDataSource } from '@angular/material/table';
6
- import { getDisplayValue, getNestedValue, isString, trackByProperty, getImageKey, } from '../../utils/table.utils';
7
- import { debounceTime, firstValueFrom, Subject, takeUntil, BehaviorSubject, } from 'rxjs';
8
- import { FormControl, FormArray } from '@angular/forms';
9
- import * as i0 from "@angular/core";
10
- import * as i1 from "@angular/router";
11
- import * as i2 from "../../services/table.service";
12
- import * as i3 from "../../services/filter.service";
13
- import * as i4 from "../../services/tooltip.service";
14
- import * as i5 from "../../services/pagination.service";
15
- import * as i6 from "@angular/fire/compat/firestore";
16
- import * as i7 from "@angular/common";
17
- import * as i8 from "@angular/forms";
18
- import * as i9 from "@angular/material/table";
19
- import * as i10 from "@angular/material/paginator";
20
- import * as i11 from "@angular/material/sort";
21
- import * as i12 from "@angular/material/form-field";
22
- import * as i13 from "@angular/material/input";
23
- import * as i14 from "@angular/material/select";
24
- import * as i15 from "@angular/material/core";
25
- import * as i16 from "@angular/material/tooltip";
26
- import * as i17 from "@angular/material/progress-spinner";
27
- import * as i18 from "@angular/material/icon";
28
- import * as i19 from "../table-tabs/table-tabs.component";
29
- import * as i20 from "../table-tooltip/table-tooltip.component";
30
- export class TableComponent {
31
- // CONSTRUCTOR
32
- constructor(router, tableService, filterService, tooltipService, paginationService, firestore, el, cdr) {
33
- this.router = router;
34
- this.tableService = tableService;
35
- this.filterService = filterService;
36
- this.tooltipService = tooltipService;
37
- this.paginationService = paginationService;
38
- this.firestore = firestore;
39
- this.el = el;
40
- this.cdr = cdr;
41
- this.arrange = null;
42
- this.currentPageNumber = 1;
43
- this.currentClientPageIndex = 0;
44
- this.isLoading$ = new BehaviorSubject(false);
45
- this.items = [];
46
- this.filteredItems = [];
47
- this.lastDoc = null;
48
- this.firstDoc = null;
49
- this.sortBy = {
50
- field: TABLE_DEFAULTS.SORT_FIELD,
51
- order: TABLE_DEFAULTS.SORT_ORDER,
52
- };
53
- this.columnProperties = [];
54
- this.selectSort = new FormControl('');
55
- this.currentArrange = '';
56
- this.hasNextPage = false;
57
- this.dropdownItems = [];
58
- this.sortableDropdownItems = [];
59
- this.pageSize = TABLE_DEFAULTS.PAGE_SIZE;
60
- this.totalItems = 0;
61
- this.filterValue = null;
62
- this.hasFilterableColumn = false;
63
- this.hasSortableColumn = false;
64
- this.filterSubject = new Subject();
65
- this.debounceTimeMs = TABLE_DEFAULTS.DEBOUNCE_TIME_MS;
66
- this.selectedTab = 0;
67
- // Controle de loading de imagens
68
- this.imageLoadingState = new Map();
69
- this.destroy$ = new Subject();
70
- this.filtersForm = new FormArray([this.filterService.createFilterGroup()]);
71
- this.tooltipState = this.tooltipService.createInitialState();
72
- }
73
- // FILTER METHODS (delegados ao FilterService)
74
- createFilterGroup() {
75
- return this.filterService.createFilterGroup();
76
- }
77
- addFilter(filterData) {
78
- this.filterService.addFilter(this.filtersForm, this.data.pagination === true, filterData);
79
- }
80
- hasActiveDateFilter() {
81
- return this.filterService.hasActiveDateFilter(this.filtersForm);
82
- }
83
- getAvailableFilterOptions() {
84
- return this.filterService.getAvailableFilterOptions(this.dropdownItems, this.data.pagination === true, this.filtersForm);
85
- }
86
- onSelectFilterChange() {
87
- const lastIndex = this.filtersForm.length - 1;
88
- const lastFilter = this.filtersForm.at(lastIndex);
89
- const selectedFilter = lastFilter.get('selectFilter')?.value;
90
- if (selectedFilter) {
91
- if (selectedFilter.arrange === 'filterByDate' &&
92
- this.data.pagination === true) {
93
- return;
94
- }
95
- this.addFilter();
96
- }
97
- }
98
- removeFilter(index) {
99
- this.filterService.removeFilter(this.filtersForm, index);
100
- }
101
- removeAllFilters() {
102
- this.filterService.removeAllFilters(this.filtersForm);
103
- this.resetFilter();
104
- }
105
- onDateInput(event, controlName, filterIndex) {
106
- const filterGroup = this.filtersForm.at(filterIndex);
107
- this.filterService.applyDateMask(event, controlName, filterGroup);
108
- }
109
- onDateFilterChange() {
110
- if (this.data.pagination === false) {
111
- this.arrange = this.filterService.buildArrangeFromFilters(this.filtersForm, this.sortBy);
112
- this.applyFiltersToDataSource();
113
- this.cdr.markForCheck();
114
- }
115
- }
116
- applyFiltersToDataSource() {
117
- if (!this.dataSource)
118
- return;
119
- let filteredItems = this.filterService.applyClientSideFilters([...this.items], this.filtersForm, this.sortBy);
120
- this.dataSource.data = filteredItems;
121
- this.filteredItems = filteredItems;
122
- this.totalItems = filteredItems.length;
123
- this.cdr.markForCheck();
124
- }
125
- applyClientSideFilters(items) {
126
- return this.filterService.applyClientSideFilters(items, this.filtersForm, this.sortBy);
127
- }
128
- buildArrangeFromFilters() {
129
- return this.filterService.buildArrangeFromFilters(this.filtersForm, this.sortBy);
130
- }
131
- // TOOLTIP METHODS (delegados ao TooltipService)
132
- getTooltipStyles() {
133
- return this.tooltipService.getTooltipStyles(this.tooltipState);
134
- }
135
- onCellMouseEnter(event, row, col) {
136
- this.tooltipService.handleMouseEnter(event, row, col, (c, r, w) => this.getDisplayValue(c, r, w), (state) => {
137
- this.tooltipState = { ...this.tooltipState, ...state };
138
- this.cdr.markForCheck();
139
- });
140
- }
141
- onCellMouseLeave() {
142
- this.tooltipService.handleMouseLeave((state) => {
143
- this.tooltipState = state;
144
- this.cdr.markForCheck();
145
- });
146
- }
147
- onCellMouseMove(event) {
148
- this.tooltipService.handleMouseMove(event, this.tooltipState.showTooltip, (position) => {
149
- this.tooltipState = { ...this.tooltipState, tooltipPosition: position };
150
- this.cdr.markForCheck();
151
- });
152
- }
153
- // TAB METHODS
154
- onTabSelected(event) {
155
- if (!this.data.tabs?.method)
156
- return;
157
- this.selectedTab = event.index;
158
- this.data.tabs.method(event.tab, event.index);
159
- this.cdr.markForCheck();
160
- }
161
- // UTILITY METHODS (delegados aos utils)
162
- getDisplayValue(col, row, withinLimit = false) {
163
- return getDisplayValue(col, row, withinLimit);
164
- }
165
- getNestedValue(obj, path) {
166
- return getNestedValue(obj, path);
167
- }
168
- isString(value) {
169
- return isString(value);
170
- }
171
- trackByProperty(_index, col) {
172
- return trackByProperty(_index, col);
173
- }
174
- // IMAGE LOADING METHODS
175
- getImageKey(row, col) {
176
- return getImageKey(row, col);
177
- }
178
- isImageLoading(row, col) {
179
- const key = this.getImageKey(row, col);
180
- return !this.imageLoadingState.has(key);
181
- }
182
- onImageLoad(row, col) {
183
- const key = this.getImageKey(row, col);
184
- this.imageLoadingState.set(key, true);
185
- this.cdr.markForCheck();
186
- }
187
- onImageError(event, row, col, defaultImage) {
188
- const key = this.getImageKey(row, col);
189
- this.imageLoadingState.set(key, true);
190
- const target = event.target;
191
- if (target) {
192
- if (defaultImage) {
193
- target.src = defaultImage;
194
- target.onerror = null;
195
- }
196
- else {
197
- target.style.display = 'none';
198
- }
199
- }
200
- this.cdr.markForCheck();
201
- }
202
- // LIFECYCLE METHODS
203
- ngOnChanges(changes) {
204
- if (changes['data']) {
205
- setTimeout(() => this.applyCustomColors(), 0);
206
- }
207
- }
208
- ngAfterViewInit() {
209
- this.applyCustomColors();
210
- }
211
- ngOnDestroy() {
212
- this.tooltipService.destroy();
213
- this.destroy$.next();
214
- this.destroy$.complete();
215
- }
216
- applyCustomColors() {
217
- const hostElement = this.el?.nativeElement;
218
- if (!hostElement || !this.data)
219
- return;
220
- if (this.data.color?.bg) {
221
- hostElement.style.setProperty('--table-color', this.data.color.bg);
222
- hostElement.style.setProperty('--table-header-bg', this.data.color.bg);
223
- }
224
- if (this.data.color?.text) {
225
- hostElement.style.setProperty('--table-text-color', this.data.color.text);
226
- hostElement.style.setProperty('--table-header-text', this.data.color.text);
227
- }
228
- setTimeout(() => {
229
- const headerCells = hostElement.querySelectorAll('.mat-header-cell, .mat-mdc-header-cell, th.mat-header-cell, th.mat-mdc-header-cell');
230
- headerCells.forEach((cell) => {
231
- if (this.data.color?.bg) {
232
- cell.style.backgroundColor = this.data.color.bg;
233
- }
234
- if (this.data.color?.text) {
235
- cell.style.color = this.data.color.text;
236
- }
237
- });
238
- }, 100);
239
- }
240
- async ngOnInit() {
241
- if (!this.data.color) {
242
- this.data.color = {
243
- bg: TABLE_DEFAULTS.DEFAULT_BG_COLOR,
244
- text: TABLE_DEFAULTS.DEFAULT_TEXT_COLOR,
245
- };
246
- }
247
- if (this.data.showSimpleSearch === undefined) {
248
- this.data.showSimpleSearch = true;
249
- }
250
- this.columnProperties = this.data.displayedColumns.map((column) => column.property);
251
- this.cdr.markForCheck();
252
- if (this.data.actionButton && !this.data.actionButton.condition) {
253
- this.data.actionButton.condition = (_row) => true;
254
- }
255
- // Inicializar dropdownItems usando FilterService
256
- const filterConfig = this.filterService.initializeDropdownItems(this.data.displayedColumns);
257
- this.dropdownItems = filterConfig.dropdownItems;
258
- this.sortableDropdownItems = filterConfig.sortableDropdownItems;
259
- this.hasFilterableColumn = filterConfig.hasFilterableColumn;
260
- this.hasSortableColumn = filterConfig.hasSortableColumn;
261
- if (this.data.filterableOptions) {
262
- this.dropdownItems = this.filterService.addFilterableOptions(this.dropdownItems, this.data.filterableOptions);
263
- }
264
- // Sem paginação
265
- if (this.data.pagination === false) {
266
- await this.loadItems();
267
- this.cdr.markForCheck();
268
- }
269
- // Com paginação
270
- if (this.data.pagination === true) {
271
- if (this.data.sortBy) {
272
- this.sortBy = {
273
- field: this.data.sortBy.field,
274
- order: this.data.sortBy.order,
275
- };
276
- }
277
- this.filterSubject
278
- .pipe(debounceTime(this.debounceTimeMs), takeUntil(this.destroy$))
279
- .subscribe(() => {
280
- this.loadItemsPaginated('reload', true);
281
- });
282
- this.isLoading$.next(true);
283
- this.cdr.markForCheck();
284
- await this.loadItemsPaginated('reload', true);
285
- this.sort.active = 'createdAt';
286
- this.sort.direction = 'desc';
287
- this.dataSource.paginator = this.paginator;
288
- this.dataSource.sort = this.sort;
289
- this.totalItems = 0;
290
- if (this.data.totalRef) {
291
- for (const totalRef of this.data.totalRef) {
292
- const totalRefDoc = await totalRef.ref.get();
293
- const docData = totalRefDoc.data();
294
- if (docData && docData[totalRef.field]) {
295
- const fieldValue = docData[totalRef.field];
296
- this.totalItems = (this.totalItems +
297
- (typeof fieldValue === 'number' ? fieldValue : 0));
298
- }
299
- }
300
- }
301
- this.isLoading$.next(false);
302
- this.cdr.markForCheck();
303
- }
304
- }
305
- // DATA LOADING METHODS
306
- async loadItems() {
307
- this.imageLoadingState.clear();
308
- this.items = (await this.tableService.getItems(this.data.collectionRef));
309
- if (this.data.conditions) {
310
- this.filterItems();
311
- }
312
- await this.loadRelations();
313
- await this.loadQueryLengths();
314
- this.totalItems = this.items.length;
315
- this.arrange = {
316
- filters: [],
317
- sortBy: this.data.sortBy || {
318
- field: TABLE_DEFAULTS.SORT_FIELD,
319
- order: TABLE_DEFAULTS.SORT_ORDER,
320
- },
321
- };
322
- this.sortBy = this.data.sortBy || {
323
- field: TABLE_DEFAULTS.SORT_FIELD,
324
- order: TABLE_DEFAULTS.SORT_ORDER,
325
- };
326
- let itemsToDisplay = [...this.items];
327
- itemsToDisplay = this.applyClientSideFilters(itemsToDisplay);
328
- this.filteredItems = itemsToDisplay;
329
- this.dataSource = new MatTableDataSource(itemsToDisplay);
330
- this.dataSource.paginator = this.paginator;
331
- this.dataSource.sort = this.sort;
332
- if (this.data.sortBy) {
333
- this.dataSource.sort.active = this.data.sortBy.field;
334
- this.dataSource.sort.direction = this.data.sortBy.order;
335
- this.dataSource.sort.sortChange.emit();
336
- }
337
- this.dataSource.filterPredicate = (data, filter) => {
338
- const item = data;
339
- return this.data.displayedColumns.some((col) => {
340
- if (col.filterPredicates) {
341
- return col.filterPredicates.some((predicate) => {
342
- const propertyValue = item[col.property];
343
- if (!propertyValue || typeof propertyValue !== 'object') {
344
- return false;
345
- }
346
- const propertyRecord = propertyValue;
347
- const predicateValue = propertyRecord[predicate];
348
- if (predicateValue === null || predicateValue === undefined) {
349
- return false;
350
- }
351
- return String(predicateValue)
352
- .trim()
353
- .toLocaleLowerCase()
354
- .includes(filter);
355
- });
356
- }
357
- if (col.property && col.isFilterable) {
358
- const propertyValue = item[col.property];
359
- if (propertyValue === null || propertyValue === undefined) {
360
- return false;
361
- }
362
- return String(propertyValue)
363
- .trim()
364
- .toLocaleLowerCase()
365
- .includes(filter);
366
- }
367
- return false;
368
- });
369
- };
370
- this.filterPredicate = this.dataSource.filterPredicate;
371
- this.cdr.markForCheck();
372
- }
373
- async loadItemsPaginated(navigation = 'reload', reset = false) {
374
- this.imageLoadingState.clear();
375
- if (reset && ['forward', 'reload'].includes(navigation)) {
376
- this.lastDoc = null;
377
- this.currentClientPageIndex = 0;
378
- }
379
- if (reset && ['backward', 'reload'].includes(navigation)) {
380
- this.firstDoc = null;
381
- this.currentClientPageIndex = 0;
382
- }
383
- const activeFilters = this.filterService.extractActiveFilters(this.filtersForm);
384
- this.arrange = {
385
- filters: activeFilters,
386
- sortBy: this.sortBy,
387
- };
388
- const paginated = {
389
- batchSize: this.pageSize,
390
- collection: this.data.collection,
391
- doc: { lastDoc: this.lastDoc, firstDoc: this.firstDoc },
392
- navigation,
393
- arrange: this.arrange,
394
- conditions: this.data.conditions,
395
- size: this.totalItems,
396
- filterFn: this.data.filterFn,
397
- clientPageIndex: this.currentClientPageIndex,
398
- };
399
- const result = await this.tableService.getPaginated(paginated);
400
- this.items = result.items;
401
- await this.loadRelations();
402
- await this.loadQueryLengths();
403
- this.lastDoc = result.lastDoc;
404
- this.firstDoc = result.firstDoc;
405
- if (result.currentClientPageIndex !== undefined) {
406
- this.currentClientPageIndex = result.currentClientPageIndex;
407
- }
408
- let sum = 0;
409
- if (this.data.totalRef) {
410
- for (const totalRef of this.data.totalRef) {
411
- const totalRefDoc = await totalRef.ref.get();
412
- const docData = totalRefDoc.data();
413
- if (docData || result.filterLength) {
414
- const fieldValue = docData ? docData[totalRef.field] : undefined;
415
- const numericValue = typeof fieldValue === 'number' ? fieldValue : 0;
416
- sum = result.filterLength ?? (sum + numericValue);
417
- }
418
- }
419
- this.totalItems = sum;
420
- }
421
- this.hasNextPage = result.hasNextPage;
422
- this.dataSource = new MatTableDataSource(this.items);
423
- this.filterPredicate = this.dataSource.filterPredicate;
424
- this.cdr.markForCheck();
425
- }
426
- // PAGINATION METHODS
427
- async onPageChange(event) {
428
- if (this.data.pagination !== true || !event)
429
- return;
430
- this.isLoading$.next(true);
431
- const currentState = {
432
- currentPageNumber: this.currentPageNumber,
433
- currentClientPageIndex: this.currentClientPageIndex,
434
- pageSize: this.pageSize,
435
- totalItems: this.totalItems,
436
- hasNextPage: this.hasNextPage,
437
- };
438
- const navigation = this.paginationService.calculateNavigation(event, currentState);
439
- if (!navigation.shouldNavigate) {
440
- this.isLoading$.next(false);
441
- this.pageEvent = event;
442
- return;
443
- }
444
- // Atualizar estado local
445
- this.currentClientPageIndex = navigation.newClientPageIndex;
446
- this.currentPageNumber = navigation.newPageNumber;
447
- // Atualizar pageSize se mudou
448
- if (event.pageSize !== this.pageSize) {
449
- this.pageSize = event.pageSize;
450
- }
451
- // Ajustar pageSize temporariamente para última página se necessário
452
- const lastPageIndex = this.paginationService.calculateLastPageIndex(this.totalItems, event.pageSize);
453
- let originalPageSize = null;
454
- if (event.pageIndex === lastPageIndex &&
455
- navigation.direction === 'backward' &&
456
- navigation.resetDocs) {
457
- const itemsExpectedInLastPage = this.paginationService.calculateLastPageItemCount(this.totalItems, event.pageSize);
458
- if (itemsExpectedInLastPage > 0 &&
459
- itemsExpectedInLastPage < event.pageSize) {
460
- originalPageSize = this.pageSize;
461
- this.pageSize = itemsExpectedInLastPage;
462
- }
463
- }
464
- try {
465
- await this.loadItemsPaginated(navigation.direction, navigation.resetDocs);
466
- }
467
- catch (error) {
468
- console.error('Erro ao carregar itens paginados:', error);
469
- }
470
- finally {
471
- if (originalPageSize !== null) {
472
- this.pageSize = originalPageSize;
473
- }
474
- }
475
- this.pageEvent = event;
476
- this.isLoading$.next(false);
477
- this.cdr.markForCheck();
478
- }
479
- // FILTER/SEARCH METHODS
480
- applyFilter(value) {
481
- if (this.data.pagination === false) {
482
- this.dataSource.filter = String(value).trim().toLowerCase();
483
- }
484
- if (this.data.pagination === true) {
485
- this.filterValue = value;
486
- this.filterSubject.next(this.filterValue);
487
- }
488
- }
489
- async search(event) {
490
- if (event) {
491
- event.preventDefault();
492
- event.stopPropagation();
493
- }
494
- if (this.selectSort.value) {
495
- if (this.selectSort.value.arrange === 'ascending') {
496
- this.sortBy = {
497
- field: this.selectSort.value.property,
498
- order: 'asc',
499
- };
500
- }
501
- if (this.selectSort.value.arrange === 'descending') {
502
- this.sortBy = {
503
- field: this.selectSort.value.property,
504
- order: 'desc',
505
- };
506
- }
507
- }
508
- if (this.data.pagination === false) {
509
- this.arrange = this.buildArrangeFromFilters();
510
- this.applyFiltersToDataSource();
511
- this.currentArrange =
512
- this.filtersForm.length > 0
513
- ? (this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? '')
514
- : '';
515
- this.cdr.markForCheck();
516
- }
517
- else {
518
- await this.loadItemsPaginated('reload', true);
519
- this.currentArrange =
520
- this.filtersForm.length > 0
521
- ? (this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? '')
522
- : '';
523
- this.paginator.firstPage();
524
- this.cdr.markForCheck();
525
- }
526
- }
527
- async resetFilter() {
528
- this.dataSource.filter = '';
529
- if (this.filterPredicate) {
530
- this.dataSource.filterPredicate = this.filterPredicate;
531
- }
532
- this.filtersForm.clear();
533
- this.addFilter();
534
- this.selectSort.patchValue('');
535
- this.sortBy = {
536
- order: this.data.sortBy ? this.data.sortBy.order : 'desc',
537
- field: this.data.sortBy ? this.data.sortBy.field : 'createdAt',
538
- };
539
- if (this.data.pagination === false) {
540
- this.arrange = {
541
- filters: [],
542
- sortBy: this.sortBy,
543
- };
544
- this.dataSource.data = [...this.items];
545
- this.totalItems = this.items.length;
546
- this.currentArrange = '';
547
- this.cdr.markForCheck();
548
- }
549
- else {
550
- await this.loadItemsPaginated('reload', true);
551
- this.currentArrange = '';
552
- this.paginator.firstPage();
553
- this.cdr.markForCheck();
554
- }
555
- }
556
- // NAVIGATION METHODS
557
- goToDetails(row) {
558
- if (this.data.isNotClickable) {
559
- return;
560
- }
561
- const urlPath = this.data.url || this.data.name;
562
- const rowWithId = row;
563
- const url = this.router.serializeUrl(this.router.createUrlTree([`/${urlPath}`, rowWithId.id]));
564
- window.open(url, '_blank');
565
- }
566
- // RELATION METHODS
567
- async getRelation(params) {
568
- try {
569
- let snapshot;
570
- if (params.id !== '' &&
571
- params.id !== undefined &&
572
- params.collection !== undefined &&
573
- params.collection !== '') {
574
- snapshot = await firstValueFrom(this.firestore.collection(params.collection).doc(params.id).get());
575
- }
576
- if (snapshot && snapshot.exists) {
577
- const data = snapshot.data();
578
- const value = data?.[params.newProperty];
579
- return typeof value === 'string' ? value : '';
580
- }
581
- return '';
582
- }
583
- catch (e) {
584
- console.error(e);
585
- return '';
586
- }
587
- }
588
- async loadRelations() {
589
- const relationPromises = this.data.displayedColumns
590
- .filter((col) => col.relation)
591
- .flatMap((col) => this.items.map(async (item) => {
592
- if (col.relation) {
593
- const itemRecord = item;
594
- itemRecord[col.property] = await this.getRelation({
595
- id: String(itemRecord[col.relation.sourceProperty] ?? ''),
596
- collection: col.relation.targetedCollection,
597
- newProperty: col.relation.targetedProperty || col.property,
598
- });
599
- }
600
- }));
601
- await Promise.all(relationPromises);
602
- }
603
- async getQueryLength(params) {
604
- const itemRecord = params.item;
605
- const snapshot = await this.firestore
606
- .collection(params.relation.collection)
607
- .ref.where(params.relation.property, params.relation.operator, itemRecord[params.relation.value])
608
- .get();
609
- return snapshot.size;
610
- }
611
- async loadQueryLengths() {
612
- const lengthPromises = this.data.displayedColumns
613
- .filter((col) => col.queryLength)
614
- .flatMap((col) => this.items.map(async (item) => {
615
- if (col.queryLength) {
616
- const itemRecord = item;
617
- itemRecord[col.property] = await this.getQueryLength({
618
- item: item,
619
- relation: col.queryLength,
620
- });
621
- }
622
- }));
623
- await Promise.all(lengthPromises);
624
- }
625
- filterItems() {
626
- if (this.data.conditions) {
627
- this.data.conditions.forEach((cond) => {
628
- this.items = this.items.filter((item) => {
629
- const operatorFunction = this.tableService.operators[cond.operator];
630
- if (operatorFunction) {
631
- const itemRecord = item;
632
- return operatorFunction(itemRecord[cond.firestoreProperty], cond.dashProperty);
633
- }
634
- return false;
635
- });
636
- });
637
- }
638
- }
639
- // PUBLIC METHODS
640
- async reloadTable() {
641
- if (this.data.pagination) {
642
- await this.loadItemsPaginated('reload', true);
643
- this.paginator.firstPage();
644
- this.cdr.markForCheck();
645
- }
646
- else {
647
- await this.loadItems();
648
- }
649
- }
650
- updateDisplayedColumns() {
651
- if (this.dataSource) {
652
- this.dataSource = new MatTableDataSource([]);
653
- }
654
- this.columnProperties = this.data.displayedColumns.map((column) => column.property);
655
- const filterConfig = this.filterService.initializeDropdownItems(this.data.displayedColumns);
656
- this.dropdownItems = filterConfig.dropdownItems;
657
- this.sortableDropdownItems = filterConfig.sortableDropdownItems;
658
- if (this.data.filterableOptions) {
659
- this.dropdownItems = this.filterService.addFilterableOptions(this.dropdownItems, this.data.filterableOptions);
660
- }
661
- this.cdr.markForCheck();
662
- }
663
- shouldShowActionButton() {
664
- if (!this.data?.actionButton) {
665
- return false;
666
- }
667
- if (!this.data.actionButton.condition) {
668
- return true;
669
- }
670
- try {
671
- return this.data.actionButton.condition(null) ?? true;
672
- }
673
- catch {
674
- return true;
675
- }
676
- }
677
- handleDownload() {
678
- if (!this.downloadTable)
679
- return;
680
- if (this.data.pagination === false) {
681
- if (this.dataSource && this.dataSource.filteredData) {
682
- this.filteredItems = [...this.dataSource.filteredData];
683
- }
684
- const arrange = this.buildArrangeFromFilters();
685
- this.downloadTable(arrange, this.data.conditions || []);
686
- }
687
- else {
688
- if (this.arrange) {
689
- this.downloadTable(this.arrange, this.data.conditions || []);
690
- }
691
- }
692
- }
693
- }
694
- TableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableComponent, deps: [{ token: i1.Router }, { token: i2.TableService }, { token: i3.FilterService }, { token: i4.TooltipService }, { token: i5.PaginationService }, { token: i6.AngularFirestore }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
695
- TableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: TableComponent, selector: "lib-table", inputs: { data: "data", downloadTable: "downloadTable" }, viewQueries: [{ propertyName: "paginator", first: true, predicate: MatPaginator, descendants: true }, { propertyName: "sort", first: true, predicate: MatSort, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"data\" class=\"card-body\">\r\n <div class=\"flex flex-col justify-between gap-6\">\r\n <!-- UNIFIED CONTROL PANEL: FILTERS, SORT & ACTIONS -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"\r\n data.pagination === true &&\r\n (dropdownItems.length > 0 ||\r\n sortableDropdownItems.length > 0 ||\r\n data.actionButton)\r\n \"\r\n >\r\n <!-- PANEL HEADER: Title and Custom Action -->\r\n <div\r\n class=\"mb-4 flex flex-col items-start justify-between gap-4 border-b-2 border-gray-200 pb-4 md:flex-row md:items-center\"\r\n >\r\n <!-- Left Side: Title & Main Action Button -->\r\n <div class=\"flex flex-wrap items-center gap-4\">\r\n <div class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-filter text-xl text-blue-500\"></i>\r\n <span class=\"text-lg font-semibold text-gray-700\"\r\n >Filtros e A\u00E7\u00F5es</span\r\n >\r\n </div>\r\n <button\r\n *ngIf=\"data.actionButton && data.actionButton.condition\"\r\n [ngClass]=\"\r\n (data.actionButton.colorClass || 'bg-blue-500') +\r\n ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\r\n \"\r\n [routerLink]=\"data.actionButton.routerLink\"\r\n (click)=\"\r\n data.actionButton.method ? data.actionButton.method($event) : null\r\n \"\r\n >\r\n <i\r\n *ngIf=\"data.actionButton.icon\"\r\n [class]=\"data.actionButton.icon\"\r\n ></i>\r\n {{ data.actionButton.label }}\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <!-- SORT FIELD (Integrated with filters) -->\r\n <div class=\"mb-4\" *ngIf=\"sortableDropdownItems.length > 0\">\r\n <div class=\"flex flex-wrap items-center gap-3 p-2\">\r\n <div class=\"min-w-[200px] flex-1\">\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\r\n <span>Ordenar por</span>\r\n </mat-label>\r\n <mat-select placeholder=\"Selecione...\" [formControl]=\"selectSort\">\r\n <mat-option\r\n *ngFor=\"let item of sortableDropdownItems\"\r\n [value]=\"item\"\r\n >\r\n <div class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\r\n <span>{{ item.title }}</span>\r\n </div>\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- FILTERS CONTENT (WITH REFINEMENTS) -->\r\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\r\n <div\r\n [formGroup]=\"$any(filterGroup)\"\r\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\r\n class=\"flex flex-wrap items-center gap-3 p-2\"\r\n >\r\n <!-- FILTER TYPE SELECTOR -->\r\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>Tipo de filtro</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione o tipo...\"\r\n formControlName=\"selectFilter\"\r\n (selectionChange)=\"onSelectFilterChange()\"\r\n >\r\n <mat-option\r\n *ngFor=\"let item of getAvailableFilterOptions()\"\r\n [value]=\"item\"\r\n >\r\n <div class=\"flex items-center gap-2\">\r\n <i\r\n [class]=\"item.icon || 'fa fa-filter'\"\r\n class=\"text-sm text-blue-500\"\r\n ></i>\r\n <span>{{ item.title }}</span>\r\n </div>\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- TEXT FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-search text-gray-400\"></i>\r\n <span>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Filtrar\"\r\n }}</span>\r\n </mat-label>\r\n <input\r\n (keyup.enter)=\"search($event)\"\r\n formControlName=\"typeFilter\"\r\n matInput\r\n placeholder=\"Digite para filtrar...\"\r\n #input\r\n />\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DROPDOWN FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value &&\r\n $any(filterGroup)\r\n .get('selectFilter')\r\n ?.value.hasOwnProperty('items')\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Selecione\"\r\n }}</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione...\"\r\n formControlName=\"selectItem\"\r\n multiple\r\n >\r\n <mat-option\r\n *ngFor=\"\r\n let item of $any(filterGroup).get('selectFilter')?.value\r\n .items\r\n \"\r\n [value]=\"item\"\r\n >\r\n {{ item.label }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DATE FILTER -->\r\n <div\r\n class=\"min-w-[340px] flex-auto\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n 'filterByDate'\r\n \"\r\n >\r\n <div\r\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Inicial</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n matInput\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'initialDate', i)\"\r\n formControlName=\"initialDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Final</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'finalDate', i)\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n\r\n <!-- REMOVE FILTER BUTTON -->\r\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\r\n <button\r\n (click)=\"removeFilter(i)\"\r\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\r\n matTooltip=\"Remover filtro\"\r\n >\r\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- PANEL FOOTER: Add Filter & Action Buttons -->\r\n <div\r\n class=\"-mb-2 flex items-start justify-between gap-4 border-t border-gray-200 pt-4\"\r\n >\r\n <!-- Add Filter Button -->\r\n <div *ngIf=\"dropdownItems.length > 0\">\r\n <button\r\n (click)=\"addFilter()\"\r\n class=\"transform rounded-full border-2 border-blue-300 bg-blue-50 px-6 py-2 text-sm font-medium text-blue-600 transition-all duration-300 hover:-translate-y-0.5 hover:border-blue-400 hover:bg-blue-100 hover:shadow-md\"\r\n matTooltip=\"Adicionar novo filtro\"\r\n >\r\n <i class=\"fa fa-plus mr-2\"></i>\r\n Adicionar Filtro\r\n </button>\r\n </div>\r\n\r\n <!-- Right Side: Search, Reset, Export -->\r\n <div\r\n class=\"flex flex-wrap gap-3 self-start sm:self-auto\"\r\n *ngIf=\"\r\n this.hasFilterableColumn === true || this.hasSortableColumn === true\r\n \"\r\n >\r\n <button\r\n (click)=\"search()\"\r\n type=\"button\"\r\n class=\"flex items-center gap-2 rounded-lg bg-green-600 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-green-700\"\r\n matTooltip=\"Aplicar filtros\"\r\n >\r\n <i class=\"fa fa-search\"></i>\r\n Pesquisar\r\n </button>\r\n\r\n <button\r\n (click)=\"resetFilter()\"\r\n class=\"flex items-center gap-2 rounded-lg bg-red-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-red-600\"\r\n matTooltip=\"Limpar filtros\"\r\n >\r\n <i class=\"fas fa-redo-alt\"></i>\r\n Resetar\r\n </button>\r\n\r\n <button\r\n *ngIf=\"data.download !== false && downloadTable\"\r\n class=\"flex items-center gap-2 rounded-lg bg-orange-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-orange-600\"\r\n matTooltipPosition=\"above\"\r\n matTooltip=\"Exportar Tabela\"\r\n [disabled]=\"\r\n this.dataSource && this.dataSource.filteredData.length <= 0\r\n \"\r\n (click)=\"\r\n $any(arrange) && downloadTable !== undefined\r\n ? downloadTable($any(arrange), data.conditions || [])\r\n : null\r\n \"\r\n >\r\n <i class=\"fa fa-download\"></i>\r\n Exportar\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- UNIFIED CONTROL PANEL (for non-paginated tables): SEARCH, SORT & FILTERS -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"\r\n data.pagination === false &&\r\n (hasFilterableColumn === true ||\r\n dropdownItems.length > 0 ||\r\n data.actionButton)\r\n \"\r\n >\r\n <!-- PANEL HEADER: Title and Custom Action -->\r\n <div\r\n class=\"mb-4 flex flex-col items-start justify-between gap-4 border-b-2 border-gray-200 pb-4 md:flex-row md:items-center\"\r\n >\r\n <!-- Left Side: Title & Main Action Button -->\r\n <div class=\"flex flex-wrap items-center gap-4\">\r\n <div class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-filter text-xl text-blue-500\"></i>\r\n <span class=\"text-lg font-semibold text-gray-700\"\r\n >Filtros e A\u00E7\u00F5es</span\r\n >\r\n </div>\r\n <button\r\n *ngIf=\"data.actionButton && data.actionButton.condition\"\r\n [ngClass]=\"\r\n (data.actionButton.colorClass || 'bg-blue-500') +\r\n ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\r\n \"\r\n [routerLink]=\"data.actionButton.routerLink\"\r\n (click)=\"\r\n data.actionButton.method ? data.actionButton.method($event) : null\r\n \"\r\n >\r\n <i\r\n *ngIf=\"data.actionButton.icon\"\r\n [class]=\"data.actionButton.icon\"\r\n ></i>\r\n {{ data.actionButton.label }}\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <!-- SIMPLE SEARCH -->\r\n <div\r\n class=\"mb-4\"\r\n *ngIf=\"hasFilterableColumn === true && data.showSimpleSearch !== false\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-search text-blue-500\"></i>\r\n Buscar\r\n </mat-label>\r\n <input\r\n matInput\r\n (keyup.enter)=\"search($event)\"\r\n (keyup)=\"applyFilter(filterInput.value)\"\r\n placeholder=\"Digite para filtrar...\"\r\n #filterInput\r\n />\r\n <mat-icon matSuffix class=\"text-gray-500\">search</mat-icon>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- FILTERS PANEL -->\r\n <div *ngIf=\"dropdownItems.length > 0\">\r\n <!-- FILTERS CONTENT -->\r\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\r\n <div\r\n [formGroup]=\"$any(filterGroup)\"\r\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\r\n class=\"flex flex-wrap items-center gap-3\"\r\n >\r\n <!-- FILTER TYPE SELECTOR -->\r\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>Tipo de filtro</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione o tipo...\"\r\n formControlName=\"selectFilter\"\r\n (selectionChange)=\"onSelectFilterChange()\"\r\n >\r\n <mat-option\r\n *ngFor=\"let item of getAvailableFilterOptions()\"\r\n [value]=\"item\"\r\n >\r\n <div class=\"flex items-center gap-2\">\r\n <i\r\n [class]=\"item.icon || 'fa fa-filter'\"\r\n class=\"text-sm text-blue-500\"\r\n ></i>\r\n <span>{{ item.title }}</span>\r\n </div>\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- TEXT FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n 'filter'\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-search text-gray-400\"></i>\r\n <span>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Filtrar\"\r\n }}</span>\r\n </mat-label>\r\n <input\r\n (keyup.enter)=\"search($event)\"\r\n formControlName=\"typeFilter\"\r\n matInput\r\n placeholder=\"Digite para filtrar...\"\r\n #input\r\n />\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DROPDOWN FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value &&\r\n $any(filterGroup)\r\n .get('selectFilter')\r\n ?.value.hasOwnProperty('items')\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Selecione\"\r\n }}</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione...\"\r\n formControlName=\"selectItem\"\r\n multiple\r\n >\r\n <mat-option\r\n *ngFor=\"\r\n let item of $any(filterGroup).get('selectFilter')?.value\r\n .items\r\n \"\r\n [value]=\"item\"\r\n >\r\n {{ item.label }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DATE FILTER -->\r\n <div\r\n class=\"min-w-[340px] flex-auto\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n 'filterByDate'\r\n \"\r\n >\r\n <div\r\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Inicial</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n matInput\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'initialDate', i)\"\r\n (blur)=\"onDateFilterChange()\"\r\n formControlName=\"initialDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Final</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'finalDate', i)\"\r\n (blur)=\"onDateFilterChange()\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n\r\n <!-- REMOVE FILTER BUTTON -->\r\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\r\n <button\r\n (click)=\"removeFilter(i)\"\r\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\r\n matTooltip=\"Remover filtro\"\r\n >\r\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- PANEL FOOTER: Add Filter & Actions -->\r\n <div\r\n class=\"-mb-2 flex items-start justify-between gap-4 border-t border-gray-200 pt-4\"\r\n >\r\n <!-- Add Filter Button -->\r\n <div *ngIf=\"dropdownItems.length > 0\">\r\n <button\r\n (click)=\"addFilter()\"\r\n class=\"transform rounded-full border-2 border-blue-300 bg-blue-50 px-6 py-2 text-sm font-medium text-blue-600 transition-all duration-300 hover:-translate-y-0.5 hover:border-blue-400 hover:bg-blue-100 hover:shadow-md\"\r\n matTooltip=\"Adicionar novo filtro\"\r\n >\r\n <i class=\"fa fa-plus mr-2\"></i>\r\n Adicionar Filtro\r\n </button>\r\n </div>\r\n\r\n <!-- Action Buttons -->\r\n <div\r\n class=\"flex flex-wrap gap-3 self-start sm:self-auto\"\r\n *ngIf=\"\r\n this.hasFilterableColumn === true ||\r\n this.hasSortableColumn === true\r\n \"\r\n >\r\n <button\r\n (click)=\"search()\"\r\n type=\"button\"\r\n class=\"flex items-center gap-2 rounded-lg bg-green-600 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-green-700\"\r\n matTooltip=\"Aplicar filtros\"\r\n >\r\n <i class=\"fa fa-search\"></i>\r\n Pesquisar\r\n </button>\r\n\r\n <button\r\n (click)=\"resetFilter()\"\r\n class=\"flex items-center gap-2 rounded-lg bg-red-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-red-600\"\r\n matTooltip=\"Limpar filtros\"\r\n >\r\n <i class=\"fas fa-redo-alt\"></i>\r\n Resetar\r\n </button>\r\n\r\n <button\r\n *ngIf=\"data.download !== false && downloadTable\"\r\n class=\"flex items-center gap-2 rounded-lg bg-orange-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-orange-600\"\r\n matTooltipPosition=\"above\"\r\n matTooltip=\"Exportar Tabela\"\r\n [disabled]=\"\r\n this.dataSource && this.dataSource.filteredData.length <= 0\r\n \"\r\n (click)=\"handleDownload()\"\r\n >\r\n <i class=\"fa fa-download\"></i>\r\n Exportar\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"flex flex-col\">\r\n <!-- TABS COMPONENT -->\r\n <lib-table-tabs\r\n *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\r\n [tabsData]=\"data.tabs.tabsData\"\r\n [selectedTab]=\"selectedTab\"\r\n (tabSelected)=\"onTabSelected($event)\"\r\n ></lib-table-tabs>\r\n <div\r\n class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl table-container\"\r\n >\r\n <table\r\n mat-table\r\n [dataSource]=\"dataSource\"\r\n matSort\r\n #sort=\"matSort\"\r\n matSortActive=\"createdAt\"\r\n matSortDirection=\"desc\"\r\n >\r\n <ng-container\r\n *ngFor=\"let col of data.displayedColumns; trackBy: trackByProperty\"\r\n matColumnDef=\"{{ col.property }}\"\r\n >\r\n <ng-container *matHeaderCellDef>\r\n <!-- IF THE COLUMN IS NOT SORTABLE, THEN DON'T SHOW THE SORT BUTTONS -->\r\n <th\r\n *ngIf=\"!col.isSortable || data.pagination === true\"\r\n mat-header-cell\r\n >\r\n {{ col.title }}\r\n </th>\r\n <!-- IF THE COLUMN IS SORTABLE, THEN SHOW THE SORT BUTTONS -->\r\n <th\r\n *ngIf=\"col.isSortable && data.pagination === false\"\r\n mat-header-cell\r\n mat-sort-header\r\n >\r\n {{ col.title }}\r\n </th>\r\n <td\r\n mat-cell\r\n *matCellDef=\"let row\"\r\n (click)=\"col.method ? col.method(row) : null\"\r\n (mouseenter)=\"onCellMouseEnter($event, row, col)\"\r\n (mouseleave)=\"onCellMouseLeave()\"\r\n (mousemove)=\"onCellMouseMove($event)\"\r\n >\r\n <!-- CHECK IF THE COLUMN MUST BE DISPLAYED -->\r\n <span *ngIf=\"!col.image && !col.iconClass && !col.method\">\r\n <ng-container>\r\n <span\r\n *ngIf=\"\r\n col.charLimit &&\r\n row[col.property] &&\r\n row[col.property].length > col.charLimit;\r\n else withinLimit\r\n \"\r\n >\r\n <a\r\n *ngIf=\"col.hasLink === true\"\r\n [href]=\"row[col.property]\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row) }}\r\n </a>\r\n <a\r\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\r\n [href]=\"col.hasLink\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row) }}\r\n </a>\r\n <span\r\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\r\n >\r\n {{ getDisplayValue(col, row) }}\r\n </span>\r\n </span>\r\n </ng-container>\r\n <ng-template #withinLimit>\r\n <a\r\n *ngIf=\"col.hasLink === true\"\r\n [href]=\"row[col.property]\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row, true) }}\r\n </a>\r\n <a\r\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\r\n [href]=\"col.hasLink\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row, true) }}\r\n </a>\r\n <span\r\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\r\n >\r\n {{ getDisplayValue(col, row, true) }}\r\n </span>\r\n </ng-template>\r\n </span>\r\n <!------------------- IMAGE ------------------>\r\n <!-- Imagem com loading -->\r\n <div\r\n *ngIf=\"\r\n col.image && col.image.path && !col.iconClass && !col.method\r\n \"\r\n class=\"relative inline-block\"\r\n >\r\n <mat-spinner\r\n *ngIf=\"isImageLoading(row, col)\"\r\n [diameter]=\"32\"\r\n class=\"absolute inset-0 m-auto\"\r\n ></mat-spinner>\r\n <img\r\n [src]=\"col.image.path + '/' + row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n [class.opacity-0]=\"isImageLoading(row, col)\"\r\n alt=\"\"\r\n (load)=\"onImageLoad(row, col)\"\r\n (error)=\"onImageError($event, row, col, col.image.default)\"\r\n />\r\n </div>\r\n\r\n <!-- Imagem com URL completa e loading -->\r\n <div\r\n *ngIf=\"\r\n col.image && col.image.url && !col.iconClass && !col.method\r\n \"\r\n class=\"relative inline-block\"\r\n >\r\n <mat-spinner\r\n *ngIf=\"isImageLoading(row, col)\"\r\n [diameter]=\"32\"\r\n class=\"absolute inset-0 m-auto\"\r\n ></mat-spinner>\r\n <img\r\n [src]=\"row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n [class.opacity-0]=\"isImageLoading(row, col)\"\r\n alt=\"\"\r\n (load)=\"onImageLoad(row, col)\"\r\n (error)=\"onImageError($event, row, col, col.image.default)\"\r\n />\r\n </div>\r\n <ng-container *ngIf=\"col.iconClass\">\r\n <button\r\n *ngFor=\"let iconClass of col.iconClass\"\r\n (click)=\"\r\n iconClass.buttonMethod\r\n ? iconClass.buttonMethod(row, $event)\r\n : $event.stopPropagation()\r\n \"\r\n >\r\n <span\r\n [ngClass]=\"iconClass.class\"\r\n *ngIf=\"\r\n iconClass.condition === undefined ||\r\n (iconClass.condition !== undefined &&\r\n $any(iconClass.condition)(row))\r\n \"\r\n >{{ iconClass.text }}</span\r\n >\r\n </button>\r\n </ng-container>\r\n </td>\r\n </ng-container>\r\n </ng-container>\r\n\r\n <tr mat-header-row *matHeaderRowDef=\"columnProperties\"></tr>\r\n <tr\r\n [ngClass]=\"{\r\n 'example-element-row': data.isNotClickable === true,\r\n 'example-element-row cursor-pointer': !data.isNotClickable,\r\n }\"\r\n mat-row\r\n *matRowDef=\"let row; columns: columnProperties\"\r\n (click)=\"goToDetails(row)\"\r\n ></tr>\r\n\r\n <!-- ROW SHOWN WHEN THERE IS NO MATCHING DATA. -->\r\n <tr class=\"mat-row\" *matNoDataRow>\r\n <td *ngIf=\"!(isLoading$ | async)\" class=\"mat-cell p-4\" colspan=\"4\">\r\n Nenhum resultado encontrado para a busca\r\n </td>\r\n </tr>\r\n </table>\r\n\r\n <div class=\"flex justify-center\" *ngIf=\"isLoading$ | async\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n\r\n <div class=\"paginator-container\">\r\n <mat-paginator\r\n #paginator\r\n [pageSizeOptions]=\"[25, 50, 100]\"\r\n [pageSize]=\"pageSize\"\r\n [length]=\"totalItems\"\r\n showFirstLastButtons\r\n aria-label=\"Select page of periodic elements\"\r\n (page)=\"onPageChange($event)\"\r\n [ngClass]=\"{\r\n 'hide-length':\r\n ['filter', 'filterByDate', 'equals'].includes(\r\n this.currentArrange\r\n ) || this.data.filterFn,\r\n 'hide-next-button': !hasNextPage && data.pagination === true,\r\n 'hide-last-button':\r\n (!hasNextPage && data.pagination === true) ||\r\n this.data.filterFn,\r\n }\"\r\n >\r\n </mat-paginator>\r\n <div\r\n *ngIf=\"\r\n !(isLoading$ | async) &&\r\n dataSource?.data &&\r\n dataSource.data.length > 0 &&\r\n data?.filterFn\r\n \"\r\n class=\"page-number-display\"\r\n >\r\n {{ currentPageNumber }}\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- TOOLTIP PERSONALIZADO -->\r\n <lib-table-tooltip\r\n [state]=\"tooltipState\"\r\n [styles]=\"getTooltipStyles()\"\r\n ></lib-table-tooltip>\r\n</div>\r\n", styles: ["@charset \"UTF-8\";:host{--table-color: #1976d2;--table-text-color: white;--table-header-bg: #1976d2;--table-header-text: white;--table-scrollbar-bg: #f5f5f5;--table-scrollbar-thumb: var(--table-color)}.table-container::-webkit-scrollbar{width:10px;height:10px;background-color:var(--table-scrollbar-bg, #f5f5f5)}.table-container::-webkit-scrollbar-thumb{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));border-radius:5px}.table-container::-webkit-scrollbar-thumb:hover{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));opacity:.8}.overflow-x-auto::-webkit-scrollbar,.overflow-y-auto::-webkit-scrollbar,.overflow-auto::-webkit-scrollbar{width:10px;height:10px;background-color:var(--table-scrollbar-bg, #f5f5f5)}.overflow-x-auto::-webkit-scrollbar-thumb,.overflow-y-auto::-webkit-scrollbar-thumb,.overflow-auto::-webkit-scrollbar-thumb{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));border-radius:5px}.overflow-x-auto::-webkit-scrollbar-thumb:hover,.overflow-y-auto::-webkit-scrollbar-thumb:hover,.overflow-auto::-webkit-scrollbar-thumb:hover{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));opacity:.8}:host ::ng-deep .mat-table .mat-header-cell,:host ::ng-deep .mat-header-cell,:host ::ng-deep th.mat-header-cell,:host ::ng-deep .mat-mdc-table .mat-mdc-header-cell,:host ::ng-deep .mat-mdc-header-cell,:host ::ng-deep th.mat-mdc-header-cell{background-color:var(--table-header-bg, #1976d2)!important;color:var(--table-header-text, white)!important;font-weight:600}::ng-deep .fa-filter,::ng-deep i.fa-filter{color:var(--table-color, #1976d2)!important}::ng-deep .rounded-xl.border{border-color:var(--table-color, #1976d2)!important}::ng-deep .border-gray-200.rounded-xl{border-color:var(--table-color, #1976d2)!important}::ng-deep .mdc-notched-outline__notch{border-right:none!important}.mat-mdc-text-field-wrapper.mdc-text-field--outlined .mat-mdc-form-field-infix{padding-bottom:0}::ng-deep .mat-mdc-text-field-wrapper.mdc-text-field--outlined .mat-mdc-form-field-infix{padding-bottom:0}.mat-mdc-text-field-wrapper.mdc-text-field--outlined.mdc-text-field--disabled{background:rgba(0,0,0,.15);background-size:1px 100%!important;background-repeat:repeat-x!important}.mdc-text-field--disabled .mdc-text-field__input{color:#525252}button:not([matTooltip=\"Adicionar novo filtro\"]):not([matTooltip=\"Aplicar filtros\"]):not([matTooltip=\"Limpar filtros\"]):not([matTooltip=\"Exportar Tabela\"]):not(:has(i.fa-plus)):not(:has(i.fa-search)):not(:has(i.fas.fa-redo-alt)):not(:has(i.fa-download)){transition:all .2s ease;cursor:pointer}button:not([matTooltip=\"Adicionar novo filtro\"]):not([matTooltip=\"Aplicar filtros\"]):not([matTooltip=\"Limpar filtros\"]):not([matTooltip=\"Exportar Tabela\"]):not(:has(i.fa-plus)):not(:has(i.fa-search)):not(:has(i.fas.fa-redo-alt)):not(:has(i.fa-download)):hover:not(:disabled){opacity:.9;transform:scale(1.02)}button:not([matTooltip=\"Adicionar novo filtro\"]):not([matTooltip=\"Aplicar filtros\"]):not([matTooltip=\"Limpar filtros\"]):not([matTooltip=\"Exportar Tabela\"]):not(:has(i.fa-plus)):not(:has(i.fa-search)):not(:has(i.fas.fa-redo-alt)):not(:has(i.fa-download)):disabled{opacity:.5;cursor:not-allowed;transform:none}::ng-deep .hide-length .mat-mdc-paginator-range-label{display:none}::ng-deep .hide-next-button .mat-mdc-tooltip-trigger.mat-mdc-paginator-navigation-next.mdc-icon-button.mat-mdc-icon-button.mat-unthemed.mat-mdc-button-base{visibility:hidden}::ng-deep .hide-next-button .mat-mdc-tooltip-trigger.mat-mdc-paginator-navigation-last.mdc-icon-button.mat-mdc-icon-button.mat-unthemed.mat-mdc-button-base.ng-star-inserted{visibility:hidden}::ng-deep .mat-mdc-text-field-wrapper.mdc-text-field.ng-tns-c162-1.mdc-text-field--filled{width:25dvw}::ng-deep .custom-filter .mat-mdc-text-field-wrapper{width:20dvw;max-width:300px}button[matTooltip=\"Adicionar novo filtro\"],button:has(i.fa-plus){display:inline-flex;align-items:center;gap:.5rem;border-radius:9999px;border:2px solid #93c5fd;background-color:#dbeafe;padding:.5rem 1.5rem;font-size:.875rem;font-weight:500;color:#2563eb;transition:all .3s;cursor:pointer;transform:translateY(0)}button[matTooltip=\"Adicionar novo filtro\"]:hover,button:has(i.fa-plus):hover{transform:translateY(-2px);border-color:#60a5fa;background-color:#bfdbfe;box-shadow:0 4px 6px #0000001a}button[matTooltip=\"Aplicar filtros\"],button:has(i.fa-search){display:inline-flex;align-items:center;gap:.5rem;border-radius:.5rem;background-color:#16a34a;padding:.5rem 1.25rem;font-size:.875rem;font-weight:500;color:#fff;transition:background-color .2s;cursor:pointer;border:none}button[matTooltip=\"Aplicar filtros\"]:hover,button:has(i.fa-search):hover{background-color:#15803d}button[matTooltip=\"Aplicar filtros\"]:disabled,button:has(i.fa-search):disabled{opacity:.5;cursor:not-allowed}button[matTooltip=\"Limpar filtros\"],button:has(i.fas.fa-redo-alt){display:inline-flex;align-items:center;gap:.5rem;border-radius:.5rem;background-color:#ef4444;padding:.5rem 1.25rem;font-size:.875rem;font-weight:500;color:#fff;transition:background-color .2s;cursor:pointer;border:none}button[matTooltip=\"Limpar filtros\"]:hover,button:has(i.fas.fa-redo-alt):hover{background-color:#dc2626}button[matTooltip=\"Exportar Tabela\"],button:has(i.fa-download){display:inline-flex;align-items:center;gap:.5rem;border-radius:.5rem;background-color:#f97316;padding:.5rem 1.25rem;font-size:.875rem;font-weight:500;color:#fff;transition:background-color .2s;cursor:pointer;border:none}button[matTooltip=\"Exportar Tabela\"]:hover:not(:disabled),button:has(i.fa-download):hover:not(:disabled){background-color:#ea580c}button[matTooltip=\"Exportar Tabela\"]:disabled,button:has(i.fa-download):disabled{opacity:.5;cursor:not-allowed}\n"], dependencies: [{ kind: "directive", type: i7.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i8.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i8.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i8.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i8.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i8.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i8.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i8.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "component", type: i9.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i9.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i9.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i9.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { kind: "directive", type: i9.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i9.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i9.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i9.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i9.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i9.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: i9.MatNoDataRow, selector: "ng-template[matNoDataRow]" }, { kind: "component", type: i10.MatPaginator, selector: "mat-paginator", inputs: ["disabled"], exportAs: ["matPaginator"] }, { kind: "directive", type: i11.MatSort, selector: "[matSort]", inputs: ["matSortDisabled", "matSortActive", "matSortStart", "matSortDirection", "matSortDisableClear"], outputs: ["matSortChange"], exportAs: ["matSort"] }, { kind: "component", type: i11.MatSortHeader, selector: "[mat-sort-header]", inputs: ["disabled", "mat-sort-header", "arrowPosition", "start", "sortActionDescription", "disableClear"], exportAs: ["matSortHeader"] }, { kind: "component", type: i12.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i12.MatLabel, selector: "mat-label" }, { kind: "directive", type: i12.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "directive", type: i13.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i14.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator"], exportAs: ["matSelect"] }, { kind: "component", type: i15.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i16.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "component", type: i17.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: i18.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i19.TableTabsComponent, selector: "lib-table-tabs", inputs: ["tabsData", "selectedTab"], outputs: ["tabSelected"] }, { kind: "component", type: i20.TableTooltipComponent, selector: "lib-table-tooltip", inputs: ["state", "styles"] }, { kind: "pipe", type: i7.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
696
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableComponent, decorators: [{
697
- type: Component,
698
- args: [{ selector: 'lib-table', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div *ngIf=\"data\" class=\"card-body\">\r\n <div class=\"flex flex-col justify-between gap-6\">\r\n <!-- UNIFIED CONTROL PANEL: FILTERS, SORT & ACTIONS -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"\r\n data.pagination === true &&\r\n (dropdownItems.length > 0 ||\r\n sortableDropdownItems.length > 0 ||\r\n data.actionButton)\r\n \"\r\n >\r\n <!-- PANEL HEADER: Title and Custom Action -->\r\n <div\r\n class=\"mb-4 flex flex-col items-start justify-between gap-4 border-b-2 border-gray-200 pb-4 md:flex-row md:items-center\"\r\n >\r\n <!-- Left Side: Title & Main Action Button -->\r\n <div class=\"flex flex-wrap items-center gap-4\">\r\n <div class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-filter text-xl text-blue-500\"></i>\r\n <span class=\"text-lg font-semibold text-gray-700\"\r\n >Filtros e A\u00E7\u00F5es</span\r\n >\r\n </div>\r\n <button\r\n *ngIf=\"data.actionButton && data.actionButton.condition\"\r\n [ngClass]=\"\r\n (data.actionButton.colorClass || 'bg-blue-500') +\r\n ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\r\n \"\r\n [routerLink]=\"data.actionButton.routerLink\"\r\n (click)=\"\r\n data.actionButton.method ? data.actionButton.method($event) : null\r\n \"\r\n >\r\n <i\r\n *ngIf=\"data.actionButton.icon\"\r\n [class]=\"data.actionButton.icon\"\r\n ></i>\r\n {{ data.actionButton.label }}\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <!-- SORT FIELD (Integrated with filters) -->\r\n <div class=\"mb-4\" *ngIf=\"sortableDropdownItems.length > 0\">\r\n <div class=\"flex flex-wrap items-center gap-3 p-2\">\r\n <div class=\"min-w-[200px] flex-1\">\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\r\n <span>Ordenar por</span>\r\n </mat-label>\r\n <mat-select placeholder=\"Selecione...\" [formControl]=\"selectSort\">\r\n <mat-option\r\n *ngFor=\"let item of sortableDropdownItems\"\r\n [value]=\"item\"\r\n >\r\n <div class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\r\n <span>{{ item.title }}</span>\r\n </div>\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- FILTERS CONTENT (WITH REFINEMENTS) -->\r\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\r\n <div\r\n [formGroup]=\"$any(filterGroup)\"\r\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\r\n class=\"flex flex-wrap items-center gap-3 p-2\"\r\n >\r\n <!-- FILTER TYPE SELECTOR -->\r\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>Tipo de filtro</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione o tipo...\"\r\n formControlName=\"selectFilter\"\r\n (selectionChange)=\"onSelectFilterChange()\"\r\n >\r\n <mat-option\r\n *ngFor=\"let item of getAvailableFilterOptions()\"\r\n [value]=\"item\"\r\n >\r\n <div class=\"flex items-center gap-2\">\r\n <i\r\n [class]=\"item.icon || 'fa fa-filter'\"\r\n class=\"text-sm text-blue-500\"\r\n ></i>\r\n <span>{{ item.title }}</span>\r\n </div>\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- TEXT FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-search text-gray-400\"></i>\r\n <span>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Filtrar\"\r\n }}</span>\r\n </mat-label>\r\n <input\r\n (keyup.enter)=\"search($event)\"\r\n formControlName=\"typeFilter\"\r\n matInput\r\n placeholder=\"Digite para filtrar...\"\r\n #input\r\n />\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DROPDOWN FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value &&\r\n $any(filterGroup)\r\n .get('selectFilter')\r\n ?.value.hasOwnProperty('items')\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Selecione\"\r\n }}</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione...\"\r\n formControlName=\"selectItem\"\r\n multiple\r\n >\r\n <mat-option\r\n *ngFor=\"\r\n let item of $any(filterGroup).get('selectFilter')?.value\r\n .items\r\n \"\r\n [value]=\"item\"\r\n >\r\n {{ item.label }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DATE FILTER -->\r\n <div\r\n class=\"min-w-[340px] flex-auto\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n 'filterByDate'\r\n \"\r\n >\r\n <div\r\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Inicial</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n matInput\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'initialDate', i)\"\r\n formControlName=\"initialDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Final</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'finalDate', i)\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n\r\n <!-- REMOVE FILTER BUTTON -->\r\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\r\n <button\r\n (click)=\"removeFilter(i)\"\r\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\r\n matTooltip=\"Remover filtro\"\r\n >\r\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- PANEL FOOTER: Add Filter & Action Buttons -->\r\n <div\r\n class=\"-mb-2 flex items-start justify-between gap-4 border-t border-gray-200 pt-4\"\r\n >\r\n <!-- Add Filter Button -->\r\n <div *ngIf=\"dropdownItems.length > 0\">\r\n <button\r\n (click)=\"addFilter()\"\r\n class=\"transform rounded-full border-2 border-blue-300 bg-blue-50 px-6 py-2 text-sm font-medium text-blue-600 transition-all duration-300 hover:-translate-y-0.5 hover:border-blue-400 hover:bg-blue-100 hover:shadow-md\"\r\n matTooltip=\"Adicionar novo filtro\"\r\n >\r\n <i class=\"fa fa-plus mr-2\"></i>\r\n Adicionar Filtro\r\n </button>\r\n </div>\r\n\r\n <!-- Right Side: Search, Reset, Export -->\r\n <div\r\n class=\"flex flex-wrap gap-3 self-start sm:self-auto\"\r\n *ngIf=\"\r\n this.hasFilterableColumn === true || this.hasSortableColumn === true\r\n \"\r\n >\r\n <button\r\n (click)=\"search()\"\r\n type=\"button\"\r\n class=\"flex items-center gap-2 rounded-lg bg-green-600 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-green-700\"\r\n matTooltip=\"Aplicar filtros\"\r\n >\r\n <i class=\"fa fa-search\"></i>\r\n Pesquisar\r\n </button>\r\n\r\n <button\r\n (click)=\"resetFilter()\"\r\n class=\"flex items-center gap-2 rounded-lg bg-red-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-red-600\"\r\n matTooltip=\"Limpar filtros\"\r\n >\r\n <i class=\"fas fa-redo-alt\"></i>\r\n Resetar\r\n </button>\r\n\r\n <button\r\n *ngIf=\"data.download !== false && downloadTable\"\r\n class=\"flex items-center gap-2 rounded-lg bg-orange-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-orange-600\"\r\n matTooltipPosition=\"above\"\r\n matTooltip=\"Exportar Tabela\"\r\n [disabled]=\"\r\n this.dataSource && this.dataSource.filteredData.length <= 0\r\n \"\r\n (click)=\"\r\n $any(arrange) && downloadTable !== undefined\r\n ? downloadTable($any(arrange), data.conditions || [])\r\n : null\r\n \"\r\n >\r\n <i class=\"fa fa-download\"></i>\r\n Exportar\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- UNIFIED CONTROL PANEL (for non-paginated tables): SEARCH, SORT & FILTERS -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"\r\n data.pagination === false &&\r\n (hasFilterableColumn === true ||\r\n dropdownItems.length > 0 ||\r\n data.actionButton)\r\n \"\r\n >\r\n <!-- PANEL HEADER: Title and Custom Action -->\r\n <div\r\n class=\"mb-4 flex flex-col items-start justify-between gap-4 border-b-2 border-gray-200 pb-4 md:flex-row md:items-center\"\r\n >\r\n <!-- Left Side: Title & Main Action Button -->\r\n <div class=\"flex flex-wrap items-center gap-4\">\r\n <div class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-filter text-xl text-blue-500\"></i>\r\n <span class=\"text-lg font-semibold text-gray-700\"\r\n >Filtros e A\u00E7\u00F5es</span\r\n >\r\n </div>\r\n <button\r\n *ngIf=\"data.actionButton && data.actionButton.condition\"\r\n [ngClass]=\"\r\n (data.actionButton.colorClass || 'bg-blue-500') +\r\n ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\r\n \"\r\n [routerLink]=\"data.actionButton.routerLink\"\r\n (click)=\"\r\n data.actionButton.method ? data.actionButton.method($event) : null\r\n \"\r\n >\r\n <i\r\n *ngIf=\"data.actionButton.icon\"\r\n [class]=\"data.actionButton.icon\"\r\n ></i>\r\n {{ data.actionButton.label }}\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <!-- SIMPLE SEARCH -->\r\n <div\r\n class=\"mb-4\"\r\n *ngIf=\"hasFilterableColumn === true && data.showSimpleSearch !== false\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-search text-blue-500\"></i>\r\n Buscar\r\n </mat-label>\r\n <input\r\n matInput\r\n (keyup.enter)=\"search($event)\"\r\n (keyup)=\"applyFilter(filterInput.value)\"\r\n placeholder=\"Digite para filtrar...\"\r\n #filterInput\r\n />\r\n <mat-icon matSuffix class=\"text-gray-500\">search</mat-icon>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- FILTERS PANEL -->\r\n <div *ngIf=\"dropdownItems.length > 0\">\r\n <!-- FILTERS CONTENT -->\r\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\r\n <div\r\n [formGroup]=\"$any(filterGroup)\"\r\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\r\n class=\"flex flex-wrap items-center gap-3\"\r\n >\r\n <!-- FILTER TYPE SELECTOR -->\r\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>Tipo de filtro</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione o tipo...\"\r\n formControlName=\"selectFilter\"\r\n (selectionChange)=\"onSelectFilterChange()\"\r\n >\r\n <mat-option\r\n *ngFor=\"let item of getAvailableFilterOptions()\"\r\n [value]=\"item\"\r\n >\r\n <div class=\"flex items-center gap-2\">\r\n <i\r\n [class]=\"item.icon || 'fa fa-filter'\"\r\n class=\"text-sm text-blue-500\"\r\n ></i>\r\n <span>{{ item.title }}</span>\r\n </div>\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- TEXT FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n 'filter'\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-search text-gray-400\"></i>\r\n <span>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Filtrar\"\r\n }}</span>\r\n </mat-label>\r\n <input\r\n (keyup.enter)=\"search($event)\"\r\n formControlName=\"typeFilter\"\r\n matInput\r\n placeholder=\"Digite para filtrar...\"\r\n #input\r\n />\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DROPDOWN FILTER -->\r\n <div\r\n class=\"min-w-[200px] flex-1\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value &&\r\n $any(filterGroup)\r\n .get('selectFilter')\r\n ?.value.hasOwnProperty('items')\r\n \"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>{{\r\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n \"Selecione\"\r\n }}</mat-label>\r\n <mat-select\r\n placeholder=\"Selecione...\"\r\n formControlName=\"selectItem\"\r\n multiple\r\n >\r\n <mat-option\r\n *ngFor=\"\r\n let item of $any(filterGroup).get('selectFilter')?.value\r\n .items\r\n \"\r\n [value]=\"item\"\r\n >\r\n {{ item.label }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- DATE FILTER -->\r\n <div\r\n class=\"min-w-[340px] flex-auto\"\r\n *ngIf=\"\r\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n 'filterByDate'\r\n \"\r\n >\r\n <div\r\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Inicial</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n matInput\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'initialDate', i)\"\r\n (blur)=\"onDateFilterChange()\"\r\n formControlName=\"initialDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n\r\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n <mat-label class=\"flex items-center gap-2\">\r\n <i class=\"fa fa-calendar text-gray-400\"></i>\r\n <span>Data Final</span>\r\n </mat-label>\r\n <input\r\n type=\"text\"\r\n (keydown.enter)=\"search($event)\"\r\n (input)=\"onDateInput($event, 'finalDate', i)\"\r\n (blur)=\"onDateFilterChange()\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n placeholder=\"DD/MM/AAAA\"\r\n maxlength=\"10\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n\r\n <!-- REMOVE FILTER BUTTON -->\r\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\r\n <button\r\n (click)=\"removeFilter(i)\"\r\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\r\n matTooltip=\"Remover filtro\"\r\n >\r\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- PANEL FOOTER: Add Filter & Actions -->\r\n <div\r\n class=\"-mb-2 flex items-start justify-between gap-4 border-t border-gray-200 pt-4\"\r\n >\r\n <!-- Add Filter Button -->\r\n <div *ngIf=\"dropdownItems.length > 0\">\r\n <button\r\n (click)=\"addFilter()\"\r\n class=\"transform rounded-full border-2 border-blue-300 bg-blue-50 px-6 py-2 text-sm font-medium text-blue-600 transition-all duration-300 hover:-translate-y-0.5 hover:border-blue-400 hover:bg-blue-100 hover:shadow-md\"\r\n matTooltip=\"Adicionar novo filtro\"\r\n >\r\n <i class=\"fa fa-plus mr-2\"></i>\r\n Adicionar Filtro\r\n </button>\r\n </div>\r\n\r\n <!-- Action Buttons -->\r\n <div\r\n class=\"flex flex-wrap gap-3 self-start sm:self-auto\"\r\n *ngIf=\"\r\n this.hasFilterableColumn === true ||\r\n this.hasSortableColumn === true\r\n \"\r\n >\r\n <button\r\n (click)=\"search()\"\r\n type=\"button\"\r\n class=\"flex items-center gap-2 rounded-lg bg-green-600 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-green-700\"\r\n matTooltip=\"Aplicar filtros\"\r\n >\r\n <i class=\"fa fa-search\"></i>\r\n Pesquisar\r\n </button>\r\n\r\n <button\r\n (click)=\"resetFilter()\"\r\n class=\"flex items-center gap-2 rounded-lg bg-red-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-red-600\"\r\n matTooltip=\"Limpar filtros\"\r\n >\r\n <i class=\"fas fa-redo-alt\"></i>\r\n Resetar\r\n </button>\r\n\r\n <button\r\n *ngIf=\"data.download !== false && downloadTable\"\r\n class=\"flex items-center gap-2 rounded-lg bg-orange-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-orange-600\"\r\n matTooltipPosition=\"above\"\r\n matTooltip=\"Exportar Tabela\"\r\n [disabled]=\"\r\n this.dataSource && this.dataSource.filteredData.length <= 0\r\n \"\r\n (click)=\"handleDownload()\"\r\n >\r\n <i class=\"fa fa-download\"></i>\r\n Exportar\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"flex flex-col\">\r\n <!-- TABS COMPONENT -->\r\n <lib-table-tabs\r\n *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\r\n [tabsData]=\"data.tabs.tabsData\"\r\n [selectedTab]=\"selectedTab\"\r\n (tabSelected)=\"onTabSelected($event)\"\r\n ></lib-table-tabs>\r\n <div\r\n class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl table-container\"\r\n >\r\n <table\r\n mat-table\r\n [dataSource]=\"dataSource\"\r\n matSort\r\n #sort=\"matSort\"\r\n matSortActive=\"createdAt\"\r\n matSortDirection=\"desc\"\r\n >\r\n <ng-container\r\n *ngFor=\"let col of data.displayedColumns; trackBy: trackByProperty\"\r\n matColumnDef=\"{{ col.property }}\"\r\n >\r\n <ng-container *matHeaderCellDef>\r\n <!-- IF THE COLUMN IS NOT SORTABLE, THEN DON'T SHOW THE SORT BUTTONS -->\r\n <th\r\n *ngIf=\"!col.isSortable || data.pagination === true\"\r\n mat-header-cell\r\n >\r\n {{ col.title }}\r\n </th>\r\n <!-- IF THE COLUMN IS SORTABLE, THEN SHOW THE SORT BUTTONS -->\r\n <th\r\n *ngIf=\"col.isSortable && data.pagination === false\"\r\n mat-header-cell\r\n mat-sort-header\r\n >\r\n {{ col.title }}\r\n </th>\r\n <td\r\n mat-cell\r\n *matCellDef=\"let row\"\r\n (click)=\"col.method ? col.method(row) : null\"\r\n (mouseenter)=\"onCellMouseEnter($event, row, col)\"\r\n (mouseleave)=\"onCellMouseLeave()\"\r\n (mousemove)=\"onCellMouseMove($event)\"\r\n >\r\n <!-- CHECK IF THE COLUMN MUST BE DISPLAYED -->\r\n <span *ngIf=\"!col.image && !col.iconClass && !col.method\">\r\n <ng-container>\r\n <span\r\n *ngIf=\"\r\n col.charLimit &&\r\n row[col.property] &&\r\n row[col.property].length > col.charLimit;\r\n else withinLimit\r\n \"\r\n >\r\n <a\r\n *ngIf=\"col.hasLink === true\"\r\n [href]=\"row[col.property]\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row) }}\r\n </a>\r\n <a\r\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\r\n [href]=\"col.hasLink\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row) }}\r\n </a>\r\n <span\r\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\r\n >\r\n {{ getDisplayValue(col, row) }}\r\n </span>\r\n </span>\r\n </ng-container>\r\n <ng-template #withinLimit>\r\n <a\r\n *ngIf=\"col.hasLink === true\"\r\n [href]=\"row[col.property]\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row, true) }}\r\n </a>\r\n <a\r\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\r\n [href]=\"col.hasLink\"\r\n target=\"_blank\"\r\n >\r\n {{ getDisplayValue(col, row, true) }}\r\n </a>\r\n <span\r\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\r\n >\r\n {{ getDisplayValue(col, row, true) }}\r\n </span>\r\n </ng-template>\r\n </span>\r\n <!------------------- IMAGE ------------------>\r\n <!-- Imagem com loading -->\r\n <div\r\n *ngIf=\"\r\n col.image && col.image.path && !col.iconClass && !col.method\r\n \"\r\n class=\"relative inline-block\"\r\n >\r\n <mat-spinner\r\n *ngIf=\"isImageLoading(row, col)\"\r\n [diameter]=\"32\"\r\n class=\"absolute inset-0 m-auto\"\r\n ></mat-spinner>\r\n <img\r\n [src]=\"col.image.path + '/' + row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n [class.opacity-0]=\"isImageLoading(row, col)\"\r\n alt=\"\"\r\n (load)=\"onImageLoad(row, col)\"\r\n (error)=\"onImageError($event, row, col, col.image.default)\"\r\n />\r\n </div>\r\n\r\n <!-- Imagem com URL completa e loading -->\r\n <div\r\n *ngIf=\"\r\n col.image && col.image.url && !col.iconClass && !col.method\r\n \"\r\n class=\"relative inline-block\"\r\n >\r\n <mat-spinner\r\n *ngIf=\"isImageLoading(row, col)\"\r\n [diameter]=\"32\"\r\n class=\"absolute inset-0 m-auto\"\r\n ></mat-spinner>\r\n <img\r\n [src]=\"row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n [class.opacity-0]=\"isImageLoading(row, col)\"\r\n alt=\"\"\r\n (load)=\"onImageLoad(row, col)\"\r\n (error)=\"onImageError($event, row, col, col.image.default)\"\r\n />\r\n </div>\r\n <ng-container *ngIf=\"col.iconClass\">\r\n <button\r\n *ngFor=\"let iconClass of col.iconClass\"\r\n (click)=\"\r\n iconClass.buttonMethod\r\n ? iconClass.buttonMethod(row, $event)\r\n : $event.stopPropagation()\r\n \"\r\n >\r\n <span\r\n [ngClass]=\"iconClass.class\"\r\n *ngIf=\"\r\n iconClass.condition === undefined ||\r\n (iconClass.condition !== undefined &&\r\n $any(iconClass.condition)(row))\r\n \"\r\n >{{ iconClass.text }}</span\r\n >\r\n </button>\r\n </ng-container>\r\n </td>\r\n </ng-container>\r\n </ng-container>\r\n\r\n <tr mat-header-row *matHeaderRowDef=\"columnProperties\"></tr>\r\n <tr\r\n [ngClass]=\"{\r\n 'example-element-row': data.isNotClickable === true,\r\n 'example-element-row cursor-pointer': !data.isNotClickable,\r\n }\"\r\n mat-row\r\n *matRowDef=\"let row; columns: columnProperties\"\r\n (click)=\"goToDetails(row)\"\r\n ></tr>\r\n\r\n <!-- ROW SHOWN WHEN THERE IS NO MATCHING DATA. -->\r\n <tr class=\"mat-row\" *matNoDataRow>\r\n <td *ngIf=\"!(isLoading$ | async)\" class=\"mat-cell p-4\" colspan=\"4\">\r\n Nenhum resultado encontrado para a busca\r\n </td>\r\n </tr>\r\n </table>\r\n\r\n <div class=\"flex justify-center\" *ngIf=\"isLoading$ | async\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n\r\n <div class=\"paginator-container\">\r\n <mat-paginator\r\n #paginator\r\n [pageSizeOptions]=\"[25, 50, 100]\"\r\n [pageSize]=\"pageSize\"\r\n [length]=\"totalItems\"\r\n showFirstLastButtons\r\n aria-label=\"Select page of periodic elements\"\r\n (page)=\"onPageChange($event)\"\r\n [ngClass]=\"{\r\n 'hide-length':\r\n ['filter', 'filterByDate', 'equals'].includes(\r\n this.currentArrange\r\n ) || this.data.filterFn,\r\n 'hide-next-button': !hasNextPage && data.pagination === true,\r\n 'hide-last-button':\r\n (!hasNextPage && data.pagination === true) ||\r\n this.data.filterFn,\r\n }\"\r\n >\r\n </mat-paginator>\r\n <div\r\n *ngIf=\"\r\n !(isLoading$ | async) &&\r\n dataSource?.data &&\r\n dataSource.data.length > 0 &&\r\n data?.filterFn\r\n \"\r\n class=\"page-number-display\"\r\n >\r\n {{ currentPageNumber }}\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- TOOLTIP PERSONALIZADO -->\r\n <lib-table-tooltip\r\n [state]=\"tooltipState\"\r\n [styles]=\"getTooltipStyles()\"\r\n ></lib-table-tooltip>\r\n</div>\r\n", styles: ["@charset \"UTF-8\";:host{--table-color: #1976d2;--table-text-color: white;--table-header-bg: #1976d2;--table-header-text: white;--table-scrollbar-bg: #f5f5f5;--table-scrollbar-thumb: var(--table-color)}.table-container::-webkit-scrollbar{width:10px;height:10px;background-color:var(--table-scrollbar-bg, #f5f5f5)}.table-container::-webkit-scrollbar-thumb{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));border-radius:5px}.table-container::-webkit-scrollbar-thumb:hover{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));opacity:.8}.overflow-x-auto::-webkit-scrollbar,.overflow-y-auto::-webkit-scrollbar,.overflow-auto::-webkit-scrollbar{width:10px;height:10px;background-color:var(--table-scrollbar-bg, #f5f5f5)}.overflow-x-auto::-webkit-scrollbar-thumb,.overflow-y-auto::-webkit-scrollbar-thumb,.overflow-auto::-webkit-scrollbar-thumb{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));border-radius:5px}.overflow-x-auto::-webkit-scrollbar-thumb:hover,.overflow-y-auto::-webkit-scrollbar-thumb:hover,.overflow-auto::-webkit-scrollbar-thumb:hover{background-color:var(--table-scrollbar-thumb, var(--table-color, #1976d2));opacity:.8}:host ::ng-deep .mat-table .mat-header-cell,:host ::ng-deep .mat-header-cell,:host ::ng-deep th.mat-header-cell,:host ::ng-deep .mat-mdc-table .mat-mdc-header-cell,:host ::ng-deep .mat-mdc-header-cell,:host ::ng-deep th.mat-mdc-header-cell{background-color:var(--table-header-bg, #1976d2)!important;color:var(--table-header-text, white)!important;font-weight:600}::ng-deep .fa-filter,::ng-deep i.fa-filter{color:var(--table-color, #1976d2)!important}::ng-deep .rounded-xl.border{border-color:var(--table-color, #1976d2)!important}::ng-deep .border-gray-200.rounded-xl{border-color:var(--table-color, #1976d2)!important}::ng-deep .mdc-notched-outline__notch{border-right:none!important}.mat-mdc-text-field-wrapper.mdc-text-field--outlined .mat-mdc-form-field-infix{padding-bottom:0}::ng-deep .mat-mdc-text-field-wrapper.mdc-text-field--outlined .mat-mdc-form-field-infix{padding-bottom:0}.mat-mdc-text-field-wrapper.mdc-text-field--outlined.mdc-text-field--disabled{background:rgba(0,0,0,.15);background-size:1px 100%!important;background-repeat:repeat-x!important}.mdc-text-field--disabled .mdc-text-field__input{color:#525252}button:not([matTooltip=\"Adicionar novo filtro\"]):not([matTooltip=\"Aplicar filtros\"]):not([matTooltip=\"Limpar filtros\"]):not([matTooltip=\"Exportar Tabela\"]):not(:has(i.fa-plus)):not(:has(i.fa-search)):not(:has(i.fas.fa-redo-alt)):not(:has(i.fa-download)){transition:all .2s ease;cursor:pointer}button:not([matTooltip=\"Adicionar novo filtro\"]):not([matTooltip=\"Aplicar filtros\"]):not([matTooltip=\"Limpar filtros\"]):not([matTooltip=\"Exportar Tabela\"]):not(:has(i.fa-plus)):not(:has(i.fa-search)):not(:has(i.fas.fa-redo-alt)):not(:has(i.fa-download)):hover:not(:disabled){opacity:.9;transform:scale(1.02)}button:not([matTooltip=\"Adicionar novo filtro\"]):not([matTooltip=\"Aplicar filtros\"]):not([matTooltip=\"Limpar filtros\"]):not([matTooltip=\"Exportar Tabela\"]):not(:has(i.fa-plus)):not(:has(i.fa-search)):not(:has(i.fas.fa-redo-alt)):not(:has(i.fa-download)):disabled{opacity:.5;cursor:not-allowed;transform:none}::ng-deep .hide-length .mat-mdc-paginator-range-label{display:none}::ng-deep .hide-next-button .mat-mdc-tooltip-trigger.mat-mdc-paginator-navigation-next.mdc-icon-button.mat-mdc-icon-button.mat-unthemed.mat-mdc-button-base{visibility:hidden}::ng-deep .hide-next-button .mat-mdc-tooltip-trigger.mat-mdc-paginator-navigation-last.mdc-icon-button.mat-mdc-icon-button.mat-unthemed.mat-mdc-button-base.ng-star-inserted{visibility:hidden}::ng-deep .mat-mdc-text-field-wrapper.mdc-text-field.ng-tns-c162-1.mdc-text-field--filled{width:25dvw}::ng-deep .custom-filter .mat-mdc-text-field-wrapper{width:20dvw;max-width:300px}button[matTooltip=\"Adicionar novo filtro\"],button:has(i.fa-plus){display:inline-flex;align-items:center;gap:.5rem;border-radius:9999px;border:2px solid #93c5fd;background-color:#dbeafe;padding:.5rem 1.5rem;font-size:.875rem;font-weight:500;color:#2563eb;transition:all .3s;cursor:pointer;transform:translateY(0)}button[matTooltip=\"Adicionar novo filtro\"]:hover,button:has(i.fa-plus):hover{transform:translateY(-2px);border-color:#60a5fa;background-color:#bfdbfe;box-shadow:0 4px 6px #0000001a}button[matTooltip=\"Aplicar filtros\"],button:has(i.fa-search){display:inline-flex;align-items:center;gap:.5rem;border-radius:.5rem;background-color:#16a34a;padding:.5rem 1.25rem;font-size:.875rem;font-weight:500;color:#fff;transition:background-color .2s;cursor:pointer;border:none}button[matTooltip=\"Aplicar filtros\"]:hover,button:has(i.fa-search):hover{background-color:#15803d}button[matTooltip=\"Aplicar filtros\"]:disabled,button:has(i.fa-search):disabled{opacity:.5;cursor:not-allowed}button[matTooltip=\"Limpar filtros\"],button:has(i.fas.fa-redo-alt){display:inline-flex;align-items:center;gap:.5rem;border-radius:.5rem;background-color:#ef4444;padding:.5rem 1.25rem;font-size:.875rem;font-weight:500;color:#fff;transition:background-color .2s;cursor:pointer;border:none}button[matTooltip=\"Limpar filtros\"]:hover,button:has(i.fas.fa-redo-alt):hover{background-color:#dc2626}button[matTooltip=\"Exportar Tabela\"],button:has(i.fa-download){display:inline-flex;align-items:center;gap:.5rem;border-radius:.5rem;background-color:#f97316;padding:.5rem 1.25rem;font-size:.875rem;font-weight:500;color:#fff;transition:background-color .2s;cursor:pointer;border:none}button[matTooltip=\"Exportar Tabela\"]:hover:not(:disabled),button:has(i.fa-download):hover:not(:disabled){background-color:#ea580c}button[matTooltip=\"Exportar Tabela\"]:disabled,button:has(i.fa-download):disabled{opacity:.5;cursor:not-allowed}\n"] }]
699
- }], ctorParameters: function () { return [{ type: i1.Router }, { type: i2.TableService }, { type: i3.FilterService }, { type: i4.TooltipService }, { type: i5.PaginationService }, { type: i6.AngularFirestore }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { data: [{
700
- type: Input
701
- }], downloadTable: [{
702
- type: Input
703
- }], paginator: [{
704
- type: ViewChild,
705
- args: [MatPaginator]
706
- }], sort: [{
707
- type: ViewChild,
708
- args: [MatSort]
709
- }] } });
710
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-firebase-table-kxp/src/lib/components/table/table.component.ts","../../../../../../projects/ng-firebase-table-kxp/src/lib/components/table/table.component.html"],"names":[],"mappings":"AAAA,OAAO,EAML,cAAc,GAIf,MAAM,mBAAmB,CAAC;AAC3B,OAAO,EACL,SAAS,EACT,KAAK,EAEL,SAAS,EAKT,uBAAuB,GAGxB,MAAM,eAAe,CAAC;AAKvB,OAAO,EAAE,YAAY,EAAa,MAAM,6BAA6B,CAAC;AACtE,OAAO,EAAE,OAAO,EAAE,MAAM,wBAAwB,CAAC;AACjD,OAAO,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAC;AAM7D,OAAO,EACL,eAAe,EACf,cAAc,EACd,QAAQ,EACR,eAAe,EACf,WAAW,GACZ,MAAM,yBAAyB,CAAC;AACjC,OAAO,EACL,YAAY,EACZ,cAAc,EACd,OAAO,EACP,SAAS,EACT,eAAe,GAChB,MAAM,MAAM,CAAC;AAKd,OAAO,EAAE,WAAW,EAAE,SAAS,EAAa,MAAM,gBAAgB,CAAC;;;;;;;;;;;;;;;;;;;;;;AAQnE,MAAM,OAAO,cAAc;IAwDzB,cAAc;IACd,YACU,MAAc,EACd,YAA0B,EAC1B,aAA4B,EAC5B,cAA8B,EAC9B,iBAAoC,EACpC,SAA2B,EAC3B,EAAc,EACd,GAAsB;QAPtB,WAAM,GAAN,MAAM,CAAQ;QACd,iBAAY,GAAZ,YAAY,CAAc;QAC1B,kBAAa,GAAb,aAAa,CAAe;QAC5B,mBAAc,GAAd,cAAc,CAAgB;QAC9B,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,cAAS,GAAT,SAAS,CAAkB;QAC3B,OAAE,GAAF,EAAE,CAAY;QACd,QAAG,GAAH,GAAG,CAAmB;QA3DhC,YAAO,GAAmB,IAAI,CAAC;QAI/B,sBAAiB,GAAG,CAAC,CAAC;QACtB,2BAAsB,GAAG,CAAC,CAAC;QAEpB,eAAU,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAEjD,UAAK,GAAQ,EAAE,CAAC;QAChB,kBAAa,GAAQ,EAAE,CAAC;QACvB,YAAO,GAAoC,IAAI,CAAC;QAChD,aAAQ,GAAoC,IAAI,CAAC;QACjD,WAAM,GAA+C;YAC3D,KAAK,EAAE,cAAc,CAAC,UAAU;YAChC,KAAK,EAAE,cAAc,CAAC,UAAU;SACjC,CAAC;QACK,qBAAgB,GAAa,EAAE,CAAC;QAGvC,eAAU,GAAgB,IAAI,WAAW,CAAC,EAAE,CAAC,CAAC;QAC9C,mBAAc,GAAG,EAAE,CAAC;QAEpB,gBAAW,GAAY,KAAK,CAAC;QAC7B,kBAAa,GAAmB,EAAE,CAAC;QACnC,0BAAqB,GAAmB,EAAE,CAAC;QAC3C,aAAQ,GAAG,cAAc,CAAC,SAAS,CAAC;QACpC,eAAU,GAAG,CAAC,CAAC;QAEf,gBAAW,GAAkB,IAAI,CAAC;QAElC,wBAAmB,GAAY,KAAK,CAAC;QACrC,sBAAiB,GAAY,KAAK,CAAC;QAE3B,kBAAa,GAAG,IAAI,OAAO,EAAU,CAAC;QAC7B,mBAAc,GAAG,cAAc,CAAC,gBAAgB,CAAC;QAElE,gBAAW,GAAW,CAAC,CAAC;QAQxB,iCAAiC;QACjC,sBAAiB,GAAG,IAAI,GAAG,EAAmB,CAAC;QAEvC,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAarC,IAAI,CAAC,WAAW,GAAG,IAAI,SAAS,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC;QAC3E,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,cAAc,CAAC,kBAAkB,EAAE,CAAC;IAC/D,CAAC;IAED,8CAA8C;IAE9C,iBAAiB;QACf,OAAO,IAAI,CAAC,aAAa,CAAC,iBAAiB,EAAE,CAAC;IAChD,CAAC;IAED,SAAS,CAAC,UAMT;QACC,IAAI,CAAC,aAAa,CAAC,SAAS,CAC1B,IAAI,CAAC,WAAW,EAChB,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,IAAI,EAC7B,UAAU,CACX,CAAC;IACJ,CAAC;IAED,mBAAmB;QACjB,OAAO,IAAI,CAAC,aAAa,CAAC,mBAAmB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAClE,CAAC;IAED,yBAAyB;QACvB,OAAO,IAAI,CAAC,aAAa,CAAC,yBAAyB,CACjD,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,IAAI,EAC7B,IAAI,CAAC,WAAW,CACjB,CAAC;IACJ,CAAC;IAED,oBAAoB;QAClB,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC;QAC9C,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC;QAClD,MAAM,cAAc,GAAG,UAAU,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,KAAK,CAAC;QAE7D,IAAI,cAAc,EAAE;YAClB,IACE,cAAc,CAAC,OAAO,KAAK,cAAc;gBACzC,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,IAAI,EAC7B;gBACA,OAAO;aACR;YACD,IAAI,CAAC,SAAS,EAAE,CAAC;SAClB;IACH,CAAC;IAED,YAAY,CAAC,KAAa;QACxB,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,CAAC,CAAC;IAC3D,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACtD,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAED,WAAW,CACT,KAAY,EACZ,WAAwC,EACxC,WAAmB;QAEnB,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,WAAW,CAAc,CAAC;QAClE,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,KAAK,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;IACpE,CAAC;IAED,kBAAkB;QAChB,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE;YAClC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,uBAAuB,CACvD,IAAI,CAAC,WAAW,EAChB,IAAI,CAAC,MAAM,CACZ,CAAC;YACF,IAAI,CAAC,wBAAwB,EAAE,CAAC;YAChC,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;IACH,CAAC;IAEO,wBAAwB;QAC9B,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAE7B,IAAI,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,sBAAsB,CAC3D,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,EACf,IAAI,CAAC,WAAW,EAChB,IAAI,CAAC,MAAM,CACZ,CAAC;QACF,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,aAAa,CAAC;QACrC,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACnC,IAAI,CAAC,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC;QAEvC,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEO,sBAAsB,CAAC,KAAU;QACvC,OAAO,IAAI,CAAC,aAAa,CAAC,sBAAsB,CAC9C,KAAK,EACL,IAAI,CAAC,WAAW,EAChB,IAAI,CAAC,MAAM,CACZ,CAAC;IACJ,CAAC;IAEO,uBAAuB;QAC7B,OAAO,IAAI,CAAC,aAAa,CAAC,uBAAuB,CAC/C,IAAI,CAAC,WAAW,EAChB,IAAI,CAAC,MAAM,CACZ,CAAC;IACJ,CAAC;IAED,gDAAgD;IAEhD,gBAAgB;QACd,OAAO,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACjE,CAAC;IAED,gBAAgB,CAAC,KAAiB,EAAE,GAAM,EAAE,GAAW;QACrD,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAClC,KAAK,EACL,GAAG,EACH,GAAG,EACH,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,EAC1C,CAAC,KAAK,EAAE,EAAE;YACR,IAAI,CAAC,YAAY,GAAG,EAAE,GAAG,IAAI,CAAC,YAAY,EAAE,GAAG,KAAK,EAAE,CAAC;YACvD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAC1B,CAAC,CACF,CAAC;IACJ,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC,KAAK,EAAE,EAAE;YAC7C,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;YAC1B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAC1B,CAAC,CAAC,CAAC;IACL,CAAC;IAED,eAAe,CAAC,KAAiB;QAC/B,IAAI,CAAC,cAAc,CAAC,eAAe,CACjC,KAAK,EACL,IAAI,CAAC,YAAY,CAAC,WAAW,EAC7B,CAAC,QAAQ,EAAE,EAAE;YACX,IAAI,CAAC,YAAY,GAAG,EAAE,GAAG,IAAI,CAAC,YAAY,EAAE,eAAe,EAAE,QAAQ,EAAE,CAAC;YACxE,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAC1B,CAAC,CACF,CAAC;IACJ,CAAC;IAED,cAAc;IAEd,aAAa,CAAC,KAAsC;QAClD,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM;YAAE,OAAO;QACpC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,KAAK,CAAC;QAC/B,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC;QAC9C,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,wCAAwC;IAExC,eAAe,CAAC,GAAW,EAAE,GAAM,EAAE,cAAuB,KAAK;QAC/D,OAAO,eAAe,CAAC,GAAG,EAAE,GAAG,EAAE,WAAW,CAAC,CAAC;IAChD,CAAC;IAED,cAAc,CAAC,GAAY,EAAE,IAAY;QACvC,OAAO,cAAc,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IACnC,CAAC;IAED,QAAQ,CAAC,KAAc;QACrB,OAAO,QAAQ,CAAC,KAAK,CAAC,CAAC;IACzB,CAAC;IAED,eAAe,CAAC,MAAc,EAAE,GAAW;QACzC,OAAO,eAAe,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IACtC,CAAC;IAED,wBAAwB;IAEhB,WAAW,CAAC,GAAM,EAAE,GAAW;QACrC,OAAO,WAAW,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;IAC/B,CAAC;IAED,cAAc,CAAC,GAAM,EAAE,GAAW;QAChC,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;QACvC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAC1C,CAAC;IAED,WAAW,CAAC,GAAM,EAAE,GAAW;QAC7B,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;QACvC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QACtC,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,YAAY,CAAC,KAAY,EAAE,GAAM,EAAE,GAAW,EAAE,YAAqB;QACnE,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;QACvC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAEtC,MAAM,MAAM,GAAG,KAAK,CAAC,MAA0B,CAAC;QAChD,IAAI,MAAM,EAAE;YACV,IAAI,YAAY,EAAE;gBAChB,MAAM,CAAC,GAAG,GAAG,YAAY,CAAC;gBAC1B,MAAM,CAAC,OAAO,GAAG,IAAI,CAAC;aACvB;iBAAM;gBACL,MAAM,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;aAC/B;SACF;QAED,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,oBAAoB;IAEpB,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE;YACnB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,EAAE,CAAC,CAAC,CAAC;SAC/C;IACH,CAAC;IAED,eAAe;QACb,IAAI,CAAC,iBAAiB,EAAE,CAAC;IAC3B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC;QAC9B,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;IAEO,iBAAiB;QACvB,MAAM,WAAW,GAAG,IAAI,CAAC,EAAE,EAAE,aAAa,CAAC;QAC3C,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO;QAEvC,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,EAAE;YACvB,WAAW,CAAC,KAAK,CAAC,WAAW,CAAC,eAAe,EAAE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YACnE,WAAW,CAAC,KAAK,CAAC,WAAW,CAAC,mBAAmB,EAAE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;SACxE;QACD,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,EAAE;YACzB,WAAW,CAAC,KAAK,CAAC,WAAW,CAAC,oBAAoB,EAAE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YAC1E,WAAW,CAAC,KAAK,CAAC,WAAW,CAC3B,qBAAqB,EACrB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CACrB,CAAC;SACH;QAED,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,WAAW,GAAG,WAAW,CAAC,gBAAgB,CAC9C,oFAAoF,CACrF,CAAC;YACF,WAAW,CAAC,OAAO,CAAC,CAAC,IAAiB,EAAE,EAAE;gBACxC,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,EAAE;oBACvB,IAAI,CAAC,KAAK,CAAC,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;iBACjD;gBACD,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,EAAE;oBACzB,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC;iBACzC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,EAAE,GAAG,CAAC,CAAC;IACV,CAAC;IAEM,KAAK,CAAC,QAAQ;QACnB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE;YACpB,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG;gBAChB,EAAE,EAAE,cAAc,CAAC,gBAAgB;gBACnC,IAAI,EAAE,cAAc,CAAC,kBAAkB;aACxC,CAAC;SACH;QACD,IAAI,IAAI,CAAC,IAAI,CAAC,gBAAgB,KAAK,SAAS,EAAE;YAC5C,IAAI,CAAC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;SACnC;QAED,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,GAAG,CACpD,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,CAC5B,CAAC;QACF,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAExB,IAAI,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE;YAC/D,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,SAAS,GAAG,CAAC,IAAQ,EAAE,EAAE,CAAC,IAAI,CAAC;SACvD;QAED,iDAAiD;QACjD,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,uBAAuB,CAC7D,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAC3B,CAAC;QACF,IAAI,CAAC,aAAa,GAAG,YAAY,CAAC,aAAa,CAAC;QAChD,IAAI,CAAC,qBAAqB,GAAG,YAAY,CAAC,qBAAqB,CAAC;QAChE,IAAI,CAAC,mBAAmB,GAAG,YAAY,CAAC,mBAAmB,CAAC;QAC5D,IAAI,CAAC,iBAAiB,GAAG,YAAY,CAAC,iBAAiB,CAAC;QAExD,IAAI,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE;YAC/B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,oBAAoB,CAC1D,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAC5B,CAAC;SACH;QAED,gBAAgB;QAChB,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE;YAClC,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;YACvB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;QAED,gBAAgB;QAChB,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,IAAI,EAAE;YACjC,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;gBACpB,IAAI,CAAC,MAAM,GAAG;oBACZ,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK;oBAC7B,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK;iBAC9B,CAAC;aACH;YACD,IAAI,CAAC,aAAa;iBACf,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBACjE,SAAS,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YAC1C,CAAC,CAAC,CAAC;YAEL,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC3B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;YACxB,MAAM,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YAE9C,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC;YAC/B,IAAI,CAAC,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC;YAC7B,IAAI,CAAC,UAAU,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;YAC3C,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;YACjC,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;YACpB,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;gBACtB,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;oBACzC,MAAM,WAAW,GAAG,MAAM,QAAQ,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC;oBAC7C,MAAM,OAAO,GAAG,WAAW,CAAC,IAAI,EAEnB,CAAC;oBACd,IAAI,OAAO,IAAI,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;wBACtC,MAAM,UAAU,GAAG,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;wBAC3C,IAAI,CAAC,UAAU,GAAG,CAAC,IAAI,CAAC,UAAU;4BAChC,CAAC,OAAO,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAW,CAAC;qBAChE;iBACF;aACF;YACD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;IACH,CAAC;IAED,uBAAuB;IAEf,KAAK,CAAC,SAAS;QACrB,IAAI,CAAC,iBAAiB,CAAC,KAAK,EAAE,CAAC;QAC/B,IAAI,CAAC,KAAK,GAAG,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,QAAQ,CAC5C,IAAI,CAAC,IAAI,CAAC,aAAa,CACxB,CAAQ,CAAC;QACV,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YACxB,IAAI,CAAC,WAAW,EAAE,CAAC;SACpB;QACD,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;QAC3B,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC9B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;QAEpC,IAAI,CAAC,OAAO,GAAG;YACb,OAAO,EAAE,EAAE;YACX,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI;gBAC1B,KAAK,EAAE,cAAc,CAAC,UAAU;gBAChC,KAAK,EAAE,cAAc,CAAC,UAAU;aACjC;SACF,CAAC;QACF,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI;YAChC,KAAK,EAAE,cAAc,CAAC,UAAU;YAChC,KAAK,EAAE,cAAc,CAAC,UAAU;SACjC,CAAC;QAEF,IAAI,cAAc,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,cAAc,GAAG,IAAI,CAAC,sBAAsB,CAAC,cAAc,CAAC,CAAC;QAC7D,IAAI,CAAC,aAAa,GAAG,cAAc,CAAC;QAEpC,IAAI,CAAC,UAAU,GAAG,IAAI,kBAAkB,CAAC,cAAc,CAAC,CAAC;QACzD,IAAI,CAAC,UAAU,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;QAC3C,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QACjC,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;YACpB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC;YACrD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC;YACxD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;SACxC;QAED,IAAI,CAAC,UAAU,CAAC,eAAe,GAAG,CAAC,IAAI,EAAE,MAAc,EAAE,EAAE;YACzD,MAAM,IAAI,GAAG,IAA+B,CAAC;YAC7C,OAAO,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC,GAAG,EAAW,EAAE;gBACtD,IAAI,GAAG,CAAC,gBAAgB,EAAE;oBACxB,OAAO,GAAG,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC,SAAS,EAAW,EAAE;wBACtD,MAAM,aAAa,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;wBACzC,IAAI,CAAC,aAAa,IAAI,OAAO,aAAa,KAAK,QAAQ,EAAE;4BACvD,OAAO,KAAK,CAAC;yBACd;wBAED,MAAM,cAAc,GAAG,aAAwC,CAAC;wBAChE,MAAM,cAAc,GAAG,cAAc,CAAC,SAAS,CAAC,CAAC;wBACjD,IAAI,cAAc,KAAK,IAAI,IAAI,cAAc,KAAK,SAAS,EAAE;4BAC3D,OAAO,KAAK,CAAC;yBACd;wBAED,OAAO,MAAM,CAAC,cAAc,CAAC;6BAC1B,IAAI,EAAE;6BACN,iBAAiB,EAAE;6BACnB,QAAQ,CAAC,MAAM,CAAC,CAAC;oBACtB,CAAC,CAAC,CAAC;iBACJ;gBAED,IAAI,GAAG,CAAC,QAAQ,IAAI,GAAG,CAAC,YAAY,EAAE;oBACpC,MAAM,aAAa,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;oBACzC,IAAI,aAAa,KAAK,IAAI,IAAI,aAAa,KAAK,SAAS,EAAE;wBACzD,OAAO,KAAK,CAAC;qBACd;oBAED,OAAO,MAAM,CAAC,aAAa,CAAC;yBACzB,IAAI,EAAE;yBACN,iBAAiB,EAAE;yBACnB,QAAQ,CAAC,MAAM,CAAC,CAAC;iBACrB;gBACD,OAAO,KAAK,CAAC;YACf,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;QACF,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC;QACvD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEO,KAAK,CAAC,kBAAkB,CAC9B,aAAgD,QAAQ,EACxD,QAAiB,KAAK;QAEtB,IAAI,CAAC,iBAAiB,CAAC,KAAK,EAAE,CAAC;QAC/B,IAAI,KAAK,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;YACvD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;YACpB,IAAI,CAAC,sBAAsB,GAAG,CAAC,CAAC;SACjC;QACD,IAAI,KAAK,IAAI,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;YACxD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;YACrB,IAAI,CAAC,sBAAsB,GAAG,CAAC,CAAC;SACjC;QAED,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,oBAAoB,CAC3D,IAAI,CAAC,WAAW,CACjB,CAAC;QAEF,IAAI,CAAC,OAAO,GAAG;YACb,OAAO,EAAE,aAAa;YACtB,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC;QAEF,MAAM,SAAS,GAAe;YAC5B,SAAS,EAAE,IAAI,CAAC,QAAQ;YACxB,UAAU,EAAE,IAAI,CAAC,IAAI,CAAC,UAAU;YAChC,GAAG,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE;YACvD,UAAU;YACV,OAAO,EAAE,IAAI,CAAC,OAAQ;YACtB,UAAU,EAAE,IAAI,CAAC,IAAI,CAAC,UAAU;YAChC,IAAI,EAAE,IAAI,CAAC,UAAU;YACrB,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ;YAC5B,eAAe,EAAE,IAAI,CAAC,sBAAsB;SAC7C,CAAC;QAEF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;QAC/D,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,KAAY,CAAC;QACjC,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;QAC3B,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC9B,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,OAA0C,CAAC;QACjE,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,QAA2C,CAAC;QAEnE,IAAI,MAAM,CAAC,sBAAsB,KAAK,SAAS,EAAE;YAC/C,IAAI,CAAC,sBAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC;SAC7D;QAED,IAAI,GAAG,GAAG,CAAC,CAAC;QACZ,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YACtB,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;gBACzC,MAAM,WAAW,GAAG,MAAM,QAAQ,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC;gBAC7C,MAAM,OAAO,GAAG,WAAW,CAAC,IAAI,EAEnB,CAAC;gBACd,IAAI,OAAO,IAAI,MAAM,CAAC,YAAY,EAAE;oBAClC,MAAM,UAAU,GAAG,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;oBACjE,MAAM,YAAY,GAAG,OAAO,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;oBACrE,GAAG,GAAG,MAAM,CAAC,YAAY,IAAK,CAAC,GAAG,GAAG,YAAY,CAAY,CAAC;iBAC/D;aACF;YACD,IAAI,CAAC,UAAU,GAAG,GAAG,CAAC;SACvB;QAED,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC,WAAW,CAAC;QACtC,IAAI,CAAC,UAAU,GAAG,IAAI,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC;QACvD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,qBAAqB;IAErB,KAAK,CAAC,YAAY,CAAC,KAAiB;QAClC,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,IAAI,IAAI,CAAC,KAAK;YAAE,OAAO;QAEpD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE3B,MAAM,YAAY,GAAG;YACnB,iBAAiB,EAAE,IAAI,CAAC,iBAAiB;YACzC,sBAAsB,EAAE,IAAI,CAAC,sBAAsB;YACnD,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,WAAW,EAAE,IAAI,CAAC,WAAW;SAC9B,CAAC;QAEF,MAAM,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,CAC3D,KAAK,EACL,YAAY,CACb,CAAC;QAEF,IAAI,CAAC,UAAU,CAAC,cAAc,EAAE;YAC9B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5B,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;YACvB,OAAO;SACR;QAED,yBAAyB;QACzB,IAAI,CAAC,sBAAsB,GAAG,UAAU,CAAC,kBAAkB,CAAC;QAC5D,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAC,aAAa,CAAC;QAElD,8BAA8B;QAC9B,IAAI,KAAK,CAAC,QAAQ,KAAK,IAAI,CAAC,QAAQ,EAAE;YACpC,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;SAChC;QAED,oEAAoE;QACpE,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,sBAAsB,CACjE,IAAI,CAAC,UAAU,EACf,KAAK,CAAC,QAAQ,CACf,CAAC;QACF,IAAI,gBAAgB,GAAkB,IAAI,CAAC;QAE3C,IACE,KAAK,CAAC,SAAS,KAAK,aAAa;YACjC,UAAU,CAAC,SAAS,KAAK,UAAU;YACnC,UAAU,CAAC,SAAS,EACpB;YACA,MAAM,uBAAuB,GAC3B,IAAI,CAAC,iBAAiB,CAAC,0BAA0B,CAC/C,IAAI,CAAC,UAAU,EACf,KAAK,CAAC,QAAQ,CACf,CAAC;YAEJ,IACE,uBAAuB,GAAG,CAAC;gBAC3B,uBAAuB,GAAG,KAAK,CAAC,QAAQ,EACxC;gBACA,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC;gBACjC,IAAI,CAAC,QAAQ,GAAG,uBAAuB,CAAC;aACzC;SACF;QAED,IAAI;YACF,MAAM,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,SAAS,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC;SAC3E;QAAC,OAAO,KAAK,EAAE;YACd,OAAO,CAAC,KAAK,CAAC,mCAAmC,EAAE,KAAK,CAAC,CAAC;SAC3D;gBAAS;YACR,IAAI,gBAAgB,KAAK,IAAI,EAAE;gBAC7B,IAAI,CAAC,QAAQ,GAAG,gBAAgB,CAAC;aAClC;SACF;QAED,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,wBAAwB;IAExB,WAAW,CAAC,KAAa;QACvB,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE;YAClC,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;SAC7D;QACD,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,IAAI,EAAE;YACjC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;YACzB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;SAC3C;IACH,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,KAAa;QACxB,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,cAAc,EAAE,CAAC;YACvB,KAAK,CAAC,eAAe,EAAE,CAAC;SACzB;QAED,IAAI,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE;YACzB,IAAI,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,OAAO,KAAK,WAAW,EAAE;gBACjD,IAAI,CAAC,MAAM,GAAG;oBACZ,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,QAAQ;oBACrC,KAAK,EAAE,KAAK;iBACb,CAAC;aACH;YACD,IAAI,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,OAAO,KAAK,YAAY,EAAE;gBAClD,IAAI,CAAC,MAAM,GAAG;oBACZ,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,QAAQ;oBACrC,KAAK,EAAE,MAAM;iBACd,CAAC;aACH;SACF;QAED,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE;YAClC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC;YAC9C,IAAI,CAAC,wBAAwB,EAAE,CAAC;YAChC,IAAI,CAAC,cAAc;gBACjB,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC;oBACzB,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,KAAK,EAAE,OAAO,IAAI,EAAE,CAAC;oBACpE,CAAC,CAAC,EAAE,CAAC;YACT,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;aAAM;YACL,MAAM,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YAC9C,IAAI,CAAC,cAAc;gBACjB,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC;oBACzB,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,KAAK,EAAE,OAAO,IAAI,EAAE,CAAC;oBACpE,CAAC,CAAC,EAAE,CAAC;YACT,IAAI,CAAC,SAAS,CAAC,SAAS,EAAE,CAAC;YAC3B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;IACH,CAAC;IAED,KAAK,CAAC,WAAW;QACf,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,EAAE,CAAC;QAC5B,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,IAAI,CAAC,UAAU,CAAC,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC;SACxD;QAED,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,CAAC;QACzB,IAAI,CAAC,SAAS,EAAE,CAAC;QACjB,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;QAC/B,IAAI,CAAC,MAAM,GAAG;YACZ,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM;YACzD,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,WAAW;SAC/D,CAAC;QAEF,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE;YAClC,IAAI,CAAC,OAAO,GAAG;gBACb,OAAO,EAAE,EAAE;gBACX,MAAM,EAAE,IAAI,CAAC,MAAM;aACpB,CAAC;YACF,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC;YACvC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;YACpC,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;YACzB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;aAAM;YACL,MAAM,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YAC9C,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;YACzB,IAAI,CAAC,SAAS,CAAC,SAAS,EAAE,CAAC;YAC3B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;IACH,CAAC;IAED,qBAAqB;IAErB,WAAW,CAAC,GAAM;QAChB,IAAI,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YAC5B,OAAO;SACR;QACD,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC;QAChD,MAAM,SAAS,GAAG,GAA+C,CAAC;QAClE,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAClC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,IAAI,OAAO,EAAE,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC,CACzD,CAAC;QACF,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;IAC7B,CAAC;IAED,mBAAmB;IAEnB,KAAK,CAAC,WAAW,CAAC,MAIjB;QACC,IAAI;YACF,IAAI,QAAyD,CAAC;YAC9D,IACE,MAAM,CAAC,EAAE,KAAK,EAAE;gBAChB,MAAM,CAAC,EAAE,KAAK,SAAS;gBACvB,MAAM,CAAC,UAAU,KAAK,SAAS;gBAC/B,MAAM,CAAC,UAAU,KAAK,EAAE,EACxB;gBACA,QAAQ,GAAG,MAAM,cAAc,CAC7B,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,GAAG,EAAE,CAClE,CAAC;aACH;YACD,IAAI,QAAQ,IAAI,QAAQ,CAAC,MAAM,EAAE;gBAC/B,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,EAAyC,CAAC;gBACpE,MAAM,KAAK,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;gBACzC,OAAO,OAAO,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC;aAC/C;YACD,OAAO,EAAE,CAAC;SACX;QAAC,OAAO,CAAC,EAAE;YACV,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACjB,OAAO,EAAE,CAAC;SACX;IACH,CAAC;IAEO,KAAK,CAAC,aAAa;QACzB,MAAM,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,gBAAgB;aAChD,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC;aAC7B,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE,CACf,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE;YAC5B,IAAI,GAAG,CAAC,QAAQ,EAAE;gBAChB,MAAM,UAAU,GAAG,IAA+B,CAAC;gBACnD,UAAU,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC;oBAChD,EAAE,EAAE,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC;oBACzD,UAAU,EAAE,GAAG,CAAC,QAAQ,CAAC,kBAAkB;oBAC3C,WAAW,EAAE,GAAG,CAAC,QAAQ,CAAC,gBAAgB,IAAI,GAAG,CAAC,QAAQ;iBAC3D,CAAC,CAAC;aACJ;QACH,CAAC,CAAC,CACH,CAAC;QAEJ,MAAM,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,cAAc,CAAC,MAQpB;QACC,MAAM,UAAU,GAAG,MAAM,CAAC,IAA+B,CAAC;QAC1D,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,SAAS;aAClC,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC;aACtC,GAAG,CAAC,KAAK,CACR,MAAM,CAAC,QAAQ,CAAC,QAAQ,EACxB,MAAM,CAAC,QAAQ,CAAC,QAAQ,EACxB,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAClC;aACA,GAAG,EAAE,CAAC;QACT,OAAO,QAAQ,CAAC,IAAI,CAAC;IACvB,CAAC;IAEO,KAAK,CAAC,gBAAgB;QAC5B,MAAM,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,gBAAgB;aAC9C,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC;aAChC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE,CACf,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE;YAC5B,IAAI,GAAG,CAAC,WAAW,EAAE;gBACnB,MAAM,UAAU,GAAG,IAA+B,CAAC;gBACnD,UAAU,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC;oBACnD,IAAI,EAAE,IAAI;oBACV,QAAQ,EAAE,GAAG,CAAC,WAAW;iBAC1B,CAAC,CAAC;aACJ;QACH,CAAC,CAAC,CACH,CAAC;QACJ,MAAM,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;IACpC,CAAC;IAEO,WAAW;QACjB,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YACxB,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;gBACpC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;oBACtC,MAAM,gBAAgB,GACpB,IAAI,CAAC,YAAY,CAAC,SAAS,CACzB,IAAI,CAAC,QAAoD,CAC1D,CAAC;oBACJ,IAAI,gBAAgB,EAAE;wBACpB,MAAM,UAAU,GAAG,IAA+B,CAAC;wBACnD,OAAO,gBAAgB,CACrB,UAAU,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAClC,IAAI,CAAC,YAAY,CAClB,CAAC;qBACH;oBACD,OAAO,KAAK,CAAC;gBACf,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAED,iBAAiB;IAEjB,KAAK,CAAC,WAAW;QACf,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YACxB,MAAM,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YAC9C,IAAI,CAAC,SAAS,CAAC,SAAS,EAAE,CAAC;YAC3B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;aAAM;YACL,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;SACxB;IACH,CAAC;IAED,sBAAsB;QACpB,IAAI,IAAI,CAAC,UAAU,EAAE;YACnB,IAAI,CAAC,UAAU,GAAG,IAAI,kBAAkB,CAAI,EAAE,CAAC,CAAC;SACjD;QAED,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,GAAG,CACpD,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,CAC5B,CAAC;QAEF,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,uBAAuB,CAC7D,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAC3B,CAAC;QACF,IAAI,CAAC,aAAa,GAAG,YAAY,CAAC,aAAa,CAAC;QAChD,IAAI,CAAC,qBAAqB,GAAG,YAAY,CAAC,qBAAqB,CAAC;QAEhE,IAAI,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE;YAC/B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,oBAAoB,CAC1D,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAC5B,CAAC;SACH;QACD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,YAAY,EAAE;YAC5B,OAAO,KAAK,CAAC;SACd;QAED,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE;YACrC,OAAO,IAAI,CAAC;SACb;QAED,IAAI;YACF,OAAO,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC;SACvD;QAAC,MAAM;YACN,OAAO,IAAI,CAAC;SACb;IACH,CAAC;IAED,cAAc;QACZ,IAAI,CAAC,IAAI,CAAC,aAAa;YAAE,OAAO;QAEhC,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE;YAClC,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,CAAC,YAAY,EAAE;gBACnD,IAAI,CAAC,aAAa,GAAG,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;aACxD;YAED,MAAM,OAAO,GAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC;YAC/C,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC;SACzD;aAAM;YACL,IAAI,IAAI,CAAC,OAAO,EAAE;gBAChB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC;aAC9D;SACF;IACH,CAAC;;4GA14BU,cAAc;gGAAd,cAAc,sJA6Cd,YAAY,uEACZ,OAAO,qEC5GpB,k3/BAwxBA;4FD1tBa,cAAc;kBAN1B,SAAS;+BACE,WAAW,mBAGJ,uBAAuB,CAAC,MAAM;ySAMtC,IAAI;sBAAZ,KAAK;gBACG,aAAa;sBAArB,KAAK;gBAwCmB,SAAS;sBAAjC,SAAS;uBAAC,YAAY;gBACH,IAAI;sBAAvB,SAAS;uBAAC,OAAO","sourcesContent":["import {\r\n  Arrange,\r\n  Condition,\r\n  TabData,\r\n  TableData,\r\n  Column,\r\n  TABLE_DEFAULTS,\r\n  DropdownItem,\r\n  FilterSelectItem,\r\n  Pagination,\r\n} from '../../types/Table';\r\nimport {\r\n  Component,\r\n  Input,\r\n  OnInit,\r\n  ViewChild,\r\n  AfterViewInit,\r\n  OnChanges,\r\n  SimpleChanges,\r\n  ElementRef,\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  OnDestroy,\r\n} from '@angular/core';\r\nimport {\r\n  AngularFirestore,\r\n  QueryDocumentSnapshot,\r\n} from '@angular/fire/compat/firestore';\r\nimport { MatPaginator, PageEvent } from '@angular/material/paginator';\r\nimport { MatSort } from '@angular/material/sort';\r\nimport { MatTableDataSource } from '@angular/material/table';\r\nimport { Router } from '@angular/router';\r\nimport { TableService } from '../../services/table.service';\r\nimport { FilterService } from '../../services/filter.service';\r\nimport { TooltipService, TooltipState } from '../../services/tooltip.service';\r\nimport { PaginationService } from '../../services/pagination.service';\r\nimport {\r\n  getDisplayValue,\r\n  getNestedValue,\r\n  isString,\r\n  trackByProperty,\r\n  getImageKey,\r\n} from '../../utils/table.utils';\r\nimport {\r\n  debounceTime,\r\n  firstValueFrom,\r\n  Subject,\r\n  takeUntil,\r\n  BehaviorSubject,\r\n} from 'rxjs';\r\nimport { OrderByDirection } from 'firebase/firestore';\r\nimport firebase from 'firebase/compat';\r\nimport WhereFilterOp = firebase.firestore.WhereFilterOp;\r\nimport firestore = firebase.firestore;\r\nimport { FormControl, FormArray, FormGroup } from '@angular/forms';\r\n\r\n@Component({\r\n  selector: 'lib-table',\r\n  templateUrl: './table.component.html',\r\n  styleUrls: ['./table.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n})\r\nexport class TableComponent<T = any>\r\n  implements OnInit, AfterViewInit, OnChanges, OnDestroy\r\n{\r\n  // INPUTS\r\n  @Input() data!: TableData;\r\n  @Input() downloadTable?: (arrange: Arrange, conditions: Condition[]) => void;\r\n  arrange: Arrange | null = null;\r\n\r\n  // VARIABLES\r\n  dataSource!: MatTableDataSource<T>;\r\n  currentPageNumber = 1;\r\n  currentClientPageIndex = 0;\r\n\r\n  public isLoading$ = new BehaviorSubject<boolean>(false);\r\n\r\n  public items: T[] = [];\r\n  public filteredItems: T[] = [];\r\n  private lastDoc: QueryDocumentSnapshot<T> | null = null;\r\n  private firstDoc: QueryDocumentSnapshot<T> | null = null;\r\n  private sortBy: { field: string; order: OrderByDirection } = {\r\n    field: TABLE_DEFAULTS.SORT_FIELD,\r\n    order: TABLE_DEFAULTS.SORT_ORDER,\r\n  };\r\n  public columnProperties: string[] = [];\r\n\r\n  filtersForm: FormArray;\r\n  selectSort: FormControl = new FormControl('');\r\n  currentArrange = '';\r\n\r\n  hasNextPage: boolean = false;\r\n  dropdownItems: DropdownItem[] = [];\r\n  sortableDropdownItems: DropdownItem[] = [];\r\n  pageSize = TABLE_DEFAULTS.PAGE_SIZE;\r\n  totalItems = 0;\r\n  pageEvent?: PageEvent;\r\n  filterValue: string | null = null;\r\n\r\n  hasFilterableColumn: boolean = false;\r\n  hasSortableColumn: boolean = false;\r\n  filterPredicate: ((data: T, filter: string) => boolean) | undefined;\r\n  private filterSubject = new Subject<string>();\r\n  private readonly debounceTimeMs = TABLE_DEFAULTS.DEBOUNCE_TIME_MS;\r\n\r\n  selectedTab: number = 0;\r\n\r\n  @ViewChild(MatPaginator) paginator!: MatPaginator;\r\n  @ViewChild(MatSort) sort!: MatSort;\r\n\r\n  // Propriedades para controle do tooltip\r\n  tooltipState: TooltipState;\r\n\r\n  // Controle de loading de imagens\r\n  imageLoadingState = new Map<string, boolean>();\r\n\r\n  private destroy$ = new Subject<void>();\r\n\r\n  // CONSTRUCTOR\r\n  constructor(\r\n    private router: Router,\r\n    private tableService: TableService,\r\n    private filterService: FilterService,\r\n    private tooltipService: TooltipService,\r\n    private paginationService: PaginationService,\r\n    private firestore: AngularFirestore,\r\n    private el: ElementRef,\r\n    private cdr: ChangeDetectorRef,\r\n  ) {\r\n    this.filtersForm = new FormArray([this.filterService.createFilterGroup()]);\r\n    this.tooltipState = this.tooltipService.createInitialState();\r\n  }\r\n\r\n  // FILTER METHODS (delegados ao FilterService)\r\n\r\n  createFilterGroup(): FormGroup {\r\n    return this.filterService.createFilterGroup();\r\n  }\r\n\r\n  addFilter(filterData?: {\r\n    selectFilter: DropdownItem | null;\r\n    typeFilter?: string;\r\n    selectItem?: FilterSelectItem[];\r\n    initialDate?: string;\r\n    finalDate?: string;\r\n  }): void {\r\n    this.filterService.addFilter(\r\n      this.filtersForm,\r\n      this.data.pagination === true,\r\n      filterData,\r\n    );\r\n  }\r\n\r\n  hasActiveDateFilter(): boolean {\r\n    return this.filterService.hasActiveDateFilter(this.filtersForm);\r\n  }\r\n\r\n  getAvailableFilterOptions(): DropdownItem[] {\r\n    return this.filterService.getAvailableFilterOptions(\r\n      this.dropdownItems,\r\n      this.data.pagination === true,\r\n      this.filtersForm,\r\n    );\r\n  }\r\n\r\n  onSelectFilterChange(): void {\r\n    const lastIndex = this.filtersForm.length - 1;\r\n    const lastFilter = this.filtersForm.at(lastIndex);\r\n    const selectedFilter = lastFilter.get('selectFilter')?.value;\r\n\r\n    if (selectedFilter) {\r\n      if (\r\n        selectedFilter.arrange === 'filterByDate' &&\r\n        this.data.pagination === true\r\n      ) {\r\n        return;\r\n      }\r\n      this.addFilter();\r\n    }\r\n  }\r\n\r\n  removeFilter(index: number): void {\r\n    this.filterService.removeFilter(this.filtersForm, index);\r\n  }\r\n\r\n  removeAllFilters(): void {\r\n    this.filterService.removeAllFilters(this.filtersForm);\r\n    this.resetFilter();\r\n  }\r\n\r\n  onDateInput(\r\n    event: Event,\r\n    controlName: 'initialDate' | 'finalDate',\r\n    filterIndex: number,\r\n  ): void {\r\n    const filterGroup = this.filtersForm.at(filterIndex) as FormGroup;\r\n    this.filterService.applyDateMask(event, controlName, filterGroup);\r\n  }\r\n\r\n  onDateFilterChange(): void {\r\n    if (this.data.pagination === false) {\r\n      this.arrange = this.filterService.buildArrangeFromFilters(\r\n        this.filtersForm,\r\n        this.sortBy,\r\n      );\r\n      this.applyFiltersToDataSource();\r\n      this.cdr.markForCheck();\r\n    }\r\n  }\r\n\r\n  private applyFiltersToDataSource(): void {\r\n    if (!this.dataSource) return;\r\n\r\n    let filteredItems = this.filterService.applyClientSideFilters<T>(\r\n      [...this.items],\r\n      this.filtersForm,\r\n      this.sortBy,\r\n    );\r\n    this.dataSource.data = filteredItems;\r\n    this.filteredItems = filteredItems;\r\n    this.totalItems = filteredItems.length;\r\n\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  private applyClientSideFilters(items: T[]): T[] {\r\n    return this.filterService.applyClientSideFilters<T>(\r\n      items,\r\n      this.filtersForm,\r\n      this.sortBy,\r\n    );\r\n  }\r\n\r\n  private buildArrangeFromFilters(): Arrange {\r\n    return this.filterService.buildArrangeFromFilters(\r\n      this.filtersForm,\r\n      this.sortBy,\r\n    );\r\n  }\r\n\r\n  // TOOLTIP METHODS (delegados ao TooltipService)\r\n\r\n  getTooltipStyles(): { [key: string]: string } {\r\n    return this.tooltipService.getTooltipStyles(this.tooltipState);\r\n  }\r\n\r\n  onCellMouseEnter(event: MouseEvent, row: T, col: Column): void {\r\n    this.tooltipService.handleMouseEnter(\r\n      event,\r\n      row,\r\n      col,\r\n      (c, r, w) => this.getDisplayValue(c, r, w),\r\n      (state) => {\r\n        this.tooltipState = { ...this.tooltipState, ...state };\r\n        this.cdr.markForCheck();\r\n      },\r\n    );\r\n  }\r\n\r\n  onCellMouseLeave(): void {\r\n    this.tooltipService.handleMouseLeave((state) => {\r\n      this.tooltipState = state;\r\n      this.cdr.markForCheck();\r\n    });\r\n  }\r\n\r\n  onCellMouseMove(event: MouseEvent): void {\r\n    this.tooltipService.handleMouseMove(\r\n      event,\r\n      this.tooltipState.showTooltip,\r\n      (position) => {\r\n        this.tooltipState = { ...this.tooltipState, tooltipPosition: position };\r\n        this.cdr.markForCheck();\r\n      },\r\n    );\r\n  }\r\n\r\n  // TAB METHODS\r\n\r\n  onTabSelected(event: { tab: TabData; index: number }): void {\r\n    if (!this.data.tabs?.method) return;\r\n    this.selectedTab = event.index;\r\n    this.data.tabs.method(event.tab, event.index);\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  // UTILITY METHODS (delegados aos utils)\r\n\r\n  getDisplayValue(col: Column, row: T, withinLimit: boolean = false): string {\r\n    return getDisplayValue(col, row, withinLimit);\r\n  }\r\n\r\n  getNestedValue(obj: unknown, path: string): unknown {\r\n    return getNestedValue(obj, path);\r\n  }\r\n\r\n  isString(value: unknown): boolean {\r\n    return isString(value);\r\n  }\r\n\r\n  trackByProperty(_index: number, col: Column): string {\r\n    return trackByProperty(_index, col);\r\n  }\r\n\r\n  // IMAGE LOADING METHODS\r\n\r\n  private getImageKey(row: T, col: Column): string {\r\n    return getImageKey(row, col);\r\n  }\r\n\r\n  isImageLoading(row: T, col: Column): boolean {\r\n    const key = this.getImageKey(row, col);\r\n    return !this.imageLoadingState.has(key);\r\n  }\r\n\r\n  onImageLoad(row: T, col: Column): void {\r\n    const key = this.getImageKey(row, col);\r\n    this.imageLoadingState.set(key, true);\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  onImageError(event: Event, row: T, col: Column, defaultImage?: string): void {\r\n    const key = this.getImageKey(row, col);\r\n    this.imageLoadingState.set(key, true);\r\n\r\n    const target = event.target as HTMLImageElement;\r\n    if (target) {\r\n      if (defaultImage) {\r\n        target.src = defaultImage;\r\n        target.onerror = null;\r\n      } else {\r\n        target.style.display = 'none';\r\n      }\r\n    }\r\n\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  // LIFECYCLE METHODS\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['data']) {\r\n      setTimeout(() => this.applyCustomColors(), 0);\r\n    }\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    this.applyCustomColors();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.tooltipService.destroy();\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  private applyCustomColors(): void {\r\n    const hostElement = this.el?.nativeElement;\r\n    if (!hostElement || !this.data) return;\r\n\r\n    if (this.data.color?.bg) {\r\n      hostElement.style.setProperty('--table-color', this.data.color.bg);\r\n      hostElement.style.setProperty('--table-header-bg', this.data.color.bg);\r\n    }\r\n    if (this.data.color?.text) {\r\n      hostElement.style.setProperty('--table-text-color', this.data.color.text);\r\n      hostElement.style.setProperty(\r\n        '--table-header-text',\r\n        this.data.color.text,\r\n      );\r\n    }\r\n\r\n    setTimeout(() => {\r\n      const headerCells = hostElement.querySelectorAll(\r\n        '.mat-header-cell, .mat-mdc-header-cell, th.mat-header-cell, th.mat-mdc-header-cell',\r\n      );\r\n      headerCells.forEach((cell: HTMLElement) => {\r\n        if (this.data.color?.bg) {\r\n          cell.style.backgroundColor = this.data.color.bg;\r\n        }\r\n        if (this.data.color?.text) {\r\n          cell.style.color = this.data.color.text;\r\n        }\r\n      });\r\n    }, 100);\r\n  }\r\n\r\n  public async ngOnInit(): Promise<void> {\r\n    if (!this.data.color) {\r\n      this.data.color = {\r\n        bg: TABLE_DEFAULTS.DEFAULT_BG_COLOR,\r\n        text: TABLE_DEFAULTS.DEFAULT_TEXT_COLOR,\r\n      };\r\n    }\r\n    if (this.data.showSimpleSearch === undefined) {\r\n      this.data.showSimpleSearch = true;\r\n    }\r\n\r\n    this.columnProperties = this.data.displayedColumns.map(\r\n      (column) => column.property,\r\n    );\r\n    this.cdr.markForCheck();\r\n\r\n    if (this.data.actionButton && !this.data.actionButton.condition) {\r\n      this.data.actionButton.condition = (_row?: T) => true;\r\n    }\r\n\r\n    // Inicializar dropdownItems usando FilterService\r\n    const filterConfig = this.filterService.initializeDropdownItems(\r\n      this.data.displayedColumns,\r\n    );\r\n    this.dropdownItems = filterConfig.dropdownItems;\r\n    this.sortableDropdownItems = filterConfig.sortableDropdownItems;\r\n    this.hasFilterableColumn = filterConfig.hasFilterableColumn;\r\n    this.hasSortableColumn = filterConfig.hasSortableColumn;\r\n\r\n    if (this.data.filterableOptions) {\r\n      this.dropdownItems = this.filterService.addFilterableOptions(\r\n        this.dropdownItems,\r\n        this.data.filterableOptions,\r\n      );\r\n    }\r\n\r\n    // Sem paginação\r\n    if (this.data.pagination === false) {\r\n      await this.loadItems();\r\n      this.cdr.markForCheck();\r\n    }\r\n\r\n    // Com paginação\r\n    if (this.data.pagination === true) {\r\n      if (this.data.sortBy) {\r\n        this.sortBy = {\r\n          field: this.data.sortBy.field,\r\n          order: this.data.sortBy.order,\r\n        };\r\n      }\r\n      this.filterSubject\r\n        .pipe(debounceTime(this.debounceTimeMs), takeUntil(this.destroy$))\r\n        .subscribe(() => {\r\n          this.loadItemsPaginated('reload', true);\r\n        });\r\n\r\n      this.isLoading$.next(true);\r\n      this.cdr.markForCheck();\r\n      await this.loadItemsPaginated('reload', true);\r\n\r\n      this.sort.active = 'createdAt';\r\n      this.sort.direction = 'desc';\r\n      this.dataSource.paginator = this.paginator;\r\n      this.dataSource.sort = this.sort;\r\n      this.totalItems = 0;\r\n      if (this.data.totalRef) {\r\n        for (const totalRef of this.data.totalRef) {\r\n          const totalRefDoc = await totalRef.ref.get();\r\n          const docData = totalRefDoc.data() as\r\n            | Record<string, unknown>\r\n            | undefined;\r\n          if (docData && docData[totalRef.field]) {\r\n            const fieldValue = docData[totalRef.field];\r\n            this.totalItems = (this.totalItems +\r\n              (typeof fieldValue === 'number' ? fieldValue : 0)) as number;\r\n          }\r\n        }\r\n      }\r\n      this.isLoading$.next(false);\r\n      this.cdr.markForCheck();\r\n    }\r\n  }\r\n\r\n  // DATA LOADING METHODS\r\n\r\n  private async loadItems(): Promise<void> {\r\n    this.imageLoadingState.clear();\r\n    this.items = (await this.tableService.getItems(\r\n      this.data.collectionRef,\r\n    )) as T[];\r\n    if (this.data.conditions) {\r\n      this.filterItems();\r\n    }\r\n    await this.loadRelations();\r\n    await this.loadQueryLengths();\r\n    this.totalItems = this.items.length;\r\n\r\n    this.arrange = {\r\n      filters: [],\r\n      sortBy: this.data.sortBy || {\r\n        field: TABLE_DEFAULTS.SORT_FIELD,\r\n        order: TABLE_DEFAULTS.SORT_ORDER,\r\n      },\r\n    };\r\n    this.sortBy = this.data.sortBy || {\r\n      field: TABLE_DEFAULTS.SORT_FIELD,\r\n      order: TABLE_DEFAULTS.SORT_ORDER,\r\n    };\r\n\r\n    let itemsToDisplay = [...this.items];\r\n    itemsToDisplay = this.applyClientSideFilters(itemsToDisplay);\r\n    this.filteredItems = itemsToDisplay;\r\n\r\n    this.dataSource = new MatTableDataSource(itemsToDisplay);\r\n    this.dataSource.paginator = this.paginator;\r\n    this.dataSource.sort = this.sort;\r\n    if (this.data.sortBy) {\r\n      this.dataSource.sort.active = this.data.sortBy.field;\r\n      this.dataSource.sort.direction = this.data.sortBy.order;\r\n      this.dataSource.sort.sortChange.emit();\r\n    }\r\n\r\n    this.dataSource.filterPredicate = (data, filter: string) => {\r\n      const item = data as Record<string, unknown>;\r\n      return this.data.displayedColumns.some((col): boolean => {\r\n        if (col.filterPredicates) {\r\n          return col.filterPredicates.some((predicate): boolean => {\r\n            const propertyValue = item[col.property];\r\n            if (!propertyValue || typeof propertyValue !== 'object') {\r\n              return false;\r\n            }\r\n\r\n            const propertyRecord = propertyValue as Record<string, unknown>;\r\n            const predicateValue = propertyRecord[predicate];\r\n            if (predicateValue === null || predicateValue === undefined) {\r\n              return false;\r\n            }\r\n\r\n            return String(predicateValue)\r\n              .trim()\r\n              .toLocaleLowerCase()\r\n              .includes(filter);\r\n          });\r\n        }\r\n\r\n        if (col.property && col.isFilterable) {\r\n          const propertyValue = item[col.property];\r\n          if (propertyValue === null || propertyValue === undefined) {\r\n            return false;\r\n          }\r\n\r\n          return String(propertyValue)\r\n            .trim()\r\n            .toLocaleLowerCase()\r\n            .includes(filter);\r\n        }\r\n        return false;\r\n      });\r\n    };\r\n    this.filterPredicate = this.dataSource.filterPredicate;\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  private async loadItemsPaginated(\r\n    navigation: 'reload' | 'forward' | 'backward' = 'reload',\r\n    reset: boolean = false,\r\n  ): Promise<void> {\r\n    this.imageLoadingState.clear();\r\n    if (reset && ['forward', 'reload'].includes(navigation)) {\r\n      this.lastDoc = null;\r\n      this.currentClientPageIndex = 0;\r\n    }\r\n    if (reset && ['backward', 'reload'].includes(navigation)) {\r\n      this.firstDoc = null;\r\n      this.currentClientPageIndex = 0;\r\n    }\r\n\r\n    const activeFilters = this.filterService.extractActiveFilters(\r\n      this.filtersForm,\r\n    );\r\n\r\n    this.arrange = {\r\n      filters: activeFilters,\r\n      sortBy: this.sortBy,\r\n    };\r\n\r\n    const paginated: Pagination = {\r\n      batchSize: this.pageSize,\r\n      collection: this.data.collection,\r\n      doc: { lastDoc: this.lastDoc, firstDoc: this.firstDoc },\r\n      navigation,\r\n      arrange: this.arrange!,\r\n      conditions: this.data.conditions,\r\n      size: this.totalItems,\r\n      filterFn: this.data.filterFn,\r\n      clientPageIndex: this.currentClientPageIndex,\r\n    };\r\n\r\n    const result = await this.tableService.getPaginated(paginated);\r\n    this.items = result.items as T[];\r\n    await this.loadRelations();\r\n    await this.loadQueryLengths();\r\n    this.lastDoc = result.lastDoc as QueryDocumentSnapshot<T> | null;\r\n    this.firstDoc = result.firstDoc as QueryDocumentSnapshot<T> | null;\r\n\r\n    if (result.currentClientPageIndex !== undefined) {\r\n      this.currentClientPageIndex = result.currentClientPageIndex;\r\n    }\r\n\r\n    let sum = 0;\r\n    if (this.data.totalRef) {\r\n      for (const totalRef of this.data.totalRef) {\r\n        const totalRefDoc = await totalRef.ref.get();\r\n        const docData = totalRefDoc.data() as\r\n          | Record<string, unknown>\r\n          | undefined;\r\n        if (docData || result.filterLength) {\r\n          const fieldValue = docData ? docData[totalRef.field] : undefined;\r\n          const numericValue = typeof fieldValue === 'number' ? fieldValue : 0;\r\n          sum = result.filterLength ?? ((sum + numericValue) as number);\r\n        }\r\n      }\r\n      this.totalItems = sum;\r\n    }\r\n\r\n    this.hasNextPage = result.hasNextPage;\r\n    this.dataSource = new MatTableDataSource(this.items);\r\n    this.filterPredicate = this.dataSource.filterPredicate;\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  // PAGINATION METHODS\r\n\r\n  async onPageChange(event?: PageEvent): Promise<void> {\r\n    if (this.data.pagination !== true || !event) return;\r\n\r\n    this.isLoading$.next(true);\r\n\r\n    const currentState = {\r\n      currentPageNumber: this.currentPageNumber,\r\n      currentClientPageIndex: this.currentClientPageIndex,\r\n      pageSize: this.pageSize,\r\n      totalItems: this.totalItems,\r\n      hasNextPage: this.hasNextPage,\r\n    };\r\n\r\n    const navigation = this.paginationService.calculateNavigation(\r\n      event,\r\n      currentState,\r\n    );\r\n\r\n    if (!navigation.shouldNavigate) {\r\n      this.isLoading$.next(false);\r\n      this.pageEvent = event;\r\n      return;\r\n    }\r\n\r\n    // Atualizar estado local\r\n    this.currentClientPageIndex = navigation.newClientPageIndex;\r\n    this.currentPageNumber = navigation.newPageNumber;\r\n\r\n    // Atualizar pageSize se mudou\r\n    if (event.pageSize !== this.pageSize) {\r\n      this.pageSize = event.pageSize;\r\n    }\r\n\r\n    // Ajustar pageSize temporariamente para última página se necessário\r\n    const lastPageIndex = this.paginationService.calculateLastPageIndex(\r\n      this.totalItems,\r\n      event.pageSize,\r\n    );\r\n    let originalPageSize: number | null = null;\r\n\r\n    if (\r\n      event.pageIndex === lastPageIndex &&\r\n      navigation.direction === 'backward' &&\r\n      navigation.resetDocs\r\n    ) {\r\n      const itemsExpectedInLastPage =\r\n        this.paginationService.calculateLastPageItemCount(\r\n          this.totalItems,\r\n          event.pageSize,\r\n        );\r\n\r\n      if (\r\n        itemsExpectedInLastPage > 0 &&\r\n        itemsExpectedInLastPage < event.pageSize\r\n      ) {\r\n        originalPageSize = this.pageSize;\r\n        this.pageSize = itemsExpectedInLastPage;\r\n      }\r\n    }\r\n\r\n    try {\r\n      await this.loadItemsPaginated(navigation.direction, navigation.resetDocs);\r\n    } catch (error) {\r\n      console.error('Erro ao carregar itens paginados:', error);\r\n    } finally {\r\n      if (originalPageSize !== null) {\r\n        this.pageSize = originalPageSize;\r\n      }\r\n    }\r\n\r\n    this.pageEvent = event;\r\n    this.isLoading$.next(false);\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  // FILTER/SEARCH METHODS\r\n\r\n  applyFilter(value: string): void {\r\n    if (this.data.pagination === false) {\r\n      this.dataSource.filter = String(value).trim().toLowerCase();\r\n    }\r\n    if (this.data.pagination === true) {\r\n      this.filterValue = value;\r\n      this.filterSubject.next(this.filterValue);\r\n    }\r\n  }\r\n\r\n  async search(event?: Event): Promise<void> {\r\n    if (event) {\r\n      event.preventDefault();\r\n      event.stopPropagation();\r\n    }\r\n\r\n    if (this.selectSort.value) {\r\n      if (this.selectSort.value.arrange === 'ascending') {\r\n        this.sortBy = {\r\n          field: this.selectSort.value.property,\r\n          order: 'asc',\r\n        };\r\n      }\r\n      if (this.selectSort.value.arrange === 'descending') {\r\n        this.sortBy = {\r\n          field: this.selectSort.value.property,\r\n          order: 'desc',\r\n        };\r\n      }\r\n    }\r\n\r\n    if (this.data.pagination === false) {\r\n      this.arrange = this.buildArrangeFromFilters();\r\n      this.applyFiltersToDataSource();\r\n      this.currentArrange =\r\n        this.filtersForm.length > 0\r\n          ? (this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? '')\r\n          : '';\r\n      this.cdr.markForCheck();\r\n    } else {\r\n      await this.loadItemsPaginated('reload', true);\r\n      this.currentArrange =\r\n        this.filtersForm.length > 0\r\n          ? (this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? '')\r\n          : '';\r\n      this.paginator.firstPage();\r\n      this.cdr.markForCheck();\r\n    }\r\n  }\r\n\r\n  async resetFilter(): Promise<void> {\r\n    this.dataSource.filter = '';\r\n    if (this.filterPredicate) {\r\n      this.dataSource.filterPredicate = this.filterPredicate;\r\n    }\r\n\r\n    this.filtersForm.clear();\r\n    this.addFilter();\r\n    this.selectSort.patchValue('');\r\n    this.sortBy = {\r\n      order: this.data.sortBy ? this.data.sortBy.order : 'desc',\r\n      field: this.data.sortBy ? this.data.sortBy.field : 'createdAt',\r\n    };\r\n\r\n    if (this.data.pagination === false) {\r\n      this.arrange = {\r\n        filters: [],\r\n        sortBy: this.sortBy,\r\n      };\r\n      this.dataSource.data = [...this.items];\r\n      this.totalItems = this.items.length;\r\n      this.currentArrange = '';\r\n      this.cdr.markForCheck();\r\n    } else {\r\n      await this.loadItemsPaginated('reload', true);\r\n      this.currentArrange = '';\r\n      this.paginator.firstPage();\r\n      this.cdr.markForCheck();\r\n    }\r\n  }\r\n\r\n  // NAVIGATION METHODS\r\n\r\n  goToDetails(row: T): void {\r\n    if (this.data.isNotClickable) {\r\n      return;\r\n    }\r\n    const urlPath = this.data.url || this.data.name;\r\n    const rowWithId = row as Record<string, unknown> & { id: string };\r\n    const url = this.router.serializeUrl(\r\n      this.router.createUrlTree([`/${urlPath}`, rowWithId.id]),\r\n    );\r\n    window.open(url, '_blank');\r\n  }\r\n\r\n  // RELATION METHODS\r\n\r\n  async getRelation(params: {\r\n    id: string;\r\n    collection: string;\r\n    newProperty: string;\r\n  }): Promise<string> {\r\n    try {\r\n      let snapshot: firestore.DocumentSnapshot<unknown> | undefined;\r\n      if (\r\n        params.id !== '' &&\r\n        params.id !== undefined &&\r\n        params.collection !== undefined &&\r\n        params.collection !== ''\r\n      ) {\r\n        snapshot = await firstValueFrom(\r\n          this.firestore.collection(params.collection).doc(params.id).get(),\r\n        );\r\n      }\r\n      if (snapshot && snapshot.exists) {\r\n        const data = snapshot.data() as Record<string, unknown> | undefined;\r\n        const value = data?.[params.newProperty];\r\n        return typeof value === 'string' ? value : '';\r\n      }\r\n      return '';\r\n    } catch (e) {\r\n      console.error(e);\r\n      return '';\r\n    }\r\n  }\r\n\r\n  private async loadRelations(): Promise<void> {\r\n    const relationPromises = this.data.displayedColumns\r\n      .filter((col) => col.relation)\r\n      .flatMap((col) =>\r\n        this.items.map(async (item) => {\r\n          if (col.relation) {\r\n            const itemRecord = item as Record<string, unknown>;\r\n            itemRecord[col.property] = await this.getRelation({\r\n              id: String(itemRecord[col.relation.sourceProperty] ?? ''),\r\n              collection: col.relation.targetedCollection,\r\n              newProperty: col.relation.targetedProperty || col.property,\r\n            });\r\n          }\r\n        }),\r\n      );\r\n\r\n    await Promise.all(relationPromises);\r\n  }\r\n\r\n  async getQueryLength(params: {\r\n    item: T;\r\n    relation: {\r\n      collection: string;\r\n      property: string;\r\n      operator: WhereFilterOp;\r\n      value: string;\r\n    };\r\n  }): Promise<number> {\r\n    const itemRecord = params.item as Record<string, unknown>;\r\n    const snapshot = await this.firestore\r\n      .collection(params.relation.collection)\r\n      .ref.where(\r\n        params.relation.property,\r\n        params.relation.operator,\r\n        itemRecord[params.relation.value],\r\n      )\r\n      .get();\r\n    return snapshot.size;\r\n  }\r\n\r\n  private async loadQueryLengths(): Promise<void> {\r\n    const lengthPromises = this.data.displayedColumns\r\n      .filter((col) => col.queryLength)\r\n      .flatMap((col) =>\r\n        this.items.map(async (item) => {\r\n          if (col.queryLength) {\r\n            const itemRecord = item as Record<string, unknown>;\r\n            itemRecord[col.property] = await this.getQueryLength({\r\n              item: item,\r\n              relation: col.queryLength,\r\n            });\r\n          }\r\n        }),\r\n      );\r\n    await Promise.all(lengthPromises);\r\n  }\r\n\r\n  private filterItems(): void {\r\n    if (this.data.conditions) {\r\n      this.data.conditions.forEach((cond) => {\r\n        this.items = this.items.filter((item) => {\r\n          const operatorFunction =\r\n            this.tableService.operators[\r\n              cond.operator as keyof typeof this.tableService.operators\r\n            ];\r\n          if (operatorFunction) {\r\n            const itemRecord = item as Record<string, unknown>;\r\n            return operatorFunction(\r\n              itemRecord[cond.firestoreProperty],\r\n              cond.dashProperty,\r\n            );\r\n          }\r\n          return false;\r\n        });\r\n      });\r\n    }\r\n  }\r\n\r\n  // PUBLIC METHODS\r\n\r\n  async reloadTable(): Promise<void> {\r\n    if (this.data.pagination) {\r\n      await this.loadItemsPaginated('reload', true);\r\n      this.paginator.firstPage();\r\n      this.cdr.markForCheck();\r\n    } else {\r\n      await this.loadItems();\r\n    }\r\n  }\r\n\r\n  updateDisplayedColumns(): void {\r\n    if (this.dataSource) {\r\n      this.dataSource = new MatTableDataSource<T>([]);\r\n    }\r\n\r\n    this.columnProperties = this.data.displayedColumns.map(\r\n      (column) => column.property,\r\n    );\r\n\r\n    const filterConfig = this.filterService.initializeDropdownItems(\r\n      this.data.displayedColumns,\r\n    );\r\n    this.dropdownItems = filterConfig.dropdownItems;\r\n    this.sortableDropdownItems = filterConfig.sortableDropdownItems;\r\n\r\n    if (this.data.filterableOptions) {\r\n      this.dropdownItems = this.filterService.addFilterableOptions(\r\n        this.dropdownItems,\r\n        this.data.filterableOptions,\r\n      );\r\n    }\r\n    this.cdr.markForCheck();\r\n  }\r\n\r\n  shouldShowActionButton(): boolean {\r\n    if (!this.data?.actionButton) {\r\n      return false;\r\n    }\r\n\r\n    if (!this.data.actionButton.condition) {\r\n      return true;\r\n    }\r\n\r\n    try {\r\n      return this.data.actionButton.condition(null) ?? true;\r\n    } catch {\r\n      return true;\r\n    }\r\n  }\r\n\r\n  handleDownload(): void {\r\n    if (!this.downloadTable) return;\r\n\r\n    if (this.data.pagination === false) {\r\n      if (this.dataSource && this.dataSource.filteredData) {\r\n        this.filteredItems = [...this.dataSource.filteredData];\r\n      }\r\n\r\n      const arrange = this.buildArrangeFromFilters();\r\n      this.downloadTable(arrange, this.data.conditions || []);\r\n    } else {\r\n      if (this.arrange) {\r\n        this.downloadTable(this.arrange, this.data.conditions || []);\r\n      }\r\n    }\r\n  }\r\n}\r\n","<div *ngIf=\"data\" class=\"card-body\">\r\n  <div class=\"flex flex-col justify-between gap-6\">\r\n    <!-- UNIFIED CONTROL PANEL: FILTERS, SORT & ACTIONS -->\r\n    <div\r\n      class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n      *ngIf=\"\r\n        data.pagination === true &&\r\n        (dropdownItems.length > 0 ||\r\n          sortableDropdownItems.length > 0 ||\r\n          data.actionButton)\r\n      \"\r\n    >\r\n      <!-- PANEL HEADER: Title and Custom Action -->\r\n      <div\r\n        class=\"mb-4 flex flex-col items-start justify-between gap-4 border-b-2 border-gray-200 pb-4 md:flex-row md:items-center\"\r\n      >\r\n        <!-- Left Side: Title & Main Action Button -->\r\n        <div class=\"flex flex-wrap items-center gap-4\">\r\n          <div class=\"flex items-center gap-2\">\r\n            <i class=\"fa fa-filter text-xl text-blue-500\"></i>\r\n            <span class=\"text-lg font-semibold text-gray-700\"\r\n              >Filtros e Ações</span\r\n            >\r\n          </div>\r\n          <button\r\n            *ngIf=\"data.actionButton && data.actionButton.condition\"\r\n            [ngClass]=\"\r\n              (data.actionButton.colorClass || 'bg-blue-500') +\r\n              ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\r\n            \"\r\n            [routerLink]=\"data.actionButton.routerLink\"\r\n            (click)=\"\r\n              data.actionButton.method ? data.actionButton.method($event) : null\r\n            \"\r\n          >\r\n            <i\r\n              *ngIf=\"data.actionButton.icon\"\r\n              [class]=\"data.actionButton.icon\"\r\n            ></i>\r\n            {{ data.actionButton.label }}\r\n          </button>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- SORT FIELD (Integrated with filters) -->\r\n      <div class=\"mb-4\" *ngIf=\"sortableDropdownItems.length > 0\">\r\n        <div class=\"flex flex-wrap items-center gap-3 p-2\">\r\n          <div class=\"min-w-[200px] flex-1\">\r\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n              <mat-label class=\"flex items-center gap-2\">\r\n                <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\r\n                <span>Ordenar por</span>\r\n              </mat-label>\r\n              <mat-select placeholder=\"Selecione...\" [formControl]=\"selectSort\">\r\n                <mat-option\r\n                  *ngFor=\"let item of sortableDropdownItems\"\r\n                  [value]=\"item\"\r\n                >\r\n                  <div class=\"flex items-center gap-2\">\r\n                    <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\r\n                    <span>{{ item.title }}</span>\r\n                  </div>\r\n                </mat-option>\r\n              </mat-select>\r\n            </mat-form-field>\r\n          </div>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- FILTERS CONTENT (WITH REFINEMENTS) -->\r\n      <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\r\n        <div\r\n          [formGroup]=\"$any(filterGroup)\"\r\n          *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\r\n          class=\"flex flex-wrap items-center gap-3 p-2\"\r\n        >\r\n          <!-- FILTER TYPE SELECTOR -->\r\n          <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\r\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n              <mat-label>Tipo de filtro</mat-label>\r\n              <mat-select\r\n                placeholder=\"Selecione o tipo...\"\r\n                formControlName=\"selectFilter\"\r\n                (selectionChange)=\"onSelectFilterChange()\"\r\n              >\r\n                <mat-option\r\n                  *ngFor=\"let item of getAvailableFilterOptions()\"\r\n                  [value]=\"item\"\r\n                >\r\n                  <div class=\"flex items-center gap-2\">\r\n                    <i\r\n                      [class]=\"item.icon || 'fa fa-filter'\"\r\n                      class=\"text-sm text-blue-500\"\r\n                    ></i>\r\n                    <span>{{ item.title }}</span>\r\n                  </div>\r\n                </mat-option>\r\n              </mat-select>\r\n            </mat-form-field>\r\n          </div>\r\n\r\n          <!-- TEXT FILTER -->\r\n          <div\r\n            class=\"min-w-[200px] flex-1\"\r\n            *ngIf=\"\r\n              $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\r\n            \"\r\n          >\r\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n              <mat-label class=\"flex items-center gap-2\">\r\n                <i class=\"fa fa-search text-gray-400\"></i>\r\n                <span>{{\r\n                  $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n                    \"Filtrar\"\r\n                }}</span>\r\n              </mat-label>\r\n              <input\r\n                (keyup.enter)=\"search($event)\"\r\n                formControlName=\"typeFilter\"\r\n                matInput\r\n                placeholder=\"Digite para filtrar...\"\r\n                #input\r\n              />\r\n            </mat-form-field>\r\n          </div>\r\n\r\n          <!-- DROPDOWN FILTER -->\r\n          <div\r\n            class=\"min-w-[200px] flex-1\"\r\n            *ngIf=\"\r\n              $any(filterGroup).get('selectFilter')?.value &&\r\n              $any(filterGroup)\r\n                .get('selectFilter')\r\n                ?.value.hasOwnProperty('items')\r\n            \"\r\n          >\r\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n              <mat-label>{{\r\n                $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n                  \"Selecione\"\r\n              }}</mat-label>\r\n              <mat-select\r\n                placeholder=\"Selecione...\"\r\n                formControlName=\"selectItem\"\r\n                multiple\r\n              >\r\n                <mat-option\r\n                  *ngFor=\"\r\n                    let item of $any(filterGroup).get('selectFilter')?.value\r\n                      .items\r\n                  \"\r\n                  [value]=\"item\"\r\n                >\r\n                  {{ item.label }}\r\n                </mat-option>\r\n              </mat-select>\r\n            </mat-form-field>\r\n          </div>\r\n\r\n          <!-- DATE FILTER -->\r\n          <div\r\n            class=\"min-w-[340px] flex-auto\"\r\n            *ngIf=\"\r\n              $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n              'filterByDate'\r\n            \"\r\n          >\r\n            <div\r\n              class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\r\n            >\r\n              <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n                <mat-label class=\"flex items-center gap-2\">\r\n                  <i class=\"fa fa-calendar text-gray-400\"></i>\r\n                  <span>Data Inicial</span>\r\n                </mat-label>\r\n                <input\r\n                  type=\"text\"\r\n                  matInput\r\n                  (keydown.enter)=\"search($event)\"\r\n                  (input)=\"onDateInput($event, 'initialDate', i)\"\r\n                  formControlName=\"initialDate\"\r\n                  placeholder=\"DD/MM/AAAA\"\r\n                  maxlength=\"10\"\r\n                />\r\n              </mat-form-field>\r\n\r\n              <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n                <mat-label class=\"flex items-center gap-2\">\r\n                  <i class=\"fa fa-calendar text-gray-400\"></i>\r\n                  <span>Data Final</span>\r\n                </mat-label>\r\n                <input\r\n                  type=\"text\"\r\n                  (keydown.enter)=\"search($event)\"\r\n                  (input)=\"onDateInput($event, 'finalDate', i)\"\r\n                  matInput\r\n                  formControlName=\"finalDate\"\r\n                  placeholder=\"DD/MM/AAAA\"\r\n                  maxlength=\"10\"\r\n                />\r\n              </mat-form-field>\r\n            </div>\r\n          </div>\r\n\r\n          <!-- REMOVE FILTER BUTTON -->\r\n          <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\r\n            <button\r\n              (click)=\"removeFilter(i)\"\r\n              class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\r\n              matTooltip=\"Remover filtro\"\r\n            >\r\n              <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\r\n            </button>\r\n          </div>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- PANEL FOOTER: Add Filter & Action Buttons -->\r\n      <div\r\n        class=\"-mb-2 flex items-start justify-between gap-4 border-t border-gray-200 pt-4\"\r\n      >\r\n        <!-- Add Filter Button -->\r\n        <div *ngIf=\"dropdownItems.length > 0\">\r\n          <button\r\n            (click)=\"addFilter()\"\r\n            class=\"transform rounded-full border-2 border-blue-300 bg-blue-50 px-6 py-2 text-sm font-medium text-blue-600 transition-all duration-300 hover:-translate-y-0.5 hover:border-blue-400 hover:bg-blue-100 hover:shadow-md\"\r\n            matTooltip=\"Adicionar novo filtro\"\r\n          >\r\n            <i class=\"fa fa-plus mr-2\"></i>\r\n            Adicionar Filtro\r\n          </button>\r\n        </div>\r\n\r\n        <!-- Right Side: Search, Reset, Export -->\r\n        <div\r\n          class=\"flex flex-wrap gap-3 self-start sm:self-auto\"\r\n          *ngIf=\"\r\n            this.hasFilterableColumn === true || this.hasSortableColumn === true\r\n          \"\r\n        >\r\n          <button\r\n            (click)=\"search()\"\r\n            type=\"button\"\r\n            class=\"flex items-center gap-2 rounded-lg bg-green-600 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-green-700\"\r\n            matTooltip=\"Aplicar filtros\"\r\n          >\r\n            <i class=\"fa fa-search\"></i>\r\n            Pesquisar\r\n          </button>\r\n\r\n          <button\r\n            (click)=\"resetFilter()\"\r\n            class=\"flex items-center gap-2 rounded-lg bg-red-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-red-600\"\r\n            matTooltip=\"Limpar filtros\"\r\n          >\r\n            <i class=\"fas fa-redo-alt\"></i>\r\n            Resetar\r\n          </button>\r\n\r\n          <button\r\n            *ngIf=\"data.download !== false && downloadTable\"\r\n            class=\"flex items-center gap-2 rounded-lg bg-orange-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-orange-600\"\r\n            matTooltipPosition=\"above\"\r\n            matTooltip=\"Exportar Tabela\"\r\n            [disabled]=\"\r\n              this.dataSource && this.dataSource.filteredData.length <= 0\r\n            \"\r\n            (click)=\"\r\n              $any(arrange) && downloadTable !== undefined\r\n                ? downloadTable($any(arrange), data.conditions || [])\r\n                : null\r\n            \"\r\n          >\r\n            <i class=\"fa fa-download\"></i>\r\n            Exportar\r\n          </button>\r\n        </div>\r\n      </div>\r\n    </div>\r\n\r\n    <!-- UNIFIED CONTROL PANEL (for non-paginated tables): SEARCH, SORT & FILTERS -->\r\n    <div\r\n      class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n      *ngIf=\"\r\n        data.pagination === false &&\r\n        (hasFilterableColumn === true ||\r\n          dropdownItems.length > 0 ||\r\n          data.actionButton)\r\n      \"\r\n    >\r\n      <!-- PANEL HEADER: Title and Custom Action -->\r\n      <div\r\n        class=\"mb-4 flex flex-col items-start justify-between gap-4 border-b-2 border-gray-200 pb-4 md:flex-row md:items-center\"\r\n      >\r\n        <!-- Left Side: Title & Main Action Button -->\r\n        <div class=\"flex flex-wrap items-center gap-4\">\r\n          <div class=\"flex items-center gap-2\">\r\n            <i class=\"fa fa-filter text-xl text-blue-500\"></i>\r\n            <span class=\"text-lg font-semibold text-gray-700\"\r\n              >Filtros e Ações</span\r\n            >\r\n          </div>\r\n          <button\r\n            *ngIf=\"data.actionButton && data.actionButton.condition\"\r\n            [ngClass]=\"\r\n              (data.actionButton.colorClass || 'bg-blue-500') +\r\n              ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\r\n            \"\r\n            [routerLink]=\"data.actionButton.routerLink\"\r\n            (click)=\"\r\n              data.actionButton.method ? data.actionButton.method($event) : null\r\n            \"\r\n          >\r\n            <i\r\n              *ngIf=\"data.actionButton.icon\"\r\n              [class]=\"data.actionButton.icon\"\r\n            ></i>\r\n            {{ data.actionButton.label }}\r\n          </button>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- SIMPLE SEARCH -->\r\n      <div\r\n        class=\"mb-4\"\r\n        *ngIf=\"hasFilterableColumn === true && data.showSimpleSearch !== false\"\r\n      >\r\n        <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n          <mat-label class=\"flex items-center gap-2\">\r\n            <i class=\"fa fa-search text-blue-500\"></i>\r\n            Buscar\r\n          </mat-label>\r\n          <input\r\n            matInput\r\n            (keyup.enter)=\"search($event)\"\r\n            (keyup)=\"applyFilter(filterInput.value)\"\r\n            placeholder=\"Digite para filtrar...\"\r\n            #filterInput\r\n          />\r\n          <mat-icon matSuffix class=\"text-gray-500\">search</mat-icon>\r\n        </mat-form-field>\r\n      </div>\r\n\r\n      <!-- FILTERS PANEL -->\r\n      <div *ngIf=\"dropdownItems.length > 0\">\r\n        <!-- FILTERS CONTENT -->\r\n        <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\r\n          <div\r\n            [formGroup]=\"$any(filterGroup)\"\r\n            *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\r\n            class=\"flex flex-wrap items-center gap-3\"\r\n          >\r\n            <!-- FILTER TYPE SELECTOR -->\r\n            <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\r\n              <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n                <mat-label>Tipo de filtro</mat-label>\r\n                <mat-select\r\n                  placeholder=\"Selecione o tipo...\"\r\n                  formControlName=\"selectFilter\"\r\n                  (selectionChange)=\"onSelectFilterChange()\"\r\n                >\r\n                  <mat-option\r\n                    *ngFor=\"let item of getAvailableFilterOptions()\"\r\n                    [value]=\"item\"\r\n                  >\r\n                    <div class=\"flex items-center gap-2\">\r\n                      <i\r\n                        [class]=\"item.icon || 'fa fa-filter'\"\r\n                        class=\"text-sm text-blue-500\"\r\n                      ></i>\r\n                      <span>{{ item.title }}</span>\r\n                    </div>\r\n                  </mat-option>\r\n                </mat-select>\r\n              </mat-form-field>\r\n            </div>\r\n\r\n            <!-- TEXT FILTER -->\r\n            <div\r\n              class=\"min-w-[200px] flex-1\"\r\n              *ngIf=\"\r\n                $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n                'filter'\r\n              \"\r\n            >\r\n              <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n                <mat-label class=\"flex items-center gap-2\">\r\n                  <i class=\"fa fa-search text-gray-400\"></i>\r\n                  <span>{{\r\n                    $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n                      \"Filtrar\"\r\n                  }}</span>\r\n                </mat-label>\r\n                <input\r\n                  (keyup.enter)=\"search($event)\"\r\n                  formControlName=\"typeFilter\"\r\n                  matInput\r\n                  placeholder=\"Digite para filtrar...\"\r\n                  #input\r\n                />\r\n              </mat-form-field>\r\n            </div>\r\n\r\n            <!-- DROPDOWN FILTER -->\r\n            <div\r\n              class=\"min-w-[200px] flex-1\"\r\n              *ngIf=\"\r\n                $any(filterGroup).get('selectFilter')?.value &&\r\n                $any(filterGroup)\r\n                  .get('selectFilter')\r\n                  ?.value.hasOwnProperty('items')\r\n              \"\r\n            >\r\n              <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n                <mat-label>{{\r\n                  $any(filterGroup).get(\"selectFilter\")?.value?.title ||\r\n                    \"Selecione\"\r\n                }}</mat-label>\r\n                <mat-select\r\n                  placeholder=\"Selecione...\"\r\n                  formControlName=\"selectItem\"\r\n                  multiple\r\n                >\r\n                  <mat-option\r\n                    *ngFor=\"\r\n                      let item of $any(filterGroup).get('selectFilter')?.value\r\n                        .items\r\n                    \"\r\n                    [value]=\"item\"\r\n                  >\r\n                    {{ item.label }}\r\n                  </mat-option>\r\n                </mat-select>\r\n              </mat-form-field>\r\n            </div>\r\n\r\n            <!-- DATE FILTER -->\r\n            <div\r\n              class=\"min-w-[340px] flex-auto\"\r\n              *ngIf=\"\r\n                $any(filterGroup).get('selectFilter')?.value?.arrange ===\r\n                'filterByDate'\r\n              \"\r\n            >\r\n              <div\r\n                class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\r\n              >\r\n                <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n                  <mat-label class=\"flex items-center gap-2\">\r\n                    <i class=\"fa fa-calendar text-gray-400\"></i>\r\n                    <span>Data Inicial</span>\r\n                  </mat-label>\r\n                  <input\r\n                    type=\"text\"\r\n                    matInput\r\n                    (keydown.enter)=\"search($event)\"\r\n                    (input)=\"onDateInput($event, 'initialDate', i)\"\r\n                    (blur)=\"onDateFilterChange()\"\r\n                    formControlName=\"initialDate\"\r\n                    placeholder=\"DD/MM/AAAA\"\r\n                    maxlength=\"10\"\r\n                  />\r\n                </mat-form-field>\r\n\r\n                <mat-form-field appearance=\"outline\" class=\"flex-1\">\r\n                  <mat-label class=\"flex items-center gap-2\">\r\n                    <i class=\"fa fa-calendar text-gray-400\"></i>\r\n                    <span>Data Final</span>\r\n                  </mat-label>\r\n                  <input\r\n                    type=\"text\"\r\n                    (keydown.enter)=\"search($event)\"\r\n                    (input)=\"onDateInput($event, 'finalDate', i)\"\r\n                    (blur)=\"onDateFilterChange()\"\r\n                    matInput\r\n                    formControlName=\"finalDate\"\r\n                    placeholder=\"DD/MM/AAAA\"\r\n                    maxlength=\"10\"\r\n                  />\r\n                </mat-form-field>\r\n              </div>\r\n            </div>\r\n\r\n            <!-- REMOVE FILTER BUTTON -->\r\n            <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\r\n              <button\r\n                (click)=\"removeFilter(i)\"\r\n                class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\r\n                matTooltip=\"Remover filtro\"\r\n              >\r\n                <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\r\n              </button>\r\n            </div>\r\n          </div>\r\n        </div>\r\n\r\n        <!-- PANEL FOOTER: Add Filter & Actions -->\r\n        <div\r\n          class=\"-mb-2 flex items-start justify-between gap-4 border-t border-gray-200 pt-4\"\r\n        >\r\n          <!-- Add Filter Button -->\r\n          <div *ngIf=\"dropdownItems.length > 0\">\r\n            <button\r\n              (click)=\"addFilter()\"\r\n              class=\"transform rounded-full border-2 border-blue-300 bg-blue-50 px-6 py-2 text-sm font-medium text-blue-600 transition-all duration-300 hover:-translate-y-0.5 hover:border-blue-400 hover:bg-blue-100 hover:shadow-md\"\r\n              matTooltip=\"Adicionar novo filtro\"\r\n            >\r\n              <i class=\"fa fa-plus mr-2\"></i>\r\n              Adicionar Filtro\r\n            </button>\r\n          </div>\r\n\r\n          <!-- Action Buttons -->\r\n          <div\r\n            class=\"flex flex-wrap gap-3 self-start sm:self-auto\"\r\n            *ngIf=\"\r\n              this.hasFilterableColumn === true ||\r\n              this.hasSortableColumn === true\r\n            \"\r\n          >\r\n            <button\r\n              (click)=\"search()\"\r\n              type=\"button\"\r\n              class=\"flex items-center gap-2 rounded-lg bg-green-600 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-green-700\"\r\n              matTooltip=\"Aplicar filtros\"\r\n            >\r\n              <i class=\"fa fa-search\"></i>\r\n              Pesquisar\r\n            </button>\r\n\r\n            <button\r\n              (click)=\"resetFilter()\"\r\n              class=\"flex items-center gap-2 rounded-lg bg-red-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-red-600\"\r\n              matTooltip=\"Limpar filtros\"\r\n            >\r\n              <i class=\"fas fa-redo-alt\"></i>\r\n              Resetar\r\n            </button>\r\n\r\n            <button\r\n              *ngIf=\"data.download !== false && downloadTable\"\r\n              class=\"flex items-center gap-2 rounded-lg bg-orange-500 px-5 py-2 text-sm font-medium text-white transition-colors hover:bg-orange-600\"\r\n              matTooltipPosition=\"above\"\r\n              matTooltip=\"Exportar Tabela\"\r\n              [disabled]=\"\r\n                this.dataSource && this.dataSource.filteredData.length <= 0\r\n              \"\r\n              (click)=\"handleDownload()\"\r\n            >\r\n              <i class=\"fa fa-download\"></i>\r\n              Exportar\r\n            </button>\r\n          </div>\r\n        </div>\r\n      </div>\r\n    </div>\r\n\r\n    <div class=\"flex flex-col\">\r\n      <!-- TABS COMPONENT -->\r\n      <lib-table-tabs\r\n        *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\r\n        [tabsData]=\"data.tabs.tabsData\"\r\n        [selectedTab]=\"selectedTab\"\r\n        (tabSelected)=\"onTabSelected($event)\"\r\n      ></lib-table-tabs>\r\n      <div\r\n        class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl table-container\"\r\n      >\r\n        <table\r\n          mat-table\r\n          [dataSource]=\"dataSource\"\r\n          matSort\r\n          #sort=\"matSort\"\r\n          matSortActive=\"createdAt\"\r\n          matSortDirection=\"desc\"\r\n        >\r\n          <ng-container\r\n            *ngFor=\"let col of data.displayedColumns; trackBy: trackByProperty\"\r\n            matColumnDef=\"{{ col.property }}\"\r\n          >\r\n            <ng-container *matHeaderCellDef>\r\n              <!-- IF THE COLUMN IS NOT SORTABLE, THEN DON'T SHOW THE SORT BUTTONS -->\r\n              <th\r\n                *ngIf=\"!col.isSortable || data.pagination === true\"\r\n                mat-header-cell\r\n              >\r\n                {{ col.title }}\r\n              </th>\r\n              <!-- IF THE COLUMN IS SORTABLE, THEN SHOW THE SORT BUTTONS -->\r\n              <th\r\n                *ngIf=\"col.isSortable && data.pagination === false\"\r\n                mat-header-cell\r\n                mat-sort-header\r\n              >\r\n                {{ col.title }}\r\n              </th>\r\n              <td\r\n                mat-cell\r\n                *matCellDef=\"let row\"\r\n                (click)=\"col.method ? col.method(row) : null\"\r\n                (mouseenter)=\"onCellMouseEnter($event, row, col)\"\r\n                (mouseleave)=\"onCellMouseLeave()\"\r\n                (mousemove)=\"onCellMouseMove($event)\"\r\n              >\r\n                <!-- CHECK IF THE COLUMN MUST BE DISPLAYED -->\r\n                <span *ngIf=\"!col.image && !col.iconClass && !col.method\">\r\n                  <ng-container>\r\n                    <span\r\n                      *ngIf=\"\r\n                        col.charLimit &&\r\n                          row[col.property] &&\r\n                          row[col.property].length > col.charLimit;\r\n                        else withinLimit\r\n                      \"\r\n                    >\r\n                      <a\r\n                        *ngIf=\"col.hasLink === true\"\r\n                        [href]=\"row[col.property]\"\r\n                        target=\"_blank\"\r\n                      >\r\n                        {{ getDisplayValue(col, row) }}\r\n                      </a>\r\n                      <a\r\n                        *ngIf=\"col.hasLink && isString(col.hasLink)\"\r\n                        [href]=\"col.hasLink\"\r\n                        target=\"_blank\"\r\n                      >\r\n                        {{ getDisplayValue(col, row) }}\r\n                      </a>\r\n                      <span\r\n                        *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\r\n                      >\r\n                        {{ getDisplayValue(col, row) }}\r\n                      </span>\r\n                    </span>\r\n                  </ng-container>\r\n                  <ng-template #withinLimit>\r\n                    <a\r\n                      *ngIf=\"col.hasLink === true\"\r\n                      [href]=\"row[col.property]\"\r\n                      target=\"_blank\"\r\n                    >\r\n                      {{ getDisplayValue(col, row, true) }}\r\n                    </a>\r\n                    <a\r\n                      *ngIf=\"col.hasLink && isString(col.hasLink)\"\r\n                      [href]=\"col.hasLink\"\r\n                      target=\"_blank\"\r\n                    >\r\n                      {{ getDisplayValue(col, row, true) }}\r\n                    </a>\r\n                    <span\r\n                      *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\r\n                    >\r\n                      {{ getDisplayValue(col, row, true) }}\r\n                    </span>\r\n                  </ng-template>\r\n                </span>\r\n                <!------------------- IMAGE ------------------>\r\n                <!-- Imagem com loading -->\r\n                <div\r\n                  *ngIf=\"\r\n                    col.image && col.image.path && !col.iconClass && !col.method\r\n                  \"\r\n                  class=\"relative inline-block\"\r\n                >\r\n                  <mat-spinner\r\n                    *ngIf=\"isImageLoading(row, col)\"\r\n                    [diameter]=\"32\"\r\n                    class=\"absolute inset-0 m-auto\"\r\n                  ></mat-spinner>\r\n                  <img\r\n                    [src]=\"col.image.path + '/' + row[col.property]\"\r\n                    [ngClass]=\"col.image.class\"\r\n                    [class.opacity-0]=\"isImageLoading(row, col)\"\r\n                    alt=\"\"\r\n                    (load)=\"onImageLoad(row, col)\"\r\n                    (error)=\"onImageError($event, row, col, col.image.default)\"\r\n                  />\r\n                </div>\r\n\r\n                <!-- Imagem com URL completa e loading -->\r\n                <div\r\n                  *ngIf=\"\r\n                    col.image && col.image.url && !col.iconClass && !col.method\r\n                  \"\r\n                  class=\"relative inline-block\"\r\n                >\r\n                  <mat-spinner\r\n                    *ngIf=\"isImageLoading(row, col)\"\r\n                    [diameter]=\"32\"\r\n                    class=\"absolute inset-0 m-auto\"\r\n                  ></mat-spinner>\r\n                  <img\r\n                    [src]=\"row[col.property]\"\r\n                    [ngClass]=\"col.image.class\"\r\n                    [class.opacity-0]=\"isImageLoading(row, col)\"\r\n                    alt=\"\"\r\n                    (load)=\"onImageLoad(row, col)\"\r\n                    (error)=\"onImageError($event, row, col, col.image.default)\"\r\n                  />\r\n                </div>\r\n                <ng-container *ngIf=\"col.iconClass\">\r\n                  <button\r\n                    *ngFor=\"let iconClass of col.iconClass\"\r\n                    (click)=\"\r\n                      iconClass.buttonMethod\r\n                        ? iconClass.buttonMethod(row, $event)\r\n                        : $event.stopPropagation()\r\n                    \"\r\n                  >\r\n                    <span\r\n                      [ngClass]=\"iconClass.class\"\r\n                      *ngIf=\"\r\n                        iconClass.condition === undefined ||\r\n                        (iconClass.condition !== undefined &&\r\n                          $any(iconClass.condition)(row))\r\n                      \"\r\n                      >{{ iconClass.text }}</span\r\n                    >\r\n                  </button>\r\n                </ng-container>\r\n              </td>\r\n            </ng-container>\r\n          </ng-container>\r\n\r\n          <tr mat-header-row *matHeaderRowDef=\"columnProperties\"></tr>\r\n          <tr\r\n            [ngClass]=\"{\r\n              'example-element-row': data.isNotClickable === true,\r\n              'example-element-row cursor-pointer': !data.isNotClickable,\r\n            }\"\r\n            mat-row\r\n            *matRowDef=\"let row; columns: columnProperties\"\r\n            (click)=\"goToDetails(row)\"\r\n          ></tr>\r\n\r\n          <!-- ROW SHOWN WHEN THERE IS NO MATCHING DATA. -->\r\n          <tr class=\"mat-row\" *matNoDataRow>\r\n            <td *ngIf=\"!(isLoading$ | async)\" class=\"mat-cell p-4\" colspan=\"4\">\r\n              Nenhum resultado encontrado para a busca\r\n            </td>\r\n          </tr>\r\n        </table>\r\n\r\n        <div class=\"flex justify-center\" *ngIf=\"isLoading$ | async\">\r\n          <mat-spinner></mat-spinner>\r\n        </div>\r\n\r\n        <div class=\"paginator-container\">\r\n          <mat-paginator\r\n            #paginator\r\n            [pageSizeOptions]=\"[25, 50, 100]\"\r\n            [pageSize]=\"pageSize\"\r\n            [length]=\"totalItems\"\r\n            showFirstLastButtons\r\n            aria-label=\"Select page of periodic elements\"\r\n            (page)=\"onPageChange($event)\"\r\n            [ngClass]=\"{\r\n              'hide-length':\r\n                ['filter', 'filterByDate', 'equals'].includes(\r\n                  this.currentArrange\r\n                ) || this.data.filterFn,\r\n              'hide-next-button': !hasNextPage && data.pagination === true,\r\n              'hide-last-button':\r\n                (!hasNextPage && data.pagination === true) ||\r\n                this.data.filterFn,\r\n            }\"\r\n          >\r\n          </mat-paginator>\r\n          <div\r\n            *ngIf=\"\r\n              !(isLoading$ | async) &&\r\n              dataSource?.data &&\r\n              dataSource.data.length > 0 &&\r\n              data?.filterFn\r\n            \"\r\n            class=\"page-number-display\"\r\n          >\r\n            {{ currentPageNumber }}\r\n          </div>\r\n        </div>\r\n      </div>\r\n    </div>\r\n  </div>\r\n\r\n  <!-- TOOLTIP PERSONALIZADO -->\r\n  <lib-table-tooltip\r\n    [state]=\"tooltipState\"\r\n    [styles]=\"getTooltipStyles()\"\r\n  ></lib-table-tooltip>\r\n</div>\r\n"]}