ng-firebase-table-kxp 1.0.11 → 1.0.13

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.
@@ -1,1020 +1,1092 @@
1
- import { Component, Input, ViewChild } from '@angular/core';
2
- import { MatPaginator } from '@angular/material/paginator';
3
- import { MatSort } from '@angular/material/sort';
4
- import { MatTableDataSource } from '@angular/material/table';
5
- import { debounceTime, firstValueFrom, Subject } from 'rxjs';
6
- import { FormControl, FormArray, FormGroup } from '@angular/forms';
7
- import * as i0 from "@angular/core";
8
- import * as i1 from "@angular/router";
9
- import * as i2 from "../../services/table.service";
10
- import * as i3 from "@angular/fire/compat/firestore";
11
- import * as i4 from "@angular/common";
12
- import * as i5 from "@angular/forms";
13
- import * as i6 from "@angular/material/table";
14
- import * as i7 from "@angular/material/paginator";
15
- import * as i8 from "@angular/material/sort";
16
- import * as i9 from "@angular/material/form-field";
17
- import * as i10 from "@angular/material/input";
18
- import * as i11 from "@angular/material/select";
19
- import * as i12 from "@angular/material/core";
20
- import * as i13 from "@angular/material/tooltip";
21
- import * as i14 from "@angular/material/progress-spinner";
22
- import * as i15 from "@angular/material/icon";
23
- import * as i16 from "ngx-mask";
24
- export class TableComponent {
25
- // CONSTRUCTOR
26
- constructor(router, tableService, firestore) {
27
- this.router = router;
28
- this.tableService = tableService;
29
- this.firestore = firestore;
30
- this.arrange = null;
31
- this.currentPageNumber = 1;
32
- this.currentClientPageIndex = 0;
33
- this.items = [];
34
- this.filteredItems = []; // Dados filtrados para modo não paginado (público para acesso externo)
35
- this.isLoading = false;
36
- this.lastDoc = null;
37
- this.firstDoc = null;
38
- this.sortBy = {
39
- field: 'createdAt',
40
- order: 'desc',
41
- };
42
- this.columnProperties = [];
43
- this.selectSort = new FormControl('');
44
- this.currentArrange = '';
45
- this.hasNextPage = false;
46
- this.dropdownItems = [];
47
- this.sortableDropdownItems = [];
48
- this.pageSize = 25;
49
- this.totalItems = 0;
50
- this.filterValue = null;
51
- this.hasFilterableColumn = false;
52
- this.hasSortableColumn = false;
53
- this.filterSubject = new Subject();
54
- this.debounceTimeMs = 500;
55
- this.selectedTab = 0;
56
- // Propriedades para controle do tooltip
57
- this.hoveredCell = null;
58
- this.showTooltip = false;
59
- this.tooltipContent = '';
60
- this.tooltipPosition = { x: 0, y: 0 };
61
- this.filtersForm = new FormArray([this.createFilterGroup()]);
62
- }
63
- createFilterGroup() {
64
- return new FormGroup({
65
- selectFilter: new FormControl(''),
66
- typeFilter: new FormControl(''),
67
- selectItem: new FormControl(''),
68
- initialDate: new FormControl('', this.tableService.dateFormatValidator()),
69
- finalDate: new FormControl('', this.tableService.dateFormatValidator()),
70
- });
71
- }
72
- addFilter(filterData) {
73
- const newFilterGroup = this.createFilterGroup();
74
- if (filterData) {
75
- if (filterData.selectFilter) {
76
- newFilterGroup.get('selectFilter')?.setValue(filterData.selectFilter);
77
- }
78
- if (filterData.typeFilter) {
79
- newFilterGroup.get('typeFilter')?.setValue(filterData.typeFilter);
80
- }
81
- if (filterData.selectItem) {
82
- newFilterGroup.get('selectItem')?.setValue(filterData.selectItem);
83
- }
84
- if (filterData.initialDate) {
85
- newFilterGroup.get('initialDate')?.setValue(filterData.initialDate);
86
- }
87
- if (filterData.finalDate) {
88
- newFilterGroup.get('finalDate')?.setValue(filterData.finalDate);
89
- }
90
- }
91
- this.filtersForm.push(newFilterGroup);
92
- }
93
- onSelectFilterChange() {
94
- const lastIndex = this.filtersForm.length - 1;
95
- const lastFilter = this.filtersForm.at(lastIndex);
96
- if (lastFilter.get('selectFilter')?.value) {
97
- this.addFilter();
98
- }
99
- }
100
- removeFilter(index) {
101
- this.filtersForm.removeAt(index);
102
- if (this.filtersForm.length === 0) {
103
- this.addFilter();
104
- }
105
- }
106
- removeAllFilters() {
107
- this.filtersForm.clear();
108
- this.addFilter();
109
- this.resetFilter();
110
- }
111
- // METHODS
112
- async ngOnInit() {
113
- if (!this.data.color)
114
- this.data.color = { bg: 'bg-primary', text: 'text-black' };
115
- this.columnProperties = this.data.displayedColumns.map((column) => {
116
- return column.property;
117
- });
118
- if (this.data.actionButton && !this.data.actionButton.condition) {
119
- this.data.actionButton.condition = (_row) => true;
120
- }
121
- this.data.displayedColumns.forEach((col) => {
122
- if (col.isFilterable) {
123
- if (this.hasFilterableColumn === false)
124
- this.hasFilterableColumn = true;
125
- this.dropdownItems.push({
126
- ...col,
127
- arrange: 'filter',
128
- title: col.title,
129
- });
130
- }
131
- if (col.isSortable) {
132
- if (this.hasSortableColumn === false)
133
- this.hasSortableColumn = true;
134
- this.sortableDropdownItems.push({
135
- ...col,
136
- arrange: 'ascending',
137
- title: col.title + ': crescente',
138
- });
139
- this.sortableDropdownItems.push({
140
- ...col,
141
- arrange: 'descending',
142
- title: col.title + ': decrescente',
143
- });
144
- }
145
- if (col.isFilterableByDate) {
146
- this.dropdownItems.push({
147
- ...col,
148
- arrange: 'filterByDate',
149
- title: col.title + ': filtro por data',
150
- });
151
- }
152
- });
153
- if (this.data.filterableOptions &&
154
- Array.isArray(this.data.filterableOptions) &&
155
- this.data.filterableOptions.length > 0) {
156
- this.data.filterableOptions.forEach((option) => this.dropdownItems.push({ ...option, arrange: 'equals' }));
157
- }
158
- // Sem paginação
159
- if (this.data.pagination === false) {
160
- await this.loadItems();
161
- }
162
- // Com paginação
163
- if (this.data.pagination === true) {
164
- if (this.data.sortBy)
165
- this.sortBy = {
166
- field: this.data.sortBy.field,
167
- order: this.data.sortBy.order,
168
- };
169
- this.filterSubject
170
- .pipe(debounceTime(this.debounceTimeMs))
171
- .subscribe(() => {
172
- this.loadItemsPaginated('reload', true);
173
- });
174
- this.isLoading = true;
175
- await this.loadItemsPaginated('reload', true);
176
- this.sort.active = 'createdAt';
177
- this.sort.direction = 'desc';
178
- this.dataSource.paginator = this.paginator;
179
- this.dataSource.sort = this.sort;
180
- this.totalItems = 0;
181
- if (this.data.totalRef) {
182
- for (const totalRef of this.data.totalRef) {
183
- const totalRefDoc = await totalRef.ref.get();
184
- const docData = totalRefDoc.data();
185
- if (docData && docData[totalRef.field])
186
- this.totalItems = (this.totalItems +
187
- docData[totalRef.field]);
188
- }
189
- }
190
- this.isLoading = false;
191
- }
192
- }
193
- getDisplayValue(col, row, withinLimit = false) {
194
- let value;
195
- if (col.calculateValue) {
196
- value = col.calculateValue(row);
197
- }
198
- else {
199
- value = this.getNestedValue(row, col.property);
200
- }
201
- if (Array.isArray(value) && col.arrayField) {
202
- value = this.formatArrayValue(value, col.arrayField);
203
- }
204
- if (col.queryLength && row[col.property]) {
205
- value = row[col.property];
206
- }
207
- value = col.pipe ? col.pipe.transform(value) : value;
208
- if (value === null || value === undefined) {
209
- value = '';
210
- }
211
- else {
212
- value = String(value);
213
- }
214
- if (typeof value !== 'string') {
215
- value = '';
216
- }
217
- if (withinLimit || !col.charLimit) {
218
- return value;
219
- }
220
- return value.substring(0, col.charLimit) + '...';
221
- }
222
- getNestedValue(obj, path) {
223
- if (!path)
224
- return undefined;
225
- const properties = path.split('.');
226
- return properties.reduce((acc, currentPart) => acc && acc[currentPart], obj);
227
- }
228
- formatArrayValue(array, field) {
229
- if (!Array.isArray(array) || array.length === 0) {
230
- return '';
231
- }
232
- const values = array
233
- .map((item) => {
234
- if (typeof item === 'object' && item !== null) {
235
- return item[field] || '';
236
- }
237
- return String(item);
238
- })
239
- .filter((value) => value !== '' && value !== null && value !== undefined);
240
- return values.join(', ');
241
- }
242
- // Métodos sem paginação
243
- async loadItems() {
244
- this.items = await this.tableService.getItems(this.data.collectionRef);
245
- if (this.data.conditions) {
246
- this.filterItems();
247
- }
248
- await this.loadRelations();
249
- await this.loadQueryLengths();
250
- this.totalItems = this.items.length;
251
- // Inicializar arrange para tabelas não paginadas
252
- this.arrange = {
253
- filters: [],
254
- sortBy: this.data.sortBy || {
255
- field: 'createdAt',
256
- order: 'desc',
257
- },
258
- };
259
- this.sortBy = this.data.sortBy || {
260
- field: 'createdAt',
261
- order: 'desc',
262
- };
263
- // Aplicar filtros client-side se existirem
264
- let itemsToDisplay = [...this.items];
265
- itemsToDisplay = this.applyClientSideFilters(itemsToDisplay);
266
- this.filteredItems = itemsToDisplay; // Armazenar dados filtrados
267
- this.dataSource = new MatTableDataSource(itemsToDisplay);
268
- this.dataSource.paginator = this.paginator;
269
- this.dataSource.sort = this.sort;
270
- if (this.data.sortBy) {
271
- this.dataSource.sort.active = this.data.sortBy.field;
272
- this.dataSource.sort.direction = this.data.sortBy.order;
273
- this.dataSource.sort.sortChange.emit();
274
- }
275
- this.dataSource.filterPredicate = (data, filter) => {
276
- return this.data.displayedColumns.some((col) => {
277
- if (col.filterPredicates) {
278
- return col.filterPredicates.some((predicate) => {
279
- const propertyValue = data[col.property];
280
- if (!propertyValue || typeof propertyValue !== 'object') {
281
- return false;
282
- }
283
- const predicateValue = propertyValue[predicate];
284
- if (predicateValue === null || predicateValue === undefined) {
285
- return false;
286
- }
287
- return String(predicateValue)
288
- .trim()
289
- .toLocaleLowerCase()
290
- .includes(filter);
291
- });
292
- }
293
- if (col.property && col.isFilterable) {
294
- const propertyValue = data[col.property];
295
- if (propertyValue === null || propertyValue === undefined) {
296
- return false;
297
- }
298
- return String(propertyValue)
299
- .trim()
300
- .toLocaleLowerCase()
301
- .includes(filter);
302
- }
303
- return false;
304
- });
305
- };
306
- this.filterPredicate = this.dataSource.filterPredicate;
307
- }
308
- // Aplicar filtros client-side (filtros por data)
309
- applyClientSideFilters(items) {
310
- let filteredItems = [...items];
311
- // Processar filtros do filtersForm
312
- this.filtersForm.controls.forEach((control) => {
313
- const group = control;
314
- const selectedFilter = group.get('selectFilter')?.value;
315
- if (!selectedFilter)
316
- return;
317
- const arrange = selectedFilter.arrange;
318
- if (arrange === 'filterByDate') {
319
- const initial = group.get('initialDate')?.value;
320
- const final = group.get('finalDate')?.value;
321
- // aplicar filtro se ambas as datas estiverem preenchidas
322
- if (initial && final && initial.trim() && final.trim()) {
323
- try {
324
- // Validar formato da data (DD/MM/AAAA)
325
- const datePattern = /^(\d{1,2})\/(\d{1,2})\/(\d{4})$/;
326
- if (!datePattern.test(initial) || !datePattern.test(final)) {
327
- return; // Ignorar se formato inválido
328
- }
329
- const [dayI, monthI, yearI] = initial.split('/');
330
- const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
331
- const [dayF, monthF, yearF] = final.split('/');
332
- const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
333
- finalDate.setHours(23, 59, 59);
334
- // Validar se as datas são válidas
335
- if (isNaN(initialDate.getTime()) || isNaN(finalDate.getTime())) {
336
- return; // Ignorar se datas inválidas
337
- }
338
- // Usar o campo da coluna ou o sortBy.field como padrão
339
- const dateField = selectedFilter.property || this.sortBy.field;
340
- filteredItems = filteredItems.filter((item) => {
341
- try {
342
- const fieldValue = item[dateField];
343
- if (!fieldValue) {
344
- return false;
345
- }
346
- let itemDate;
347
- if (typeof fieldValue.toDate === 'function') {
348
- itemDate = fieldValue.toDate();
349
- }
350
- else if (fieldValue instanceof Date) {
351
- itemDate = fieldValue;
352
- }
353
- else if (typeof fieldValue === 'string') {
354
- itemDate = new Date(fieldValue);
355
- if (isNaN(itemDate.getTime())) {
356
- return false;
357
- }
358
- }
359
- else if (typeof fieldValue === 'number') {
360
- itemDate = new Date(fieldValue);
361
- }
362
- else {
363
- return false;
364
- }
365
- return itemDate >= initialDate && itemDate <= finalDate;
366
- }
367
- catch (error) {
368
- console.warn('Erro ao processar filtro de data para o item:', item.id, error);
369
- return false;
370
- }
371
- });
372
- }
373
- catch (error) {
374
- console.warn('Erro ao processar datas do filtro:', error);
375
- }
376
- }
377
- // Se as datas não estiverem preenchidas, não aplicar filtro (retornar todos os itens)
378
- }
379
- });
380
- return filteredItems;
381
- }
382
- // Handler para mudanças de data no filtro
383
- onDateFilterChange() {
384
- if (this.data.pagination === false) {
385
- // Atualizar arrange com os filtros ativos
386
- this.arrange = this.buildArrangeFromFilters();
387
- // Aplicar filtros (ou remover se campos estiverem vazios)
388
- this.applyFiltersToDataSource();
389
- }
390
- }
391
- // Aplicar filtros ao dataSource quando não há paginação
392
- applyFiltersToDataSource() {
393
- if (!this.dataSource)
394
- return;
395
- let filteredItems = this.applyClientSideFilters([...this.items]);
396
- this.dataSource.data = filteredItems;
397
- // Atualizar filteredItems com os dados filtrados (será atualizado novamente no handleDownload com filteredData)
398
- this.filteredItems = filteredItems;
399
- this.totalItems = filteredItems.length;
400
- }
401
- // Métodos com paginação
402
- async loadItemsPaginated(navigation = 'reload', reset = false) {
403
- if (reset && ['forward', 'reload'].includes(navigation)) {
404
- this.lastDoc = null;
405
- this.currentClientPageIndex = 0;
406
- }
407
- if (reset && ['backward', 'reload'].includes(navigation)) {
408
- this.firstDoc = null;
409
- this.currentClientPageIndex = 0;
410
- }
411
- const activeFilters = this.filtersForm.controls
412
- .flatMap((control) => {
413
- const group = control;
414
- const selectedFilter = group.get('selectFilter')?.value;
415
- if (!selectedFilter)
416
- return [];
417
- const arrange = selectedFilter.arrange;
418
- if (arrange === 'filter') {
419
- const filterValue = group.get('typeFilter')?.value;
420
- if (!filterValue)
421
- return [];
422
- return {
423
- arrange,
424
- filter: {
425
- property: selectedFilter.property,
426
- filtering: filterValue,
427
- },
428
- dateFilter: undefined,
429
- };
430
- }
431
- if (arrange === 'filterByDate') {
432
- const initial = group.get('initialDate')?.value;
433
- const final = group.get('finalDate')?.value;
434
- if (initial && final) {
435
- const [dayI, monthI, yearI] = initial.split('/');
436
- const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
437
- const [dayF, monthF, yearF] = final.split('/');
438
- const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
439
- finalDate.setHours(23, 59, 59);
440
- return {
441
- arrange,
442
- filter: undefined,
443
- dateFilter: {
444
- initial: initialDate,
445
- final: finalDate,
446
- },
447
- };
448
- }
449
- return [];
450
- }
451
- if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {
452
- const selectedItems = group.get('selectItem')?.value;
453
- if (Array.isArray(selectedItems) && selectedItems.length > 0) {
454
- return selectedItems.map((item) => ({
455
- arrange,
456
- filter: {
457
- property: item.property,
458
- filtering: item.value,
459
- },
460
- dateFilter: undefined,
461
- }));
462
- }
463
- }
464
- return [];
465
- })
466
- .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));
467
- this.arrange = {
468
- filters: activeFilters,
469
- sortBy: this.sortBy,
470
- };
471
- const paginated = {
472
- batchSize: this.pageSize,
473
- collection: this.data.collection,
474
- doc: { lastDoc: this.lastDoc, firstDoc: this.firstDoc },
475
- navigation,
476
- arrange: this.arrange,
477
- conditions: this.data.conditions,
478
- size: this.totalItems,
479
- filterFn: this.data.filterFn,
480
- clientPageIndex: this.currentClientPageIndex,
481
- };
482
- const result = await this.tableService.getPaginated(paginated);
483
- this.items = result.items;
484
- await this.loadRelations();
485
- await this.loadQueryLengths();
486
- this.lastDoc = result.lastDoc;
487
- this.firstDoc = result.firstDoc;
488
- // Atualizar currentClientPageIndex se retornado pelo fallback
489
- if (result.currentClientPageIndex !== undefined) {
490
- this.currentClientPageIndex = result.currentClientPageIndex;
491
- }
492
- let sum = 0;
493
- if (this.data.totalRef) {
494
- for (const totalRef of this.data.totalRef) {
495
- const totalRefDoc = await totalRef.ref.get();
496
- const docData = totalRefDoc.data();
497
- if (docData || result.filterLength) {
498
- sum =
499
- result.filterLength ??
500
- (sum + (docData ? docData[totalRef.field] : 0));
501
- }
502
- }
503
- this.totalItems = sum;
504
- }
505
- this.hasNextPage = result.hasNextPage;
506
- this.dataSource = new MatTableDataSource(this.items);
507
- this.filterPredicate = this.dataSource.filterPredicate;
508
- }
509
- async onPageChange(event) {
510
- if (this.data.pagination === true && event) {
511
- this.isLoading = true;
512
- const previousPageIndex = event.previousPageIndex ?? 0;
513
- const pageIndex = event.pageIndex;
514
- const currentComponentPageSize = this.pageSize;
515
- const eventPageSize = event.pageSize;
516
- const totalItems = this.totalItems;
517
- let navigationDirection;
518
- let resetDocs = false;
519
- let originalPageSize = null;
520
- const lastPageIndex = Math.max(0, Math.ceil(totalItems / eventPageSize) - 1);
521
- // Atualizar currentClientPageIndex sempre para o fallback
522
- this.currentClientPageIndex = pageIndex;
523
- if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {
524
- this.currentPageNumber++;
525
- }
526
- else if (previousPageIndex !== undefined &&
527
- pageIndex < previousPageIndex) {
528
- this.currentPageNumber = Math.max(1, this.currentPageNumber - 1);
529
- }
530
- if (eventPageSize !== currentComponentPageSize) {
531
- console.log('Alterou a quantidade de elementos exibidos por página');
532
- this.pageSize = eventPageSize;
533
- navigationDirection = 'forward';
534
- resetDocs = true;
535
- this.currentClientPageIndex = 0;
536
- }
537
- else if (pageIndex === 0 &&
538
- previousPageIndex !== undefined &&
539
- pageIndex < previousPageIndex) {
540
- console.log('Pulou para a primeira página');
541
- navigationDirection = 'forward';
542
- this.currentPageNumber = 1;
543
- this.currentClientPageIndex = 0;
544
- resetDocs = true;
545
- }
546
- else if (pageIndex === lastPageIndex &&
547
- previousPageIndex !== undefined &&
548
- pageIndex > previousPageIndex &&
549
- pageIndex - previousPageIndex > 1) {
550
- console.log('Pulou para a ultima página');
551
- navigationDirection = 'backward';
552
- resetDocs = true;
553
- const itemsExpectedInLastPage = totalItems - lastPageIndex * eventPageSize;
554
- if (itemsExpectedInLastPage > 0 &&
555
- itemsExpectedInLastPage < eventPageSize) {
556
- originalPageSize = this.pageSize;
557
- this.pageSize = itemsExpectedInLastPage;
558
- }
559
- }
560
- else if (previousPageIndex !== undefined &&
561
- pageIndex > previousPageIndex) {
562
- console.log('Procedendo');
563
- navigationDirection = 'forward';
564
- resetDocs = false;
565
- }
566
- else if (previousPageIndex !== undefined &&
567
- pageIndex < previousPageIndex) {
568
- console.log('Retrocedendo.');
569
- navigationDirection = 'backward';
570
- resetDocs = false;
571
- }
572
- else if (previousPageIndex !== undefined &&
573
- pageIndex === previousPageIndex) {
574
- console.log('Recarregando.');
575
- navigationDirection = 'reload';
576
- resetDocs = false;
577
- }
578
- else if (previousPageIndex === undefined && pageIndex === 0) {
579
- console.log('Evento inicial do paginador para pág 0. ngOnInit carregou.');
580
- this.isLoading = false;
581
- if (event)
582
- this.pageEvent = event;
583
- return;
584
- }
585
- else {
586
- console.warn('INESPERADO! Condição de navegação não tratada:', event);
587
- this.isLoading = false;
588
- if (event)
589
- this.pageEvent = event;
590
- return;
591
- }
592
- if (navigationDirection) {
593
- try {
594
- await this.loadItemsPaginated(navigationDirection, resetDocs);
595
- }
596
- catch (error) {
597
- console.error('Erro ao carregar itens paginados:', error);
598
- }
599
- finally {
600
- if (originalPageSize !== null) {
601
- this.pageSize = originalPageSize;
602
- }
603
- }
604
- }
605
- if (event)
606
- this.pageEvent = event;
607
- this.isLoading = false;
608
- }
609
- }
610
- // Outros métodos
611
- applyFilter(value) {
612
- // Sem paginação
613
- if (this.data.pagination === false) {
614
- this.dataSource.filter = String(value).trim().toLowerCase();
615
- }
616
- // Com paginação
617
- if (this.data.pagination === true) {
618
- this.filterValue = value;
619
- this.filterSubject.next(this.filterValue);
620
- }
621
- }
622
- goToDetails(row) {
623
- if (this.data.isNotClickable) {
624
- return;
625
- }
626
- const urlPath = this.data.url || this.data.name;
627
- const url = this.router.serializeUrl(this.router.createUrlTree([`/${urlPath}`, row.id]));
628
- window.open(url, '_blank');
629
- }
630
- async getRelation(params) {
631
- try {
632
- let snapshot;
633
- if (params.id !== '' &&
634
- params.id !== undefined &&
635
- params.collection !== undefined &&
636
- params.collection !== '') {
637
- snapshot = await firstValueFrom(this.firestore.collection(params.collection).doc(params.id).get());
638
- }
639
- if (snapshot && snapshot.exists) {
640
- const data = snapshot.data();
641
- return data?.[params.newProperty] ?? '';
642
- }
643
- return '';
644
- }
645
- catch (e) {
646
- console.log(e);
647
- return '';
648
- }
649
- }
650
- async loadRelations() {
651
- const relationPromises = this.data.displayedColumns
652
- .filter((col) => col.relation)
653
- .flatMap((col) => this.items.map(async (item) => {
654
- if (col.relation) {
655
- item[col.property] = await this.getRelation({
656
- id: item[col.relation.property],
657
- collection: col.relation.collection,
658
- newProperty: col.relation.newProperty,
659
- });
660
- }
661
- }));
662
- await Promise.all(relationPromises);
663
- }
664
- async getQueryLength(params) {
665
- const snapshot = await this.firestore
666
- .collection(params.relation.collection)
667
- .ref.where(params.relation.property, params.relation.operator, params.item[params.relation.value])
668
- .get();
669
- return snapshot.size;
670
- }
671
- async loadQueryLengths() {
672
- const lengthPromises = this.data.displayedColumns
673
- .filter((col) => col.queryLength)
674
- .flatMap((col) => this.items.map(async (item) => {
675
- if (col.queryLength) {
676
- item[col.property] = await this.getQueryLength({
677
- item: item,
678
- relation: col.queryLength,
679
- });
680
- }
681
- }));
682
- await Promise.all(lengthPromises);
683
- }
684
- filterItems() {
685
- if (this.data.conditions) {
686
- this.data.conditions.forEach((cond) => {
687
- this.items = this.items.filter((item) => {
688
- const operatorFunction = this.tableService.operators[cond.operator];
689
- if (operatorFunction) {
690
- return operatorFunction(item[cond.firestoreProperty], cond.dashProperty);
691
- }
692
- return false;
693
- });
694
- });
695
- }
696
- }
697
- buildArrangeFromFilters() {
698
- const activeFilters = this.filtersForm.controls
699
- .flatMap((control) => {
700
- const group = control;
701
- const selectedFilter = group.get('selectFilter')?.value;
702
- if (!selectedFilter)
703
- return [];
704
- const arrange = selectedFilter.arrange;
705
- if (arrange === 'filter') {
706
- const filterValue = group.get('typeFilter')?.value;
707
- if (!filterValue)
708
- return [];
709
- return {
710
- arrange,
711
- filter: {
712
- property: selectedFilter.property,
713
- filtering: filterValue,
714
- },
715
- dateFilter: undefined,
716
- };
717
- }
718
- if (arrange === 'filterByDate') {
719
- const initial = group.get('initialDate')?.value;
720
- const final = group.get('finalDate')?.value;
721
- if (initial && final) {
722
- try {
723
- const [dayI, monthI, yearI] = initial.split('/');
724
- const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
725
- const [dayF, monthF, yearF] = final.split('/');
726
- const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
727
- finalDate.setHours(23, 59, 59);
728
- return {
729
- arrange,
730
- filter: undefined,
731
- dateFilter: {
732
- initial: initialDate,
733
- final: finalDate,
734
- },
735
- };
736
- }
737
- catch (error) {
738
- return [];
739
- }
740
- }
741
- return [];
742
- }
743
- if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {
744
- const selectedItems = group.get('selectItem')?.value;
745
- if (Array.isArray(selectedItems) && selectedItems.length > 0) {
746
- return selectedItems.map((item) => ({
747
- arrange,
748
- filter: {
749
- property: item.property,
750
- filtering: item.value,
751
- },
752
- dateFilter: undefined,
753
- }));
754
- }
755
- }
756
- return [];
757
- })
758
- .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));
759
- return {
760
- filters: activeFilters,
761
- sortBy: this.sortBy,
762
- };
763
- }
764
- // Filtro de data
765
- async search() {
766
- if (this.selectSort.value) {
767
- if (this.selectSort.value.arrange === 'ascending') {
768
- this.sortBy = {
769
- field: this.selectSort.value.property,
770
- order: 'asc',
771
- };
772
- }
773
- if (this.selectSort.value.arrange === 'descending') {
774
- this.sortBy = {
775
- field: this.selectSort.value.property,
776
- order: 'desc',
777
- };
778
- }
779
- }
780
- // Sem paginação: aplicar filtros client-side
781
- if (this.data.pagination === false) {
782
- // Atualizar arrange com os filtros ativos
783
- this.arrange = this.buildArrangeFromFilters();
784
- this.applyFiltersToDataSource();
785
- this.currentArrange =
786
- this.filtersForm.length > 0
787
- ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''
788
- : '';
789
- }
790
- else {
791
- // Com paginação: comportamento original
792
- await this.loadItemsPaginated('reload', true);
793
- this.currentArrange =
794
- this.filtersForm.length > 0
795
- ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''
796
- : '';
797
- this.paginator.firstPage();
798
- }
799
- }
800
- async resetFilter() {
801
- this.dataSource.filter = '';
802
- if (this.filterPredicate) {
803
- this.dataSource.filterPredicate = this.filterPredicate;
804
- }
805
- this.filtersForm.clear();
806
- this.addFilter();
807
- this.selectSort.patchValue('');
808
- this.sortBy = {
809
- order: this.data.sortBy ? this.data.sortBy.order : 'desc',
810
- field: this.data.sortBy ? this.data.sortBy.field : 'createdAt',
811
- };
812
- // Sem paginação: recarregar itens originais
813
- if (this.data.pagination === false) {
814
- // Resetar arrange para valores padrão
815
- this.arrange = {
816
- filters: [],
817
- sortBy: this.sortBy,
818
- };
819
- this.dataSource.data = [...this.items];
820
- this.totalItems = this.items.length;
821
- this.currentArrange = '';
822
- }
823
- else {
824
- // Com paginação: comportamento original
825
- await this.loadItemsPaginated('reload', true);
826
- this.currentArrange = '';
827
- this.paginator.firstPage();
828
- }
829
- }
830
- // Método público para recarregar a tabela
831
- async reloadTable() {
832
- if (this.data.pagination) {
833
- await this.loadItemsPaginated('reload', true);
834
- this.paginator.firstPage();
835
- }
836
- else {
837
- await this.loadItems();
838
- }
839
- }
840
- updateDisplayedColumns() {
841
- if (this.dataSource) {
842
- this.dataSource = new MatTableDataSource([]);
843
- }
844
- this.columnProperties = this.data.displayedColumns.map((column) => {
845
- return column.property;
846
- });
847
- this.dropdownItems = [];
848
- this.sortableDropdownItems = [];
849
- this.data.displayedColumns.forEach((col) => {
850
- if (col.isFilterable) {
851
- this.dropdownItems.push({
852
- ...col,
853
- arrange: 'filter',
854
- title: col.title,
855
- });
856
- }
857
- if (col.isSortable) {
858
- this.sortableDropdownItems.push({
859
- ...col,
860
- arrange: 'ascending',
861
- title: col.title + ': crescente',
862
- });
863
- this.sortableDropdownItems.push({
864
- ...col,
865
- arrange: 'descending',
866
- title: col.title + ': decrescente',
867
- });
868
- }
869
- if (col.isFilterableByDate) {
870
- this.dropdownItems.push({
871
- ...col,
872
- arrange: 'filterByDate',
873
- title: col.title + ': filtro por data',
874
- });
875
- }
876
- });
877
- if (this.data.filterableOptions &&
878
- Array.isArray(this.data.filterableOptions)) {
879
- this.data.filterableOptions.forEach((option) => this.dropdownItems.push({ ...option, arrange: 'equals' }));
880
- }
881
- }
882
- isString(value) {
883
- return typeof value === 'string';
884
- }
885
- // Métodos para controle do tooltip
886
- onCellMouseEnter(event, row, col) {
887
- // mostrar tooltip se a coluna tiver charLimit definido
888
- if (!col.charLimit) {
889
- return;
890
- }
891
- const fullValue = this.getDisplayValue(col, row, true);
892
- // mostrar tooltip se o valor completo for maior que o limite
893
- if (fullValue.length <= col.charLimit) {
894
- return;
895
- }
896
- this.hoveredCell = { row, col };
897
- this.tooltipContent = fullValue;
898
- // Definir posição do tooltip
899
- this.tooltipPosition = {
900
- x: event.clientX + 10,
901
- y: event.clientY - 10,
902
- };
903
- // Timeout para mostrar o tooltip
904
- this.tooltipTimeout = setTimeout(() => {
905
- if (this.hoveredCell &&
906
- this.hoveredCell.row === row &&
907
- this.hoveredCell.col === col) {
908
- this.showTooltip = true;
909
- }
910
- }, 500);
911
- }
912
- onCellMouseLeave() {
913
- if (this.tooltipTimeout) {
914
- clearTimeout(this.tooltipTimeout);
915
- this.tooltipTimeout = null;
916
- }
917
- this.showTooltip = false;
918
- this.hoveredCell = null;
919
- this.tooltipContent = '';
920
- }
921
- onCellMouseMove(event) {
922
- if (this.showTooltip) {
923
- this.tooltipPosition = {
924
- x: event.clientX + 10,
925
- y: event.clientY - 10,
926
- };
927
- }
928
- }
929
- // Métodos para inversão vertical dos tabs
930
- getTabGroups(tabs) {
931
- if (!tabs || tabs.length === 0)
932
- return [];
933
- const totalGroups = Math.ceil(tabs.length / 6);
934
- const groups = [];
935
- // Criar array de índices invertidos (último grupo primeiro)
936
- for (let i = totalGroups - 1; i >= 0; i--) {
937
- groups.push(i);
938
- }
939
- return groups;
940
- }
941
- getTabGroup(tabs, groupIndex) {
942
- if (!tabs || tabs.length === 0)
943
- return [];
944
- const startIndex = groupIndex * 6;
945
- const endIndex = Math.min(startIndex + 6, tabs.length);
946
- return tabs.slice(startIndex, endIndex);
947
- }
948
- getRealTabIndex(groupIndex, tabIndexInGroup) {
949
- if (!this.data.tabs?.tabsData)
950
- return 0;
951
- const totalGroups = Math.ceil(this.data.tabs.tabsData.length / 6);
952
- const realGroupIndex = totalGroups - 1 - groupIndex;
953
- return realGroupIndex * 6 + tabIndexInGroup;
954
- }
955
- onTableSelected(i, j) {
956
- if (!this.data.tabs?.tabsData || !this.data.tabs.method)
957
- return;
958
- this.selectedTab = this.getRealTabIndex(i, j);
959
- const tab = this.data.tabs.tabsData[this.selectedTab];
960
- if (tab) {
961
- this.data.tabs.method(tab, this.selectedTab);
962
- }
963
- }
964
- isTabSelected(originalIndex) {
965
- return this.selectedTab === originalIndex;
966
- }
967
- shouldShowActionButton() {
968
- if (!this.data?.actionButton) {
969
- return false;
970
- }
971
- if (!this.data.actionButton.condition) {
972
- return true;
973
- }
974
- try {
975
- return this.data.actionButton.condition(null) ?? true;
976
- }
977
- catch {
978
- return true;
979
- }
980
- }
981
- // Método para lidar com download (diferente para paginado e não paginado)
982
- handleDownload() {
983
- if (!this.downloadTable)
984
- return;
985
- // Se não há paginação, usar dados filtrados do dataSource
986
- if (this.data.pagination === false) {
987
- // Atualizar filteredItems com os dados filtrados do dataSource
988
- // (que já inclui filtro de texto aplicado via filterPredicate)
989
- if (this.dataSource && this.dataSource.filteredData) {
990
- this.filteredItems = [...this.dataSource.filteredData];
991
- }
992
- // Construir arrange com os filtros ativos (se houver)
993
- const arrange = this.buildArrangeFromFilters();
994
- this.downloadTable(arrange, this.data.conditions || []);
995
- }
996
- else {
997
- // Modo paginado: usar arrange existente (comportamento original)
998
- if (this.arrange) {
999
- this.downloadTable(this.arrange, this.data.conditions || []);
1000
- }
1001
- }
1002
- }
1003
- }
1004
- 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.AngularFirestore }], target: i0.ɵɵFactoryTarget.Component });
1005
- 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 }], 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, Custom Action, and Global Actions -->\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\r\n <!-- Right Side: Search, Reset, Export -->\r\n <div\r\n class=\"flex flex-wrap gap-3\"\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\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 rounded-lg border border-gray-200 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 *ngFor=\"let item of dropdownItems\" [value]=\"item\">\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()\"\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 matInput\r\n (keyup.enter)=\"search()\"\r\n formControlName=\"initialDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 (keyup.enter)=\"search()\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 & Sort -->\r\n <div\r\n class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\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 <!-- Sort Dropdown -->\r\n <div\r\n class=\"w-full sm:w-auto sm:min-w-[250px]\"\r\n *ngIf=\"sortableDropdownItems.length > 0\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>Ordenar por</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 <!-- SIMPLE SEARCH (for non-paginated tables) -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"data.pagination === false && hasFilterableColumn === true\"\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()\"\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 <button\r\n *ngIf=\"data.actionButton\"\r\n [ngClass]=\"\r\n (data.actionButton.colorClass || 'bg-blue-500') +\r\n ' float-right 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 *ngIf=\"data.actionButton.icon\" [class]=\"data.actionButton.icon\"></i>\r\n {{ data.actionButton.label }}\r\n </button>\r\n </div>\r\n\r\n <!-- FILTERS PANEL (for non-paginated tables) -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"data.pagination === false && dropdownItems.length > 0\"\r\n >\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 rounded-lg border border-gray-200 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 *ngFor=\"let item of dropdownItems\" [value]=\"item\">\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()\"\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 matInput\r\n (keyup.enter)=\"search()\"\r\n (blur)=\"onDateFilterChange()\"\r\n formControlName=\"initialDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 (keyup.enter)=\"search()\"\r\n (blur)=\"onDateFilterChange()\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\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 class=\"flex flex-wrap gap-3\">\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 Aplicar\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\r\n <div class=\"flex flex-col\">\r\n <div\r\n class=\"mx-auto flex flex-col\"\r\n *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\r\n >\r\n <!-- Calcular quantos grupos de 6 tabs existem -->\r\n <ng-container\r\n *ngFor=\"\r\n let groupIndex of getTabGroups(data.tabs.tabsData);\r\n let i = index\r\n \"\r\n >\r\n <div class=\"mx-auto flex flex-row\">\r\n <ng-container\r\n *ngFor=\"\r\n let tab of getTabGroup(data.tabs.tabsData, groupIndex);\r\n let j = index\r\n \"\r\n >\r\n <button\r\n class=\"border-2 border-gray-300 bg-gray-200 px-4 py-2 font-medium transition hover:brightness-95\"\r\n [ngClass]=\"\r\n isTabSelected(getRealTabIndex(i, j))\r\n ? 'border-b-0 brightness-110'\r\n : ''\r\n \"\r\n (click)=\"onTableSelected(i, j)\"\r\n >\r\n {{ tab.label }}\r\n <span\r\n *ngIf=\"tab.counter !== undefined\"\r\n class=\"ml-2 text-xs font-bold\"\r\n [ngClass]=\"tab.counterClass\"\r\n >\r\n {{ tab.counter }}\r\n </span>\r\n </button>\r\n </ng-container>\r\n </div>\r\n </ng-container>\r\n </div>\r\n <div class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl\">\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\"\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 [ngClass]=\"\r\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\r\n (data.color?.text ? ' ' + $any(data.color).text : '')\r\n \"\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 [ngClass]=\"\r\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\r\n (data.color?.text ? ' ' + $any(data.color).text : '')\r\n \"\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 <img\r\n *ngIf=\"\r\n col.image && col.image.path && !col.iconClass && !col.method\r\n \"\r\n [src]=\"col.image.path + '/' + row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n alt=\"Imagem\"\r\n />\r\n <img\r\n *ngIf=\"\r\n col.image && col.image.url && !col.iconClass && !col.method\r\n \"\r\n [src]=\"row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n alt=\"Imagem\"\r\n />\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\" 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\">\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) || this.data.filterFn\r\n }\"\r\n >\r\n </mat-paginator>\r\n <div\r\n *ngIf=\"\r\n !isLoading &&\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 <div\r\n *ngIf=\"showTooltip\"\r\n class=\"fixed z-50 max-w-md break-words rounded-lg bg-gray-800 px-3 py-2 text-sm text-white shadow-lg\"\r\n [style.left.px]=\"tooltipPosition.x\"\r\n [style.top.px]=\"tooltipPosition.y\"\r\n [style.pointer-events]=\"'none'\"\r\n >\r\n {{ tooltipContent }}\r\n </div>\r\n</div>\r\n", styles: ["::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}\n"], dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i5.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: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i5.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i5.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i5.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i5.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: i6.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i6.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i6.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i6.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { kind: "directive", type: i6.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i6.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i6.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i6.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i6.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i6.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: i6.MatNoDataRow, selector: "ng-template[matNoDataRow]" }, { kind: "component", type: i7.MatPaginator, selector: "mat-paginator", inputs: ["disabled"], exportAs: ["matPaginator"] }, { kind: "directive", type: i8.MatSort, selector: "[matSort]", inputs: ["matSortDisabled", "matSortActive", "matSortStart", "matSortDirection", "matSortDisableClear"], outputs: ["matSortChange"], exportAs: ["matSort"] }, { kind: "component", type: i8.MatSortHeader, selector: "[mat-sort-header]", inputs: ["disabled", "mat-sort-header", "arrowPosition", "start", "sortActionDescription", "disableClear"], exportAs: ["matSortHeader"] }, { kind: "component", type: i9.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i9.MatLabel, selector: "mat-label" }, { kind: "directive", type: i9.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "directive", type: i10.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: i11.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator"], exportAs: ["matSelect"] }, { kind: "component", type: i12.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i13.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "component", type: i14.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: i15.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i16.NgxMaskDirective, selector: "input[mask], textarea[mask]", inputs: ["mask", "specialCharacters", "patterns", "prefix", "suffix", "thousandSeparator", "decimalMarker", "dropSpecialCharacters", "hiddenInput", "showMaskTyped", "placeHolderCharacter", "shownMaskExpression", "showTemplate", "clearIfNotMatch", "validation", "separatorLimit", "allowNegativeNumbers", "leadZeroDateTime", "leadZero", "triggerOnMaskChange", "apm", "inputTransformFn", "outputTransformFn", "keepCharacterPositions"], outputs: ["maskFilled"], exportAs: ["mask", "ngxMask"] }] });
1006
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableComponent, decorators: [{
1007
- type: Component,
1008
- args: [{ selector: 'lib-table', 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, Custom Action, and Global Actions -->\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\r\n <!-- Right Side: Search, Reset, Export -->\r\n <div\r\n class=\"flex flex-wrap gap-3\"\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\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 rounded-lg border border-gray-200 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 *ngFor=\"let item of dropdownItems\" [value]=\"item\">\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()\"\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 matInput\r\n (keyup.enter)=\"search()\"\r\n formControlName=\"initialDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 (keyup.enter)=\"search()\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 & Sort -->\r\n <div\r\n class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\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 <!-- Sort Dropdown -->\r\n <div\r\n class=\"w-full sm:w-auto sm:min-w-[250px]\"\r\n *ngIf=\"sortableDropdownItems.length > 0\"\r\n >\r\n <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n <mat-label>Ordenar por</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 <!-- SIMPLE SEARCH (for non-paginated tables) -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"data.pagination === false && hasFilterableColumn === true\"\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()\"\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 <button\r\n *ngIf=\"data.actionButton\"\r\n [ngClass]=\"\r\n (data.actionButton.colorClass || 'bg-blue-500') +\r\n ' float-right 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 *ngIf=\"data.actionButton.icon\" [class]=\"data.actionButton.icon\"></i>\r\n {{ data.actionButton.label }}\r\n </button>\r\n </div>\r\n\r\n <!-- FILTERS PANEL (for non-paginated tables) -->\r\n <div\r\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n *ngIf=\"data.pagination === false && dropdownItems.length > 0\"\r\n >\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 rounded-lg border border-gray-200 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 *ngFor=\"let item of dropdownItems\" [value]=\"item\">\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()\"\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 matInput\r\n (keyup.enter)=\"search()\"\r\n (blur)=\"onDateFilterChange()\"\r\n formControlName=\"initialDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 (keyup.enter)=\"search()\"\r\n (blur)=\"onDateFilterChange()\"\r\n matInput\r\n formControlName=\"finalDate\"\r\n [dropSpecialCharacters]=\"false\"\r\n mask=\"d0/M0/0000\"\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 flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\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 class=\"flex flex-wrap gap-3\">\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 Aplicar\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\r\n <div class=\"flex flex-col\">\r\n <div\r\n class=\"mx-auto flex flex-col\"\r\n *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\r\n >\r\n <!-- Calcular quantos grupos de 6 tabs existem -->\r\n <ng-container\r\n *ngFor=\"\r\n let groupIndex of getTabGroups(data.tabs.tabsData);\r\n let i = index\r\n \"\r\n >\r\n <div class=\"mx-auto flex flex-row\">\r\n <ng-container\r\n *ngFor=\"\r\n let tab of getTabGroup(data.tabs.tabsData, groupIndex);\r\n let j = index\r\n \"\r\n >\r\n <button\r\n class=\"border-2 border-gray-300 bg-gray-200 px-4 py-2 font-medium transition hover:brightness-95\"\r\n [ngClass]=\"\r\n isTabSelected(getRealTabIndex(i, j))\r\n ? 'border-b-0 brightness-110'\r\n : ''\r\n \"\r\n (click)=\"onTableSelected(i, j)\"\r\n >\r\n {{ tab.label }}\r\n <span\r\n *ngIf=\"tab.counter !== undefined\"\r\n class=\"ml-2 text-xs font-bold\"\r\n [ngClass]=\"tab.counterClass\"\r\n >\r\n {{ tab.counter }}\r\n </span>\r\n </button>\r\n </ng-container>\r\n </div>\r\n </ng-container>\r\n </div>\r\n <div class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl\">\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\"\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 [ngClass]=\"\r\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\r\n (data.color?.text ? ' ' + $any(data.color).text : '')\r\n \"\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 [ngClass]=\"\r\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\r\n (data.color?.text ? ' ' + $any(data.color).text : '')\r\n \"\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 <img\r\n *ngIf=\"\r\n col.image && col.image.path && !col.iconClass && !col.method\r\n \"\r\n [src]=\"col.image.path + '/' + row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n alt=\"Imagem\"\r\n />\r\n <img\r\n *ngIf=\"\r\n col.image && col.image.url && !col.iconClass && !col.method\r\n \"\r\n [src]=\"row[col.property]\"\r\n [ngClass]=\"col.image.class\"\r\n alt=\"Imagem\"\r\n />\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\" 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\">\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) || this.data.filterFn\r\n }\"\r\n >\r\n </mat-paginator>\r\n <div\r\n *ngIf=\"\r\n !isLoading &&\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 <div\r\n *ngIf=\"showTooltip\"\r\n class=\"fixed z-50 max-w-md break-words rounded-lg bg-gray-800 px-3 py-2 text-sm text-white shadow-lg\"\r\n [style.left.px]=\"tooltipPosition.x\"\r\n [style.top.px]=\"tooltipPosition.y\"\r\n [style.pointer-events]=\"'none'\"\r\n >\r\n {{ tooltipContent }}\r\n </div>\r\n</div>\r\n", styles: ["::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}\n"] }]
1009
- }], ctorParameters: function () { return [{ type: i1.Router }, { type: i2.TableService }, { type: i3.AngularFirestore }]; }, propDecorators: { data: [{
1010
- type: Input
1011
- }], downloadTable: [{
1012
- type: Input
1013
- }], paginator: [{
1014
- type: ViewChild,
1015
- args: [MatPaginator]
1016
- }], sort: [{
1017
- type: ViewChild,
1018
- args: [MatSort]
1019
- }] } });
1020
- //# 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":"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","sourcesContent":["import { Arrange, Condition, Tab, TabData, TableData } from '../../types/Table';\nimport { Component, Input, OnInit, ViewChild } from '@angular/core';\nimport {\n  AngularFirestore,\n  QueryDocumentSnapshot,\n} from '@angular/fire/compat/firestore';\nimport { MatPaginator, PageEvent } from '@angular/material/paginator';\nimport { MatSort } from '@angular/material/sort';\nimport { MatTableDataSource } from '@angular/material/table';\nimport { Router } from '@angular/router';\nimport { TableService } from '../../services/table.service';\nimport { debounceTime, firstValueFrom, Subject } from 'rxjs';\nimport { OrderByDirection } from 'firebase/firestore';\nimport firebase from 'firebase/compat';\nimport WhereFilterOp = firebase.firestore.WhereFilterOp;\nimport firestore = firebase.firestore;\nimport { FormControl, FormArray, FormGroup } from '@angular/forms';\n\n@Component({\n  selector: 'lib-table',\n  templateUrl: './table.component.html',\n  styleUrls: ['./table.component.scss'],\n})\nexport class TableComponent implements OnInit {\n  // INPUTS\n  @Input() data!: TableData;\n  @Input() downloadTable?: (arrange: Arrange, conditions: Condition[]) => void;\n  arrange: Arrange | null = null;\n  // VARIABLES\n  dataSource!: MatTableDataSource<any>;\n  currentPageNumber = 1;\n  currentClientPageIndex = 0;\n\n  public items: any[] = [];\n  public filteredItems: any[] = []; // Dados filtrados para modo não paginado (público para acesso externo)\n  public isLoading = false;\n  private lastDoc: QueryDocumentSnapshot<any> | null = null;\n  private firstDoc: QueryDocumentSnapshot<any> | null = null;\n  private sortBy: { field: string; order: OrderByDirection } = {\n    field: 'createdAt',\n    order: 'desc',\n  };\n  public columnProperties: string[] = [];\n\n  filtersForm: FormArray;\n\n  selectSort: FormControl = new FormControl('');\n  currentArrange = '';\n\n  hasNextPage: boolean = false;\n\n  dropdownItems: any[] = [];\n  sortableDropdownItems: any[] = [];\n  pageSize = 25;\n  totalItems = 0;\n  pageEvent?: PageEvent;\n  filterValue: string | null = null;\n\n  hasFilterableColumn: boolean = false;\n  hasSortableColumn: boolean = false;\n  filterPredicate: ((data: any, filter: string) => boolean) | undefined;\n  private filterSubject = new Subject<string>();\n  private readonly debounceTimeMs = 500;\n\n  selectedTab: number = 0;\n\n  @ViewChild(MatPaginator) paginator!: MatPaginator;\n  @ViewChild(MatSort) sort!: MatSort;\n\n  // Propriedades para controle do tooltip\n  hoveredCell: { row: any; col: any } | null = null;\n  showTooltip: boolean = false;\n  tooltipTimeout: any;\n  tooltipContent: string = '';\n  tooltipPosition: { x: number; y: number } = { x: 0, y: 0 };\n\n  // CONSTRUCTOR\n  constructor(\n    private router: Router,\n    private tableService: TableService,\n    private firestore: AngularFirestore\n  ) {\n    this.filtersForm = new FormArray([this.createFilterGroup()]);\n  }\n\n  createFilterGroup(): FormGroup {\n    return new FormGroup({\n      selectFilter: new FormControl(''),\n      typeFilter: new FormControl(''),\n      selectItem: new FormControl(''),\n      initialDate: new FormControl('', this.tableService.dateFormatValidator()),\n      finalDate: new FormControl('', this.tableService.dateFormatValidator()),\n    });\n  }\n\n  addFilter(filterData?: {\n    selectFilter: any;\n    typeFilter?: string;\n    selectItem?: any[];\n    initialDate?: string;\n    finalDate?: string;\n  }): void {\n    const newFilterGroup = this.createFilterGroup();\n\n    if (filterData) {\n      if (filterData.selectFilter) {\n        newFilterGroup.get('selectFilter')?.setValue(filterData.selectFilter);\n      }\n      if (filterData.typeFilter) {\n        newFilterGroup.get('typeFilter')?.setValue(filterData.typeFilter);\n      }\n      if (filterData.selectItem) {\n        newFilterGroup.get('selectItem')?.setValue(filterData.selectItem);\n      }\n      if (filterData.initialDate) {\n        newFilterGroup.get('initialDate')?.setValue(filterData.initialDate);\n      }\n      if (filterData.finalDate) {\n        newFilterGroup.get('finalDate')?.setValue(filterData.finalDate);\n      }\n    }\n\n    this.filtersForm.push(newFilterGroup);\n  }\n\n  onSelectFilterChange() {\n    const lastIndex = this.filtersForm.length - 1;\n    const lastFilter = this.filtersForm.at(lastIndex);\n    if (lastFilter.get('selectFilter')?.value) {\n      this.addFilter();\n    }\n  }\n\n  removeFilter(index: number): void {\n    this.filtersForm.removeAt(index);\n    if (this.filtersForm.length === 0) {\n      this.addFilter();\n    }\n  }\n\n  removeAllFilters(): void {\n    this.filtersForm.clear();\n    this.addFilter();\n    this.resetFilter();\n  }\n\n  // METHODS\n  public async ngOnInit() {\n    if (!this.data.color)\n      this.data.color = { bg: 'bg-primary', text: 'text-black' };\n    this.columnProperties = this.data.displayedColumns.map((column) => {\n      return column.property;\n    });\n\n    if (this.data.actionButton && !this.data.actionButton.condition) {\n      this.data.actionButton.condition = (_row?: any) => true;\n    }\n\n    this.data.displayedColumns.forEach((col) => {\n      if (col.isFilterable) {\n        if (this.hasFilterableColumn === false) this.hasFilterableColumn = true;\n\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filter',\n          title: col.title,\n        });\n      }\n      if (col.isSortable) {\n        if (this.hasSortableColumn === false) this.hasSortableColumn = true;\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'ascending',\n          title: col.title + ': crescente',\n        });\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'descending',\n          title: col.title + ': decrescente',\n        });\n      }\n      if (col.isFilterableByDate) {\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filterByDate',\n          title: col.title + ': filtro por data',\n        });\n      }\n    });\n    if (\n      this.data.filterableOptions &&\n      Array.isArray(this.data.filterableOptions) &&\n      this.data.filterableOptions.length > 0\n    ) {\n      this.data.filterableOptions.forEach((option) =>\n        this.dropdownItems.push({ ...option, arrange: 'equals' })\n      );\n    }\n\n    // Sem paginação\n    if (this.data.pagination === false) {\n      await this.loadItems();\n    }\n    // Com paginação\n    if (this.data.pagination === true) {\n      if (this.data.sortBy)\n        this.sortBy = {\n          field: this.data.sortBy.field,\n          order: this.data.sortBy.order,\n        };\n      this.filterSubject\n        .pipe(debounceTime(this.debounceTimeMs))\n        .subscribe(() => {\n          this.loadItemsPaginated('reload', true);\n        });\n\n      this.isLoading = true;\n      await this.loadItemsPaginated('reload', true);\n\n      this.sort.active = 'createdAt';\n      this.sort.direction = 'desc';\n      this.dataSource.paginator = this.paginator;\n      this.dataSource.sort = this.sort;\n      this.totalItems = 0;\n      if (this.data.totalRef) {\n        for (const totalRef of this.data.totalRef) {\n          const totalRefDoc = await totalRef.ref.get();\n          const docData = totalRefDoc.data() as any;\n          if (docData && docData[totalRef.field])\n            this.totalItems = (this.totalItems +\n              docData[totalRef.field]) as number;\n        }\n      }\n      this.isLoading = false;\n    }\n  }\n\n  getDisplayValue(col: any, row: any, withinLimit: boolean = false): string {\n    let value: any;\n\n    if (col.calculateValue) {\n      value = col.calculateValue(row);\n    } else {\n      value = this.getNestedValue(row, col.property);\n    }\n\n    if (Array.isArray(value) && col.arrayField) {\n      value = this.formatArrayValue(value, col.arrayField);\n    }\n\n    if (col.queryLength && row[col.property]) {\n      value = row[col.property];\n    }\n\n    value = col.pipe ? col.pipe.transform(value) : value;\n\n    if (value === null || value === undefined) {\n      value = '';\n    } else {\n      value = String(value);\n    }\n\n    if (typeof value !== 'string') {\n      value = '';\n    }\n\n    if (withinLimit || !col.charLimit) {\n      return value;\n    }\n\n    return value.substring(0, col.charLimit) + '...';\n  }\n\n  getNestedValue(obj: any, path: string): any {\n    if (!path) return undefined;\n    const properties = path.split('.');\n    return properties.reduce(\n      (acc, currentPart) => acc && acc[currentPart],\n      obj\n    );\n  }\n\n  private formatArrayValue(array: any[], field: string): string {\n    if (!Array.isArray(array) || array.length === 0) {\n      return '';\n    }\n\n    const values = array\n      .map((item) => {\n        if (typeof item === 'object' && item !== null) {\n          return item[field] || '';\n        }\n        return String(item);\n      })\n      .filter((value) => value !== '' && value !== null && value !== undefined);\n\n    return values.join(', ');\n  }\n\n  // Métodos sem paginação\n  private async loadItems() {\n    this.items = await this.tableService.getItems(this.data.collectionRef);\n    if (this.data.conditions) {\n      this.filterItems();\n    }\n    await this.loadRelations();\n    await this.loadQueryLengths();\n    this.totalItems = this.items.length;\n\n    // Inicializar arrange para tabelas não paginadas\n    this.arrange = {\n      filters: [],\n      sortBy: this.data.sortBy || {\n        field: 'createdAt',\n        order: 'desc',\n      },\n    };\n    this.sortBy = this.data.sortBy || {\n      field: 'createdAt',\n      order: 'desc',\n    };\n\n    // Aplicar filtros client-side se existirem\n    let itemsToDisplay = [...this.items];\n    itemsToDisplay = this.applyClientSideFilters(itemsToDisplay);\n    this.filteredItems = itemsToDisplay; // Armazenar dados filtrados\n\n    this.dataSource = new MatTableDataSource(itemsToDisplay);\n    this.dataSource.paginator = this.paginator;\n    this.dataSource.sort = this.sort;\n    if (this.data.sortBy) {\n      this.dataSource.sort.active = this.data.sortBy.field;\n      this.dataSource.sort.direction = this.data.sortBy.order;\n      this.dataSource.sort.sortChange.emit();\n    }\n\n    this.dataSource.filterPredicate = (data, filter: string) => {\n      return this.data.displayedColumns.some((col): boolean => {\n        if (col.filterPredicates) {\n          return col.filterPredicates.some((predicate): boolean => {\n            const propertyValue = data[col.property];\n            if (!propertyValue || typeof propertyValue !== 'object') {\n              return false;\n            }\n\n            const predicateValue = propertyValue[predicate];\n            if (predicateValue === null || predicateValue === undefined) {\n              return false;\n            }\n\n            return String(predicateValue)\n              .trim()\n              .toLocaleLowerCase()\n              .includes(filter);\n          });\n        }\n\n        if (col.property && col.isFilterable) {\n          const propertyValue = data[col.property];\n          if (propertyValue === null || propertyValue === undefined) {\n            return false;\n          }\n\n          return String(propertyValue)\n            .trim()\n            .toLocaleLowerCase()\n            .includes(filter);\n        }\n        return false;\n      });\n    };\n    this.filterPredicate = this.dataSource.filterPredicate;\n  }\n\n  // Aplicar filtros client-side (filtros por data)\n  private applyClientSideFilters(items: any[]): any[] {\n    let filteredItems = [...items];\n\n    // Processar filtros do filtersForm\n    this.filtersForm.controls.forEach((control) => {\n      const group = control as FormGroup;\n      const selectedFilter = group.get('selectFilter')?.value;\n      if (!selectedFilter) return;\n\n      const arrange = selectedFilter.arrange;\n\n      if (arrange === 'filterByDate') {\n        const initial = group.get('initialDate')?.value;\n        const final = group.get('finalDate')?.value;\n\n        // Só aplicar filtro se ambas as datas estiverem preenchidas\n        if (initial && final && initial.trim() && final.trim()) {\n          try {\n            // Validar formato da data (DD/MM/AAAA)\n            const datePattern = /^(\\d{1,2})\\/(\\d{1,2})\\/(\\d{4})$/;\n            if (!datePattern.test(initial) || !datePattern.test(final)) {\n              return; // Ignorar se formato inválido\n            }\n\n            const [dayI, monthI, yearI] = initial.split('/');\n            const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\n            const [dayF, monthF, yearF] = final.split('/');\n            const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\n            finalDate.setHours(23, 59, 59);\n\n            // Validar se as datas são válidas\n            if (isNaN(initialDate.getTime()) || isNaN(finalDate.getTime())) {\n              return; // Ignorar se datas inválidas\n            }\n\n            // Usar o campo da coluna ou o sortBy.field como padrão\n            const dateField = selectedFilter.property || this.sortBy.field;\n\n            filteredItems = filteredItems.filter((item: any) => {\n              try {\n                const fieldValue = item[dateField];\n\n                if (!fieldValue) {\n                  return false;\n                }\n\n                let itemDate: Date;\n                if (typeof fieldValue.toDate === 'function') {\n                  itemDate = fieldValue.toDate();\n                } else if (fieldValue instanceof Date) {\n                  itemDate = fieldValue;\n                } else if (typeof fieldValue === 'string') {\n                  itemDate = new Date(fieldValue);\n                  if (isNaN(itemDate.getTime())) {\n                    return false;\n                  }\n                } else if (typeof fieldValue === 'number') {\n                  itemDate = new Date(fieldValue);\n                } else {\n                  return false;\n                }\n\n                return itemDate >= initialDate && itemDate <= finalDate;\n              } catch (error) {\n                console.warn(\n                  'Erro ao processar filtro de data para o item:',\n                  item.id,\n                  error\n                );\n                return false;\n              }\n            });\n          } catch (error) {\n            console.warn('Erro ao processar datas do filtro:', error);\n          }\n        }\n        // Se as datas não estiverem preenchidas, não aplicar filtro (retornar todos os itens)\n      }\n    });\n\n    return filteredItems;\n  }\n\n  // Handler para mudanças de data no filtro\n  onDateFilterChange(): void {\n    if (this.data.pagination === false) {\n      // Atualizar arrange com os filtros ativos\n      this.arrange = this.buildArrangeFromFilters();\n      // Aplicar filtros (ou remover se campos estiverem vazios)\n      this.applyFiltersToDataSource();\n    }\n  }\n\n  // Aplicar filtros ao dataSource quando não há paginação\n  private applyFiltersToDataSource(): void {\n    if (!this.dataSource) return;\n\n    let filteredItems = this.applyClientSideFilters([...this.items]);\n    this.dataSource.data = filteredItems;\n    // Atualizar filteredItems com os dados filtrados (será atualizado novamente no handleDownload com filteredData)\n    this.filteredItems = filteredItems;\n    this.totalItems = filteredItems.length;\n  }\n\n  // Métodos com paginação\n  private async loadItemsPaginated(\n    navigation: string = 'reload',\n    reset: boolean = false\n  ) {\n    if (reset && ['forward', 'reload'].includes(navigation)) {\n      this.lastDoc = null;\n      this.currentClientPageIndex = 0;\n    }\n    if (reset && ['backward', 'reload'].includes(navigation)) {\n      this.firstDoc = null;\n      this.currentClientPageIndex = 0;\n    }\n\n    const activeFilters = this.filtersForm.controls\n      .flatMap((control) => {\n        const group = control as FormGroup;\n        const selectedFilter = group.get('selectFilter')?.value;\n        if (!selectedFilter) return [];\n\n        const arrange = selectedFilter.arrange;\n\n        if (arrange === 'filter') {\n          const filterValue = group.get('typeFilter')?.value;\n          if (!filterValue) return [];\n          return {\n            arrange,\n            filter: {\n              property: selectedFilter.property,\n              filtering: filterValue,\n            },\n            dateFilter: undefined,\n          };\n        }\n\n        if (arrange === 'filterByDate') {\n          const initial = group.get('initialDate')?.value;\n          const final = group.get('finalDate')?.value;\n          if (initial && final) {\n            const [dayI, monthI, yearI] = initial.split('/');\n            const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\n            const [dayF, monthF, yearF] = final.split('/');\n            const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\n            finalDate.setHours(23, 59, 59);\n            return {\n              arrange,\n              filter: undefined,\n              dateFilter: {\n                initial: initialDate,\n                final: finalDate,\n              },\n            };\n          }\n          return [];\n        }\n\n        if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {\n          const selectedItems = group.get('selectItem')?.value;\n          if (Array.isArray(selectedItems) && selectedItems.length > 0) {\n            return selectedItems.map((item) => ({\n              arrange,\n              filter: {\n                property: item.property,\n                filtering: item.value,\n              },\n              dateFilter: undefined,\n            }));\n          }\n        }\n\n        return [];\n      })\n      .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));\n\n    this.arrange = {\n      filters: activeFilters,\n      sortBy: this.sortBy,\n    };\n\n    const paginated: any = {\n      batchSize: this.pageSize,\n      collection: this.data.collection,\n      doc: { lastDoc: this.lastDoc, firstDoc: this.firstDoc },\n      navigation,\n      arrange: this.arrange,\n      conditions: this.data.conditions,\n      size: this.totalItems,\n      filterFn: this.data.filterFn,\n      clientPageIndex: this.currentClientPageIndex,\n    };\n\n    const result = await this.tableService.getPaginated(paginated);\n    this.items = result.items;\n    await this.loadRelations();\n    await this.loadQueryLengths();\n    this.lastDoc = result.lastDoc as any;\n    this.firstDoc = result.firstDoc as any;\n\n    // Atualizar currentClientPageIndex se retornado pelo fallback\n    if (result.currentClientPageIndex !== undefined) {\n      this.currentClientPageIndex = result.currentClientPageIndex;\n    }\n\n    let sum = 0;\n    if (this.data.totalRef) {\n      for (const totalRef of this.data.totalRef) {\n        const totalRefDoc = await totalRef.ref.get();\n        const docData = totalRefDoc.data() as any;\n        if (docData || result.filterLength) {\n          sum =\n            result.filterLength ??\n            ((sum + (docData ? docData[totalRef.field] : 0)) as number);\n        }\n      }\n      this.totalItems = sum;\n    }\n\n    this.hasNextPage = result.hasNextPage;\n    this.dataSource = new MatTableDataSource(this.items);\n    this.filterPredicate = this.dataSource.filterPredicate;\n  }\n\n  async onPageChange(event?: PageEvent) {\n    if (this.data.pagination === true && event) {\n      this.isLoading = true;\n\n      const previousPageIndex = event.previousPageIndex ?? 0;\n      const pageIndex = event.pageIndex;\n      const currentComponentPageSize = this.pageSize;\n      const eventPageSize = event.pageSize;\n      const totalItems = this.totalItems;\n\n      let navigationDirection: string;\n      let resetDocs = false;\n      let originalPageSize: number | null = null;\n\n      const lastPageIndex = Math.max(\n        0,\n        Math.ceil(totalItems / eventPageSize) - 1\n      );\n\n      // Atualizar currentClientPageIndex sempre para o fallback\n      this.currentClientPageIndex = pageIndex;\n\n      if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {\n        this.currentPageNumber++;\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex < previousPageIndex\n      ) {\n        this.currentPageNumber = Math.max(1, this.currentPageNumber - 1);\n      }\n\n      if (eventPageSize !== currentComponentPageSize) {\n        console.log('Alterou a quantidade de elementos exibidos por página');\n        this.pageSize = eventPageSize;\n        navigationDirection = 'forward';\n        resetDocs = true;\n        this.currentClientPageIndex = 0;\n      } else if (\n        pageIndex === 0 &&\n        previousPageIndex !== undefined &&\n        pageIndex < previousPageIndex\n      ) {\n        console.log('Pulou para a primeira página');\n        navigationDirection = 'forward';\n        this.currentPageNumber = 1;\n        this.currentClientPageIndex = 0;\n        resetDocs = true;\n      } else if (\n        pageIndex === lastPageIndex &&\n        previousPageIndex !== undefined &&\n        pageIndex > previousPageIndex &&\n        pageIndex - previousPageIndex > 1\n      ) {\n        console.log('Pulou para a ultima página');\n        navigationDirection = 'backward';\n        resetDocs = true;\n\n        const itemsExpectedInLastPage =\n          totalItems - lastPageIndex * eventPageSize;\n\n        if (\n          itemsExpectedInLastPage > 0 &&\n          itemsExpectedInLastPage < eventPageSize\n        ) {\n          originalPageSize = this.pageSize;\n          this.pageSize = itemsExpectedInLastPage;\n        }\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex > previousPageIndex\n      ) {\n        console.log('Procedendo');\n        navigationDirection = 'forward';\n        resetDocs = false;\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex < previousPageIndex\n      ) {\n        console.log('Retrocedendo.');\n        navigationDirection = 'backward';\n        resetDocs = false;\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex === previousPageIndex\n      ) {\n        console.log('Recarregando.');\n        navigationDirection = 'reload';\n        resetDocs = false;\n      } else if (previousPageIndex === undefined && pageIndex === 0) {\n        console.log(\n          'Evento inicial do paginador para pág 0. ngOnInit carregou.'\n        );\n        this.isLoading = false;\n        if (event) this.pageEvent = event;\n        return;\n      } else {\n        console.warn('INESPERADO! Condição de navegação não tratada:', event);\n        this.isLoading = false;\n        if (event) this.pageEvent = event;\n        return;\n      }\n\n      if (navigationDirection) {\n        try {\n          await this.loadItemsPaginated(navigationDirection, resetDocs);\n        } catch (error) {\n          console.error('Erro ao carregar itens paginados:', error);\n        } finally {\n          if (originalPageSize !== null) {\n            this.pageSize = originalPageSize;\n          }\n        }\n      }\n\n      if (event) this.pageEvent = event;\n      this.isLoading = false;\n    }\n  }\n\n  // Outros métodos\n\n  applyFilter(value: string) {\n    // Sem paginação\n    if (this.data.pagination === false) {\n      this.dataSource.filter = String(value).trim().toLowerCase();\n    }\n    // Com paginação\n    if (this.data.pagination === true) {\n      this.filterValue = value;\n      this.filterSubject.next(this.filterValue);\n    }\n  }\n\n  goToDetails(row: any): void {\n    if (this.data.isNotClickable) {\n      return;\n    }\n    const urlPath = this.data.url || this.data.name;\n    const url = this.router.serializeUrl(\n      this.router.createUrlTree([`/${urlPath}`, row.id])\n    );\n    window.open(url, '_blank');\n  }\n\n  async getRelation(params: {\n    id: any;\n    collection: string;\n    newProperty: string;\n  }): Promise<string> {\n    try {\n      let snapshot: firestore.DocumentSnapshot<unknown> | undefined;\n      if (\n        params.id !== '' &&\n        params.id !== undefined &&\n        params.collection !== undefined &&\n        params.collection !== ''\n      ) {\n        snapshot = await firstValueFrom(\n          this.firestore.collection(params.collection).doc(params.id).get()\n        );\n      }\n      if (snapshot && snapshot.exists) {\n        const data = snapshot.data() as any;\n        return data?.[params.newProperty] ?? '';\n      }\n      return '';\n    } catch (e) {\n      console.log(e);\n      return '';\n    }\n  }\n\n  private async loadRelations() {\n    const relationPromises = this.data.displayedColumns\n      .filter((col) => col.relation)\n      .flatMap((col) =>\n        this.items.map(async (item) => {\n          if (col.relation) {\n            item[col.property] = await this.getRelation({\n              id: item[col.relation.property],\n              collection: col.relation.collection,\n              newProperty: col.relation.newProperty,\n            });\n          }\n        })\n      );\n\n    await Promise.all(relationPromises);\n  }\n\n  async getQueryLength(params: {\n    item: any;\n    relation: {\n      collection: string;\n      property: string;\n      operator: WhereFilterOp;\n      value: string;\n    };\n  }) {\n    const snapshot = await this.firestore\n      .collection(params.relation.collection)\n      .ref.where(\n        params.relation.property,\n        params.relation.operator,\n        params.item[params.relation.value]\n      )\n      .get();\n    return snapshot.size;\n  }\n\n  private async loadQueryLengths() {\n    const lengthPromises = this.data.displayedColumns\n      .filter((col) => col.queryLength)\n      .flatMap((col) =>\n        this.items.map(async (item) => {\n          if (col.queryLength) {\n            item[col.property] = await this.getQueryLength({\n              item: item,\n              relation: col.queryLength,\n            });\n          }\n        })\n      );\n    await Promise.all(lengthPromises);\n  }\n\n  private filterItems() {\n    if (this.data.conditions) {\n      this.data.conditions.forEach((cond) => {\n        this.items = this.items.filter((item) => {\n          const operatorFunction = (this.tableService.operators as any)[\n            cond.operator\n          ];\n          if (operatorFunction) {\n            return operatorFunction(\n              item[cond.firestoreProperty],\n              cond.dashProperty\n            );\n          }\n          return false;\n        });\n      });\n    }\n  }\n\n  private buildArrangeFromFilters(): Arrange {\n    const activeFilters = this.filtersForm.controls\n      .flatMap((control) => {\n        const group = control as FormGroup;\n        const selectedFilter = group.get('selectFilter')?.value;\n        if (!selectedFilter) return [];\n\n        const arrange = selectedFilter.arrange;\n\n        if (arrange === 'filter') {\n          const filterValue = group.get('typeFilter')?.value;\n          if (!filterValue) return [];\n          return {\n            arrange,\n            filter: {\n              property: selectedFilter.property,\n              filtering: filterValue,\n            },\n            dateFilter: undefined,\n          };\n        }\n\n        if (arrange === 'filterByDate') {\n          const initial = group.get('initialDate')?.value;\n          const final = group.get('finalDate')?.value;\n          if (initial && final) {\n            try {\n              const [dayI, monthI, yearI] = initial.split('/');\n              const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\n              const [dayF, monthF, yearF] = final.split('/');\n              const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\n              finalDate.setHours(23, 59, 59);\n              return {\n                arrange,\n                filter: undefined,\n                dateFilter: {\n                  initial: initialDate,\n                  final: finalDate,\n                },\n              };\n            } catch (error) {\n              return [];\n            }\n          }\n          return [];\n        }\n\n        if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {\n          const selectedItems = group.get('selectItem')?.value;\n          if (Array.isArray(selectedItems) && selectedItems.length > 0) {\n            return selectedItems.map((item) => ({\n              arrange,\n              filter: {\n                property: item.property,\n                filtering: item.value,\n              },\n              dateFilter: undefined,\n            }));\n          }\n        }\n\n        return [];\n      })\n      .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));\n\n    return {\n      filters: activeFilters,\n      sortBy: this.sortBy,\n    };\n  }\n\n  // Filtro de data\n  async search() {\n    if (this.selectSort.value) {\n      if (this.selectSort.value.arrange === 'ascending') {\n        this.sortBy = {\n          field: this.selectSort.value.property,\n          order: 'asc',\n        };\n      }\n      if (this.selectSort.value.arrange === 'descending') {\n        this.sortBy = {\n          field: this.selectSort.value.property,\n          order: 'desc',\n        };\n      }\n    }\n\n    // Sem paginação: aplicar filtros client-side\n    if (this.data.pagination === false) {\n      // Atualizar arrange com os filtros ativos\n      this.arrange = this.buildArrangeFromFilters();\n      this.applyFiltersToDataSource();\n      this.currentArrange =\n        this.filtersForm.length > 0\n          ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''\n          : '';\n    } else {\n      // Com paginação: comportamento original\n      await this.loadItemsPaginated('reload', true);\n      this.currentArrange =\n        this.filtersForm.length > 0\n          ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''\n          : '';\n      this.paginator.firstPage();\n    }\n  }\n\n  async resetFilter() {\n    this.dataSource.filter = '';\n    if (this.filterPredicate) {\n      this.dataSource.filterPredicate = this.filterPredicate;\n    }\n\n    this.filtersForm.clear();\n    this.addFilter();\n    this.selectSort.patchValue('');\n    this.sortBy = {\n      order: this.data.sortBy ? this.data.sortBy.order : 'desc',\n      field: this.data.sortBy ? this.data.sortBy.field : 'createdAt',\n    };\n\n    // Sem paginação: recarregar itens originais\n    if (this.data.pagination === false) {\n      // Resetar arrange para valores padrão\n      this.arrange = {\n        filters: [],\n        sortBy: this.sortBy,\n      };\n      this.dataSource.data = [...this.items];\n      this.totalItems = this.items.length;\n      this.currentArrange = '';\n    } else {\n      // Com paginação: comportamento original\n      await this.loadItemsPaginated('reload', true);\n      this.currentArrange = '';\n      this.paginator.firstPage();\n    }\n  }\n\n  // Método público para recarregar a tabela\n  async reloadTable() {\n    if (this.data.pagination) {\n      await this.loadItemsPaginated('reload', true);\n      this.paginator.firstPage();\n    } else {\n      await this.loadItems();\n    }\n  }\n\n  updateDisplayedColumns() {\n    if (this.dataSource) {\n      this.dataSource = new MatTableDataSource<any>([]);\n    }\n\n    this.columnProperties = this.data.displayedColumns.map((column) => {\n      return column.property;\n    });\n\n    this.dropdownItems = [];\n    this.sortableDropdownItems = [];\n\n    this.data.displayedColumns.forEach((col) => {\n      if (col.isFilterable) {\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filter',\n          title: col.title,\n        });\n      }\n      if (col.isSortable) {\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'ascending',\n          title: col.title + ': crescente',\n        });\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'descending',\n          title: col.title + ': decrescente',\n        });\n      }\n      if (col.isFilterableByDate) {\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filterByDate',\n          title: col.title + ': filtro por data',\n        });\n      }\n    });\n\n    if (\n      this.data.filterableOptions &&\n      Array.isArray(this.data.filterableOptions)\n    ) {\n      this.data.filterableOptions.forEach((option) =>\n        this.dropdownItems.push({ ...option, arrange: 'equals' })\n      );\n    }\n  }\n\n  isString(value: any): boolean {\n    return typeof value === 'string';\n  }\n\n  // Métodos para controle do tooltip\n  onCellMouseEnter(event: MouseEvent, row: any, col: any): void {\n    // Só mostrar tooltip se a coluna tiver charLimit definido\n    if (!col.charLimit) {\n      return;\n    }\n\n    const fullValue = this.getDisplayValue(col, row, true);\n\n    // Só mostrar tooltip se o valor completo for maior que o limite\n    if (fullValue.length <= col.charLimit) {\n      return;\n    }\n\n    this.hoveredCell = { row, col };\n    this.tooltipContent = fullValue;\n\n    // Definir posição do tooltip\n    this.tooltipPosition = {\n      x: event.clientX + 10,\n      y: event.clientY - 10,\n    };\n\n    // Timeout para mostrar o tooltip\n    this.tooltipTimeout = setTimeout(() => {\n      if (\n        this.hoveredCell &&\n        this.hoveredCell.row === row &&\n        this.hoveredCell.col === col\n      ) {\n        this.showTooltip = true;\n      }\n    }, 500);\n  }\n\n  onCellMouseLeave(): void {\n    if (this.tooltipTimeout) {\n      clearTimeout(this.tooltipTimeout);\n      this.tooltipTimeout = null;\n    }\n\n    this.showTooltip = false;\n    this.hoveredCell = null;\n    this.tooltipContent = '';\n  }\n\n  onCellMouseMove(event: MouseEvent): void {\n    if (this.showTooltip) {\n      this.tooltipPosition = {\n        x: event.clientX + 10,\n        y: event.clientY - 10,\n      };\n    }\n  }\n\n  // Métodos para inversão vertical dos tabs\n  getTabGroups(tabs: any[]): number[] {\n    if (!tabs || tabs.length === 0) return [];\n\n    const totalGroups = Math.ceil(tabs.length / 6);\n    const groups: number[] = [];\n\n    // Criar array de índices invertidos (último grupo primeiro)\n    for (let i = totalGroups - 1; i >= 0; i--) {\n      groups.push(i);\n    }\n\n    return groups;\n  }\n\n  getTabGroup(tabs: any[], groupIndex: number): any[] {\n    if (!tabs || tabs.length === 0) return [];\n\n    const startIndex = groupIndex * 6;\n    const endIndex = Math.min(startIndex + 6, tabs.length);\n\n    return tabs.slice(startIndex, endIndex);\n  }\n\n  getRealTabIndex(groupIndex: number, tabIndexInGroup: number): number {\n    if (!this.data.tabs?.tabsData) return 0;\n    const totalGroups = Math.ceil(this.data.tabs.tabsData.length / 6);\n    const realGroupIndex = totalGroups - 1 - groupIndex;\n    return realGroupIndex * 6 + tabIndexInGroup;\n  }\n\n  onTableSelected(i: number, j: number) {\n    if (!this.data.tabs?.tabsData || !this.data.tabs.method) return;\n    this.selectedTab = this.getRealTabIndex(i, j);\n    const tab = this.data.tabs.tabsData[this.selectedTab];\n    if (tab) {\n      this.data.tabs.method(tab, this.selectedTab);\n    }\n  }\n\n  isTabSelected(originalIndex: number): boolean {\n    return this.selectedTab === originalIndex;\n  }\n\n  shouldShowActionButton(): boolean {\n    if (!this.data?.actionButton) {\n      return false;\n    }\n\n    if (!this.data.actionButton.condition) {\n      return true;\n    }\n\n    try {\n      return this.data.actionButton.condition(null) ?? true;\n    } catch {\n      return true;\n    }\n  }\n\n  // Método para lidar com download (diferente para paginado e não paginado)\n  handleDownload(): void {\n    if (!this.downloadTable) return;\n\n    // Se não há paginação, usar dados filtrados do dataSource\n    if (this.data.pagination === false) {\n      // Atualizar filteredItems com os dados filtrados do dataSource\n      // (que já inclui filtro de texto aplicado via filterPredicate)\n      if (this.dataSource && this.dataSource.filteredData) {\n        this.filteredItems = [...this.dataSource.filteredData];\n      }\n\n      // Construir arrange com os filtros ativos (se houver)\n      const arrange = this.buildArrangeFromFilters();\n      this.downloadTable(arrange, this.data.conditions || []);\n    } else {\n      // Modo paginado: usar arrange existente (comportamento original)\n      if (this.arrange) {\n        this.downloadTable(this.arrange, this.data.conditions || []);\n      }\n    }\n  }\n}\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, Custom Action, and Global Actions -->\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\r\n        <!-- Right Side: Search, Reset, Export -->\r\n        <div\r\n          class=\"flex flex-wrap gap-3\"\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\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 rounded-lg border border-gray-200 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 *ngFor=\"let item of dropdownItems\" [value]=\"item\">\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()\"\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                  matInput\r\n                  (keyup.enter)=\"search()\"\r\n                  formControlName=\"initialDate\"\r\n                  [dropSpecialCharacters]=\"false\"\r\n                  mask=\"d0/M0/0000\"\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                  (keyup.enter)=\"search()\"\r\n                  matInput\r\n                  formControlName=\"finalDate\"\r\n                  [dropSpecialCharacters]=\"false\"\r\n                  mask=\"d0/M0/0000\"\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 & Sort -->\r\n      <div\r\n        class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\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        <!-- Sort Dropdown -->\r\n        <div\r\n          class=\"w-full sm:w-auto sm:min-w-[250px]\"\r\n          *ngIf=\"sortableDropdownItems.length > 0\"\r\n        >\r\n          <mat-form-field appearance=\"outline\" class=\"w-full\">\r\n            <mat-label>Ordenar por</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    <!-- SIMPLE SEARCH (for non-paginated tables) -->\r\n    <div\r\n      class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n      *ngIf=\"data.pagination === false && hasFilterableColumn === true\"\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()\"\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      <button\r\n        *ngIf=\"data.actionButton\"\r\n        [ngClass]=\"\r\n          (data.actionButton.colorClass || 'bg-blue-500') +\r\n          ' float-right 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 *ngIf=\"data.actionButton.icon\" [class]=\"data.actionButton.icon\"></i>\r\n        {{ data.actionButton.label }}\r\n      </button>\r\n    </div>\r\n\r\n    <!-- FILTERS PANEL (for non-paginated tables) -->\r\n    <div\r\n      class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\r\n      *ngIf=\"data.pagination === false && dropdownItems.length > 0\"\r\n    >\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 rounded-lg border border-gray-200 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 *ngFor=\"let item of dropdownItems\" [value]=\"item\">\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()\"\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                  matInput\r\n                  (keyup.enter)=\"search()\"\r\n                  (blur)=\"onDateFilterChange()\"\r\n                  formControlName=\"initialDate\"\r\n                  [dropSpecialCharacters]=\"false\"\r\n                  mask=\"d0/M0/0000\"\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                  (keyup.enter)=\"search()\"\r\n                  (blur)=\"onDateFilterChange()\"\r\n                  matInput\r\n                  formControlName=\"finalDate\"\r\n                  [dropSpecialCharacters]=\"false\"\r\n                  mask=\"d0/M0/0000\"\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 flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\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 class=\"flex flex-wrap gap-3\">\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            Aplicar\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\r\n    <div class=\"flex flex-col\">\r\n      <div\r\n        class=\"mx-auto flex flex-col\"\r\n        *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\r\n      >\r\n        <!-- Calcular quantos grupos de 6 tabs existem -->\r\n        <ng-container\r\n          *ngFor=\"\r\n            let groupIndex of getTabGroups(data.tabs.tabsData);\r\n            let i = index\r\n          \"\r\n        >\r\n          <div class=\"mx-auto flex flex-row\">\r\n            <ng-container\r\n              *ngFor=\"\r\n                let tab of getTabGroup(data.tabs.tabsData, groupIndex);\r\n                let j = index\r\n              \"\r\n            >\r\n              <button\r\n                class=\"border-2 border-gray-300 bg-gray-200 px-4 py-2 font-medium transition hover:brightness-95\"\r\n                [ngClass]=\"\r\n                  isTabSelected(getRealTabIndex(i, j))\r\n                    ? 'border-b-0 brightness-110'\r\n                    : ''\r\n                \"\r\n                (click)=\"onTableSelected(i, j)\"\r\n              >\r\n                {{ tab.label }}\r\n                <span\r\n                  *ngIf=\"tab.counter !== undefined\"\r\n                  class=\"ml-2 text-xs font-bold\"\r\n                  [ngClass]=\"tab.counterClass\"\r\n                >\r\n                  {{ tab.counter }}\r\n                </span>\r\n              </button>\r\n            </ng-container>\r\n          </div>\r\n        </ng-container>\r\n      </div>\r\n      <div class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl\">\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\"\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                [ngClass]=\"\r\n                  (data.color?.bg ? ' ' + $any(data.color).bg : '') +\r\n                  (data.color?.text ? ' ' + $any(data.color).text : '')\r\n                \"\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                [ngClass]=\"\r\n                  (data.color?.bg ? ' ' + $any(data.color).bg : '') +\r\n                  (data.color?.text ? ' ' + $any(data.color).text : '')\r\n                \"\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                <img\r\n                  *ngIf=\"\r\n                    col.image && col.image.path && !col.iconClass && !col.method\r\n                  \"\r\n                  [src]=\"col.image.path + '/' + row[col.property]\"\r\n                  [ngClass]=\"col.image.class\"\r\n                  alt=\"Imagem\"\r\n                />\r\n                <img\r\n                  *ngIf=\"\r\n                    col.image && col.image.url && !col.iconClass && !col.method\r\n                  \"\r\n                  [src]=\"row[col.property]\"\r\n                  [ngClass]=\"col.image.class\"\r\n                  alt=\"Imagem\"\r\n                />\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\" 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\">\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) || this.data.filterFn\r\n            }\"\r\n          >\r\n          </mat-paginator>\r\n          <div\r\n            *ngIf=\"\r\n              !isLoading &&\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  <div\r\n    *ngIf=\"showTooltip\"\r\n    class=\"fixed z-50 max-w-md break-words rounded-lg bg-gray-800 px-3 py-2 text-sm text-white shadow-lg\"\r\n    [style.left.px]=\"tooltipPosition.x\"\r\n    [style.top.px]=\"tooltipPosition.y\"\r\n    [style.pointer-events]=\"'none'\"\r\n  >\r\n    {{ tooltipContent }}\r\n  </div>\r\n</div>\r\n"]}
1
+ import { Component, Input, ViewChild } from '@angular/core';
2
+ import { MatPaginator } from '@angular/material/paginator';
3
+ import { MatSort } from '@angular/material/sort';
4
+ import { MatTableDataSource } from '@angular/material/table';
5
+ import { debounceTime, firstValueFrom, Subject } from 'rxjs';
6
+ import { FormControl, FormArray, FormGroup } from '@angular/forms';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/router";
9
+ import * as i2 from "../../services/table.service";
10
+ import * as i3 from "@angular/fire/compat/firestore";
11
+ import * as i4 from "@angular/common";
12
+ import * as i5 from "@angular/forms";
13
+ import * as i6 from "@angular/material/table";
14
+ import * as i7 from "@angular/material/paginator";
15
+ import * as i8 from "@angular/material/sort";
16
+ import * as i9 from "@angular/material/form-field";
17
+ import * as i10 from "@angular/material/input";
18
+ import * as i11 from "@angular/material/select";
19
+ import * as i12 from "@angular/material/core";
20
+ import * as i13 from "@angular/material/tooltip";
21
+ import * as i14 from "@angular/material/progress-spinner";
22
+ import * as i15 from "@angular/material/icon";
23
+ import * as i16 from "ngx-mask";
24
+ export class TableComponent {
25
+ // CONSTRUCTOR
26
+ constructor(router, tableService, firestore) {
27
+ this.router = router;
28
+ this.tableService = tableService;
29
+ this.firestore = firestore;
30
+ this.arrange = null;
31
+ this.currentPageNumber = 1;
32
+ this.currentClientPageIndex = 0;
33
+ this.items = [];
34
+ this.filteredItems = []; // Dados filtrados para modo não paginado (público para acesso externo)
35
+ this.isLoading = false;
36
+ this.lastDoc = null;
37
+ this.firstDoc = null;
38
+ this.sortBy = {
39
+ field: 'createdAt',
40
+ order: 'desc',
41
+ };
42
+ this.columnProperties = [];
43
+ this.selectSort = new FormControl('');
44
+ this.currentArrange = '';
45
+ this.hasNextPage = false;
46
+ this.dropdownItems = [];
47
+ this.sortableDropdownItems = [];
48
+ this.pageSize = 25;
49
+ this.totalItems = 0;
50
+ this.filterValue = null;
51
+ this.hasFilterableColumn = false;
52
+ this.hasSortableColumn = false;
53
+ this.filterSubject = new Subject();
54
+ this.debounceTimeMs = 500;
55
+ this.selectedTab = 0;
56
+ // Propriedades para controle do tooltip
57
+ this.hoveredCell = null;
58
+ this.showTooltip = false;
59
+ this.tooltipContent = '';
60
+ this.tooltipPosition = { x: 0, y: 0 };
61
+ this.filtersForm = new FormArray([this.createFilterGroup()]);
62
+ }
63
+ createFilterGroup() {
64
+ return new FormGroup({
65
+ selectFilter: new FormControl(''),
66
+ typeFilter: new FormControl(''),
67
+ selectItem: new FormControl(''),
68
+ initialDate: new FormControl('', [
69
+ this.tableService.dateFormatValidator(),
70
+ ]),
71
+ finalDate: new FormControl('', [this.tableService.dateFormatValidator()]),
72
+ });
73
+ }
74
+ addFilter(filterData) {
75
+ if (this.data.pagination === true &&
76
+ this.hasActiveDateFilter() &&
77
+ filterData?.selectFilter?.arrange === 'filterByDate') {
78
+ // Não permitir adicionar novo filtro de data quando já existe um ativo
79
+ return;
80
+ }
81
+ const newFilterGroup = this.createFilterGroup();
82
+ if (filterData) {
83
+ if (filterData.selectFilter) {
84
+ newFilterGroup.get('selectFilter')?.setValue(filterData.selectFilter);
85
+ }
86
+ if (filterData.typeFilter) {
87
+ newFilterGroup.get('typeFilter')?.setValue(filterData.typeFilter);
88
+ }
89
+ if (filterData.selectItem) {
90
+ newFilterGroup.get('selectItem')?.setValue(filterData.selectItem);
91
+ }
92
+ if (filterData.initialDate) {
93
+ newFilterGroup.get('initialDate')?.setValue(filterData.initialDate);
94
+ }
95
+ if (filterData.finalDate) {
96
+ newFilterGroup.get('finalDate')?.setValue(filterData.finalDate);
97
+ }
98
+ }
99
+ this.filtersForm.push(newFilterGroup);
100
+ }
101
+ hasActiveDateFilter() {
102
+ return this.filtersForm.controls.some((control) => {
103
+ const group = control;
104
+ const selectedFilter = group.get('selectFilter')?.value;
105
+ return selectedFilter?.arrange === 'filterByDate';
106
+ });
107
+ }
108
+ getAvailableFilterOptions() {
109
+ if (this.data.pagination === true && this.hasActiveDateFilter()) {
110
+ return this.dropdownItems.filter((item) => item.arrange !== 'filterByDate');
111
+ }
112
+ return this.dropdownItems;
113
+ }
114
+ onSelectFilterChange() {
115
+ const lastIndex = this.filtersForm.length - 1;
116
+ const lastFilter = this.filtersForm.at(lastIndex);
117
+ const selectedFilter = lastFilter.get('selectFilter')?.value;
118
+ if (selectedFilter) {
119
+ if (selectedFilter.arrange === 'filterByDate' &&
120
+ this.data.pagination === true) {
121
+ return;
122
+ }
123
+ this.addFilter();
124
+ }
125
+ }
126
+ removeFilter(index) {
127
+ this.filtersForm.removeAt(index);
128
+ if (this.filtersForm.length === 0) {
129
+ this.addFilter();
130
+ }
131
+ }
132
+ removeAllFilters() {
133
+ this.filtersForm.clear();
134
+ this.addFilter();
135
+ this.resetFilter();
136
+ }
137
+ // METHODS
138
+ async ngOnInit() {
139
+ if (!this.data.color)
140
+ this.data.color = { bg: 'bg-primary', text: 'text-black' };
141
+ this.columnProperties = this.data.displayedColumns.map((column) => {
142
+ return column.property;
143
+ });
144
+ if (this.data.actionButton && !this.data.actionButton.condition) {
145
+ this.data.actionButton.condition = (_row) => true;
146
+ }
147
+ this.data.displayedColumns.forEach((col) => {
148
+ if (col.isFilterable) {
149
+ if (this.hasFilterableColumn === false)
150
+ this.hasFilterableColumn = true;
151
+ this.dropdownItems.push({
152
+ ...col,
153
+ arrange: 'filter',
154
+ title: col.title,
155
+ });
156
+ }
157
+ if (col.isSortable) {
158
+ if (this.hasSortableColumn === false)
159
+ this.hasSortableColumn = true;
160
+ this.sortableDropdownItems.push({
161
+ ...col,
162
+ arrange: 'ascending',
163
+ title: col.title + ': crescente',
164
+ });
165
+ this.sortableDropdownItems.push({
166
+ ...col,
167
+ arrange: 'descending',
168
+ title: col.title + ': decrescente',
169
+ });
170
+ }
171
+ if (col.isFilterableByDate) {
172
+ this.dropdownItems.push({
173
+ ...col,
174
+ arrange: 'filterByDate',
175
+ title: col.title + ': filtro por data',
176
+ });
177
+ }
178
+ });
179
+ if (this.data.filterableOptions &&
180
+ Array.isArray(this.data.filterableOptions) &&
181
+ this.data.filterableOptions.length > 0) {
182
+ this.data.filterableOptions.forEach((option) => this.dropdownItems.push({ ...option, arrange: 'equals' }));
183
+ }
184
+ // Sem paginação
185
+ if (this.data.pagination === false) {
186
+ await this.loadItems();
187
+ }
188
+ // Com paginação
189
+ if (this.data.pagination === true) {
190
+ if (this.data.sortBy)
191
+ this.sortBy = {
192
+ field: this.data.sortBy.field,
193
+ order: this.data.sortBy.order,
194
+ };
195
+ this.filterSubject
196
+ .pipe(debounceTime(this.debounceTimeMs))
197
+ .subscribe(() => {
198
+ this.loadItemsPaginated('reload', true);
199
+ });
200
+ this.isLoading = true;
201
+ await this.loadItemsPaginated('reload', true);
202
+ this.sort.active = 'createdAt';
203
+ this.sort.direction = 'desc';
204
+ this.dataSource.paginator = this.paginator;
205
+ this.dataSource.sort = this.sort;
206
+ this.totalItems = 0;
207
+ if (this.data.totalRef) {
208
+ for (const totalRef of this.data.totalRef) {
209
+ const totalRefDoc = await totalRef.ref.get();
210
+ const docData = totalRefDoc.data();
211
+ if (docData && docData[totalRef.field])
212
+ this.totalItems = (this.totalItems +
213
+ docData[totalRef.field]);
214
+ }
215
+ }
216
+ this.isLoading = false;
217
+ }
218
+ }
219
+ getDisplayValue(col, row, withinLimit = false) {
220
+ let value;
221
+ if (col.calculateValue) {
222
+ value = col.calculateValue(row);
223
+ }
224
+ else {
225
+ value = this.getNestedValue(row, col.property);
226
+ }
227
+ if (Array.isArray(value) && col.arrayField) {
228
+ value = this.formatArrayValue(value, col.arrayField);
229
+ }
230
+ if (col.queryLength && row[col.property]) {
231
+ value = row[col.property];
232
+ }
233
+ value = col.pipe ? col.pipe.transform(value) : value;
234
+ if (value === null || value === undefined) {
235
+ value = '';
236
+ }
237
+ else {
238
+ value = String(value);
239
+ }
240
+ if (typeof value !== 'string') {
241
+ value = '';
242
+ }
243
+ if (withinLimit || !col.charLimit) {
244
+ return value;
245
+ }
246
+ return value.substring(0, col.charLimit) + '...';
247
+ }
248
+ getNestedValue(obj, path) {
249
+ if (!path)
250
+ return undefined;
251
+ const properties = path.split('.');
252
+ return properties.reduce((acc, currentPart) => acc && acc[currentPart], obj);
253
+ }
254
+ formatArrayValue(array, field) {
255
+ if (!Array.isArray(array) || array.length === 0) {
256
+ return '';
257
+ }
258
+ const values = array
259
+ .map((item) => {
260
+ if (typeof item === 'object' && item !== null) {
261
+ return item[field] || '';
262
+ }
263
+ return String(item);
264
+ })
265
+ .filter((value) => value !== '' && value !== null && value !== undefined);
266
+ return values.join(', ');
267
+ }
268
+ // Métodos sem paginação
269
+ async loadItems() {
270
+ this.items = await this.tableService.getItems(this.data.collectionRef);
271
+ if (this.data.conditions) {
272
+ this.filterItems();
273
+ }
274
+ await this.loadRelations();
275
+ await this.loadQueryLengths();
276
+ this.totalItems = this.items.length;
277
+ // Inicializar arrange para tabelas não paginadas
278
+ this.arrange = {
279
+ filters: [],
280
+ sortBy: this.data.sortBy || {
281
+ field: 'createdAt',
282
+ order: 'desc',
283
+ },
284
+ };
285
+ this.sortBy = this.data.sortBy || {
286
+ field: 'createdAt',
287
+ order: 'desc',
288
+ };
289
+ // Aplicar filtros client-side se existirem
290
+ let itemsToDisplay = [...this.items];
291
+ itemsToDisplay = this.applyClientSideFilters(itemsToDisplay);
292
+ this.filteredItems = itemsToDisplay; // Armazenar dados filtrados
293
+ this.dataSource = new MatTableDataSource(itemsToDisplay);
294
+ this.dataSource.paginator = this.paginator;
295
+ this.dataSource.sort = this.sort;
296
+ if (this.data.sortBy) {
297
+ this.dataSource.sort.active = this.data.sortBy.field;
298
+ this.dataSource.sort.direction = this.data.sortBy.order;
299
+ this.dataSource.sort.sortChange.emit();
300
+ }
301
+ this.dataSource.filterPredicate = (data, filter) => {
302
+ return this.data.displayedColumns.some((col) => {
303
+ if (col.filterPredicates) {
304
+ return col.filterPredicates.some((predicate) => {
305
+ const propertyValue = data[col.property];
306
+ if (!propertyValue || typeof propertyValue !== 'object') {
307
+ return false;
308
+ }
309
+ const predicateValue = propertyValue[predicate];
310
+ if (predicateValue === null || predicateValue === undefined) {
311
+ return false;
312
+ }
313
+ return String(predicateValue)
314
+ .trim()
315
+ .toLocaleLowerCase()
316
+ .includes(filter);
317
+ });
318
+ }
319
+ if (col.property && col.isFilterable) {
320
+ const propertyValue = data[col.property];
321
+ if (propertyValue === null || propertyValue === undefined) {
322
+ return false;
323
+ }
324
+ return String(propertyValue)
325
+ .trim()
326
+ .toLocaleLowerCase()
327
+ .includes(filter);
328
+ }
329
+ return false;
330
+ });
331
+ };
332
+ this.filterPredicate = this.dataSource.filterPredicate;
333
+ }
334
+ // Aplicar filtros client-side (filtros por data)
335
+ applyClientSideFilters(items) {
336
+ let filteredItems = [...items];
337
+ // Agrupar filtros de data por propriedade
338
+ const dateFiltersByProperty = {};
339
+ const otherFilters = [];
340
+ this.filtersForm.controls.forEach((control) => {
341
+ const group = control;
342
+ const selectedFilter = group.get('selectFilter')?.value;
343
+ if (!selectedFilter)
344
+ return;
345
+ const arrange = selectedFilter.arrange;
346
+ if (arrange === 'filterByDate') {
347
+ const initial = group.get('initialDate')?.value;
348
+ const final = group.get('finalDate')?.value;
349
+ // Só aplicar filtro se ambas as datas estiverem preenchidas
350
+ if (initial && final && initial.trim() && final.trim()) {
351
+ try {
352
+ // Validar formato da data (DD/MM/AAAA)
353
+ const datePattern = /^(\d{1,2})\/(\d{1,2})\/(\d{4})$/;
354
+ if (!datePattern.test(initial) || !datePattern.test(final)) {
355
+ return; // Ignorar se formato inválido
356
+ }
357
+ const [dayI, monthI, yearI] = initial.split('/');
358
+ const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
359
+ const [dayF, monthF, yearF] = final.split('/');
360
+ const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
361
+ finalDate.setHours(23, 59, 59);
362
+ // Validar se as datas são válidas
363
+ if (isNaN(initialDate.getTime()) || isNaN(finalDate.getTime())) {
364
+ return; // Ignorar se datas inválidas
365
+ }
366
+ // Usar o campo da coluna ou o sortBy.field como padrão
367
+ const dateField = selectedFilter.property || this.sortBy.field;
368
+ if (!dateFiltersByProperty[dateField]) {
369
+ dateFiltersByProperty[dateField] = [];
370
+ }
371
+ dateFiltersByProperty[dateField].push({
372
+ initial: initialDate,
373
+ final: finalDate,
374
+ });
375
+ }
376
+ catch (error) {
377
+ console.warn('Erro ao processar datas do filtro:', error);
378
+ }
379
+ }
380
+ }
381
+ else {
382
+ otherFilters.push({ group, selectedFilter, arrange });
383
+ }
384
+ });
385
+ Object.keys(dateFiltersByProperty).forEach((dateField) => {
386
+ const intervals = dateFiltersByProperty[dateField];
387
+ filteredItems = filteredItems.filter((item) => {
388
+ try {
389
+ const fieldValue = item[dateField];
390
+ if (!fieldValue) {
391
+ return false;
392
+ }
393
+ let itemDate;
394
+ if (typeof fieldValue.toDate === 'function') {
395
+ itemDate = fieldValue.toDate();
396
+ }
397
+ else if (fieldValue instanceof Date) {
398
+ itemDate = fieldValue;
399
+ }
400
+ else if (typeof fieldValue === 'string') {
401
+ itemDate = new Date(fieldValue);
402
+ if (isNaN(itemDate.getTime())) {
403
+ return false;
404
+ }
405
+ }
406
+ else if (typeof fieldValue === 'number') {
407
+ itemDate = new Date(fieldValue);
408
+ }
409
+ else {
410
+ return false;
411
+ }
412
+ return intervals.some((interval) => itemDate >= interval.initial && itemDate <= interval.final);
413
+ }
414
+ catch (error) {
415
+ console.warn('Erro ao processar filtro de data para o item:', item.id, error);
416
+ return false;
417
+ }
418
+ });
419
+ });
420
+ otherFilters.forEach(({ group, selectedFilter, arrange }) => {
421
+ if (arrange === 'filter') {
422
+ const filterValue = group.get('typeFilter')?.value;
423
+ if (filterValue && filterValue.trim()) {
424
+ const property = selectedFilter.property;
425
+ filteredItems = filteredItems.filter((item) => {
426
+ const value = String(item[property] || '')
427
+ .trim()
428
+ .toLocaleLowerCase();
429
+ return value.includes(filterValue.trim().toLocaleLowerCase());
430
+ });
431
+ }
432
+ }
433
+ else if (selectedFilter.hasOwnProperty('items') &&
434
+ arrange === 'equals') {
435
+ const selectedItems = group.get('selectItem')?.value;
436
+ if (Array.isArray(selectedItems) && selectedItems.length > 0) {
437
+ const values = selectedItems.map((item) => item.value);
438
+ filteredItems = filteredItems.filter((item) => {
439
+ return selectedItems.some((selectedItem) => {
440
+ const itemValue = item[selectedItem.property];
441
+ return itemValue === selectedItem.value;
442
+ });
443
+ });
444
+ }
445
+ }
446
+ });
447
+ return filteredItems;
448
+ }
449
+ // Handler para mudanças de data no filtro
450
+ onDateFilterChange() {
451
+ if (this.data.pagination === false) {
452
+ // Atualizar arrange com os filtros ativos
453
+ this.arrange = this.buildArrangeFromFilters();
454
+ // Aplicar filtros (ou remover se campos estiverem vazios)
455
+ this.applyFiltersToDataSource();
456
+ }
457
+ }
458
+ // Aplicar filtros ao dataSource quando não há paginação
459
+ applyFiltersToDataSource() {
460
+ if (!this.dataSource)
461
+ return;
462
+ let filteredItems = this.applyClientSideFilters([...this.items]);
463
+ this.dataSource.data = filteredItems;
464
+ // Atualizar filteredItems com os dados filtrados (será atualizado novamente no handleDownload com filteredData)
465
+ this.filteredItems = filteredItems;
466
+ this.totalItems = filteredItems.length;
467
+ }
468
+ // Métodos com paginação
469
+ async loadItemsPaginated(navigation = 'reload', reset = false) {
470
+ if (reset && ['forward', 'reload'].includes(navigation)) {
471
+ this.lastDoc = null;
472
+ this.currentClientPageIndex = 0;
473
+ }
474
+ if (reset && ['backward', 'reload'].includes(navigation)) {
475
+ this.firstDoc = null;
476
+ this.currentClientPageIndex = 0;
477
+ }
478
+ const activeFilters = this.filtersForm.controls
479
+ .flatMap((control) => {
480
+ const group = control;
481
+ const selectedFilter = group.get('selectFilter')?.value;
482
+ if (!selectedFilter)
483
+ return [];
484
+ const arrange = selectedFilter.arrange;
485
+ if (arrange === 'filter') {
486
+ const filterValue = group.get('typeFilter')?.value;
487
+ if (!filterValue)
488
+ return [];
489
+ return {
490
+ arrange,
491
+ filter: {
492
+ property: selectedFilter.property,
493
+ filtering: filterValue,
494
+ },
495
+ dateFilter: undefined,
496
+ };
497
+ }
498
+ if (arrange === 'filterByDate') {
499
+ const initial = group.get('initialDate')?.value;
500
+ const final = group.get('finalDate')?.value;
501
+ if (initial && final) {
502
+ const [dayI, monthI, yearI] = initial.split('/');
503
+ const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
504
+ const [dayF, monthF, yearF] = final.split('/');
505
+ const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
506
+ finalDate.setHours(23, 59, 59);
507
+ return {
508
+ arrange,
509
+ filter: undefined,
510
+ dateFilter: {
511
+ initial: initialDate,
512
+ final: finalDate,
513
+ },
514
+ };
515
+ }
516
+ return [];
517
+ }
518
+ if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {
519
+ const selectedItems = group.get('selectItem')?.value;
520
+ if (Array.isArray(selectedItems) && selectedItems.length > 0) {
521
+ return selectedItems.map((item) => ({
522
+ arrange,
523
+ filter: {
524
+ property: item.property,
525
+ filtering: item.value,
526
+ },
527
+ dateFilter: undefined,
528
+ }));
529
+ }
530
+ }
531
+ return [];
532
+ })
533
+ .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));
534
+ this.arrange = {
535
+ filters: activeFilters,
536
+ sortBy: this.sortBy,
537
+ };
538
+ const paginated = {
539
+ batchSize: this.pageSize,
540
+ collection: this.data.collection,
541
+ doc: { lastDoc: this.lastDoc, firstDoc: this.firstDoc },
542
+ navigation,
543
+ arrange: this.arrange,
544
+ conditions: this.data.conditions,
545
+ size: this.totalItems,
546
+ filterFn: this.data.filterFn,
547
+ clientPageIndex: this.currentClientPageIndex,
548
+ };
549
+ const result = await this.tableService.getPaginated(paginated);
550
+ this.items = result.items;
551
+ await this.loadRelations();
552
+ await this.loadQueryLengths();
553
+ this.lastDoc = result.lastDoc;
554
+ this.firstDoc = result.firstDoc;
555
+ // Atualizar currentClientPageIndex se retornado pelo fallback
556
+ if (result.currentClientPageIndex !== undefined) {
557
+ this.currentClientPageIndex = result.currentClientPageIndex;
558
+ }
559
+ let sum = 0;
560
+ if (this.data.totalRef) {
561
+ for (const totalRef of this.data.totalRef) {
562
+ const totalRefDoc = await totalRef.ref.get();
563
+ const docData = totalRefDoc.data();
564
+ if (docData || result.filterLength) {
565
+ sum =
566
+ result.filterLength ??
567
+ (sum + (docData ? docData[totalRef.field] : 0));
568
+ }
569
+ }
570
+ this.totalItems = sum;
571
+ }
572
+ this.hasNextPage = result.hasNextPage;
573
+ this.dataSource = new MatTableDataSource(this.items);
574
+ this.filterPredicate = this.dataSource.filterPredicate;
575
+ }
576
+ async onPageChange(event) {
577
+ if (this.data.pagination === true && event) {
578
+ this.isLoading = true;
579
+ const previousPageIndex = event.previousPageIndex ?? 0;
580
+ const pageIndex = event.pageIndex;
581
+ const currentComponentPageSize = this.pageSize;
582
+ const eventPageSize = event.pageSize;
583
+ const totalItems = this.totalItems;
584
+ let navigationDirection;
585
+ let resetDocs = false;
586
+ let originalPageSize = null;
587
+ const lastPageIndex = Math.max(0, Math.ceil(totalItems / eventPageSize) - 1);
588
+ // Atualizar currentClientPageIndex sempre para o fallback
589
+ this.currentClientPageIndex = pageIndex;
590
+ if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {
591
+ this.currentPageNumber++;
592
+ }
593
+ else if (previousPageIndex !== undefined &&
594
+ pageIndex < previousPageIndex) {
595
+ this.currentPageNumber = Math.max(1, this.currentPageNumber - 1);
596
+ }
597
+ if (eventPageSize !== currentComponentPageSize) {
598
+ console.log('Alterou a quantidade de elementos exibidos por página');
599
+ this.pageSize = eventPageSize;
600
+ navigationDirection = 'forward';
601
+ resetDocs = true;
602
+ this.currentClientPageIndex = 0;
603
+ }
604
+ else if (pageIndex === 0 &&
605
+ previousPageIndex !== undefined &&
606
+ pageIndex < previousPageIndex) {
607
+ console.log('Pulou para a primeira página');
608
+ navigationDirection = 'forward';
609
+ this.currentPageNumber = 1;
610
+ this.currentClientPageIndex = 0;
611
+ resetDocs = true;
612
+ }
613
+ else if (pageIndex === lastPageIndex &&
614
+ previousPageIndex !== undefined &&
615
+ pageIndex > previousPageIndex &&
616
+ pageIndex - previousPageIndex > 1) {
617
+ console.log('Pulou para a ultima página');
618
+ navigationDirection = 'backward';
619
+ resetDocs = true;
620
+ const itemsExpectedInLastPage = totalItems - lastPageIndex * eventPageSize;
621
+ if (itemsExpectedInLastPage > 0 &&
622
+ itemsExpectedInLastPage < eventPageSize) {
623
+ originalPageSize = this.pageSize;
624
+ this.pageSize = itemsExpectedInLastPage;
625
+ }
626
+ }
627
+ else if (previousPageIndex !== undefined &&
628
+ pageIndex > previousPageIndex) {
629
+ console.log('Procedendo');
630
+ navigationDirection = 'forward';
631
+ resetDocs = false;
632
+ }
633
+ else if (previousPageIndex !== undefined &&
634
+ pageIndex < previousPageIndex) {
635
+ console.log('Retrocedendo.');
636
+ navigationDirection = 'backward';
637
+ resetDocs = false;
638
+ }
639
+ else if (previousPageIndex !== undefined &&
640
+ pageIndex === previousPageIndex) {
641
+ console.log('Recarregando.');
642
+ navigationDirection = 'reload';
643
+ resetDocs = false;
644
+ }
645
+ else if (previousPageIndex === undefined && pageIndex === 0) {
646
+ console.log('Evento inicial do paginador para pág 0. ngOnInit carregou.');
647
+ this.isLoading = false;
648
+ if (event)
649
+ this.pageEvent = event;
650
+ return;
651
+ }
652
+ else {
653
+ console.warn('INESPERADO! Condição de navegação não tratada:', event);
654
+ this.isLoading = false;
655
+ if (event)
656
+ this.pageEvent = event;
657
+ return;
658
+ }
659
+ if (navigationDirection) {
660
+ try {
661
+ await this.loadItemsPaginated(navigationDirection, resetDocs);
662
+ }
663
+ catch (error) {
664
+ console.error('Erro ao carregar itens paginados:', error);
665
+ }
666
+ finally {
667
+ if (originalPageSize !== null) {
668
+ this.pageSize = originalPageSize;
669
+ }
670
+ }
671
+ }
672
+ if (event)
673
+ this.pageEvent = event;
674
+ this.isLoading = false;
675
+ }
676
+ }
677
+ // Outros métodos
678
+ applyFilter(value) {
679
+ // Sem paginação
680
+ if (this.data.pagination === false) {
681
+ this.dataSource.filter = String(value).trim().toLowerCase();
682
+ }
683
+ // Com paginação
684
+ if (this.data.pagination === true) {
685
+ this.filterValue = value;
686
+ this.filterSubject.next(this.filterValue);
687
+ }
688
+ }
689
+ goToDetails(row) {
690
+ if (this.data.isNotClickable) {
691
+ return;
692
+ }
693
+ const urlPath = this.data.url || this.data.name;
694
+ const url = this.router.serializeUrl(this.router.createUrlTree([`/${urlPath}`, row.id]));
695
+ window.open(url, '_blank');
696
+ }
697
+ async getRelation(params) {
698
+ try {
699
+ let snapshot;
700
+ if (params.id !== '' &&
701
+ params.id !== undefined &&
702
+ params.collection !== undefined &&
703
+ params.collection !== '') {
704
+ snapshot = await firstValueFrom(this.firestore.collection(params.collection).doc(params.id).get());
705
+ }
706
+ if (snapshot && snapshot.exists) {
707
+ const data = snapshot.data();
708
+ return data?.[params.newProperty] ?? '';
709
+ }
710
+ return '';
711
+ }
712
+ catch (e) {
713
+ console.log(e);
714
+ return '';
715
+ }
716
+ }
717
+ async loadRelations() {
718
+ const relationPromises = this.data.displayedColumns
719
+ .filter((col) => col.relation)
720
+ .flatMap((col) => this.items.map(async (item) => {
721
+ if (col.relation) {
722
+ item[col.property] = await this.getRelation({
723
+ id: item[col.relation.property],
724
+ collection: col.relation.collection,
725
+ newProperty: col.relation.newProperty,
726
+ });
727
+ }
728
+ }));
729
+ await Promise.all(relationPromises);
730
+ }
731
+ async getQueryLength(params) {
732
+ const snapshot = await this.firestore
733
+ .collection(params.relation.collection)
734
+ .ref.where(params.relation.property, params.relation.operator, params.item[params.relation.value])
735
+ .get();
736
+ return snapshot.size;
737
+ }
738
+ async loadQueryLengths() {
739
+ const lengthPromises = this.data.displayedColumns
740
+ .filter((col) => col.queryLength)
741
+ .flatMap((col) => this.items.map(async (item) => {
742
+ if (col.queryLength) {
743
+ item[col.property] = await this.getQueryLength({
744
+ item: item,
745
+ relation: col.queryLength,
746
+ });
747
+ }
748
+ }));
749
+ await Promise.all(lengthPromises);
750
+ }
751
+ filterItems() {
752
+ if (this.data.conditions) {
753
+ this.data.conditions.forEach((cond) => {
754
+ this.items = this.items.filter((item) => {
755
+ const operatorFunction = this.tableService.operators[cond.operator];
756
+ if (operatorFunction) {
757
+ return operatorFunction(item[cond.firestoreProperty], cond.dashProperty);
758
+ }
759
+ return false;
760
+ });
761
+ });
762
+ }
763
+ }
764
+ buildArrangeFromFilters() {
765
+ const activeFilters = this.filtersForm.controls
766
+ .flatMap((control) => {
767
+ const group = control;
768
+ const selectedFilter = group.get('selectFilter')?.value;
769
+ if (!selectedFilter)
770
+ return [];
771
+ const arrange = selectedFilter.arrange;
772
+ if (arrange === 'filter') {
773
+ const filterValue = group.get('typeFilter')?.value;
774
+ if (!filterValue)
775
+ return [];
776
+ return {
777
+ arrange,
778
+ filter: {
779
+ property: selectedFilter.property,
780
+ filtering: filterValue,
781
+ },
782
+ dateFilter: undefined,
783
+ };
784
+ }
785
+ if (arrange === 'filterByDate') {
786
+ const initial = group.get('initialDate')?.value;
787
+ const final = group.get('finalDate')?.value;
788
+ if (initial && final) {
789
+ try {
790
+ const [dayI, monthI, yearI] = initial.split('/');
791
+ const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
792
+ const [dayF, monthF, yearF] = final.split('/');
793
+ const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
794
+ finalDate.setHours(23, 59, 59);
795
+ return {
796
+ arrange,
797
+ filter: undefined,
798
+ dateFilter: {
799
+ initial: initialDate,
800
+ final: finalDate,
801
+ },
802
+ };
803
+ }
804
+ catch (error) {
805
+ return [];
806
+ }
807
+ }
808
+ return [];
809
+ }
810
+ if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {
811
+ const selectedItems = group.get('selectItem')?.value;
812
+ if (Array.isArray(selectedItems) && selectedItems.length > 0) {
813
+ return selectedItems.map((item) => ({
814
+ arrange,
815
+ filter: {
816
+ property: item.property,
817
+ filtering: item.value,
818
+ },
819
+ dateFilter: undefined,
820
+ }));
821
+ }
822
+ }
823
+ return [];
824
+ })
825
+ .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));
826
+ return {
827
+ filters: activeFilters,
828
+ sortBy: this.sortBy,
829
+ };
830
+ }
831
+ // Filtro de data
832
+ async search(event) {
833
+ // Prevenir comportamento padrão do formulário ao pressionar Enter
834
+ if (event) {
835
+ event.preventDefault();
836
+ event.stopPropagation();
837
+ }
838
+ if (this.selectSort.value) {
839
+ if (this.selectSort.value.arrange === 'ascending') {
840
+ this.sortBy = {
841
+ field: this.selectSort.value.property,
842
+ order: 'asc',
843
+ };
844
+ }
845
+ if (this.selectSort.value.arrange === 'descending') {
846
+ this.sortBy = {
847
+ field: this.selectSort.value.property,
848
+ order: 'desc',
849
+ };
850
+ }
851
+ }
852
+ // Sem paginação: aplicar filtros client-side
853
+ if (this.data.pagination === false) {
854
+ // Atualizar arrange com os filtros ativos
855
+ this.arrange = this.buildArrangeFromFilters();
856
+ this.applyFiltersToDataSource();
857
+ this.currentArrange =
858
+ this.filtersForm.length > 0
859
+ ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''
860
+ : '';
861
+ }
862
+ else {
863
+ // Com paginação: comportamento original
864
+ await this.loadItemsPaginated('reload', true);
865
+ this.currentArrange =
866
+ this.filtersForm.length > 0
867
+ ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''
868
+ : '';
869
+ this.paginator.firstPage();
870
+ }
871
+ }
872
+ async resetFilter() {
873
+ this.dataSource.filter = '';
874
+ if (this.filterPredicate) {
875
+ this.dataSource.filterPredicate = this.filterPredicate;
876
+ }
877
+ this.filtersForm.clear();
878
+ this.addFilter();
879
+ this.selectSort.patchValue('');
880
+ this.sortBy = {
881
+ order: this.data.sortBy ? this.data.sortBy.order : 'desc',
882
+ field: this.data.sortBy ? this.data.sortBy.field : 'createdAt',
883
+ };
884
+ // Sem paginação: recarregar itens originais
885
+ if (this.data.pagination === false) {
886
+ // Resetar arrange para valores padrão
887
+ this.arrange = {
888
+ filters: [],
889
+ sortBy: this.sortBy,
890
+ };
891
+ this.dataSource.data = [...this.items];
892
+ this.totalItems = this.items.length;
893
+ this.currentArrange = '';
894
+ }
895
+ else {
896
+ // Com paginação: comportamento original
897
+ await this.loadItemsPaginated('reload', true);
898
+ this.currentArrange = '';
899
+ this.paginator.firstPage();
900
+ }
901
+ }
902
+ // Método público para recarregar a tabela
903
+ async reloadTable() {
904
+ if (this.data.pagination) {
905
+ await this.loadItemsPaginated('reload', true);
906
+ this.paginator.firstPage();
907
+ }
908
+ else {
909
+ await this.loadItems();
910
+ }
911
+ }
912
+ updateDisplayedColumns() {
913
+ if (this.dataSource) {
914
+ this.dataSource = new MatTableDataSource([]);
915
+ }
916
+ this.columnProperties = this.data.displayedColumns.map((column) => {
917
+ return column.property;
918
+ });
919
+ this.dropdownItems = [];
920
+ this.sortableDropdownItems = [];
921
+ this.data.displayedColumns.forEach((col) => {
922
+ if (col.isFilterable) {
923
+ this.dropdownItems.push({
924
+ ...col,
925
+ arrange: 'filter',
926
+ title: col.title,
927
+ });
928
+ }
929
+ if (col.isSortable) {
930
+ this.sortableDropdownItems.push({
931
+ ...col,
932
+ arrange: 'ascending',
933
+ title: col.title + ': crescente',
934
+ });
935
+ this.sortableDropdownItems.push({
936
+ ...col,
937
+ arrange: 'descending',
938
+ title: col.title + ': decrescente',
939
+ });
940
+ }
941
+ if (col.isFilterableByDate) {
942
+ this.dropdownItems.push({
943
+ ...col,
944
+ arrange: 'filterByDate',
945
+ title: col.title + ': filtro por data',
946
+ });
947
+ }
948
+ });
949
+ if (this.data.filterableOptions &&
950
+ Array.isArray(this.data.filterableOptions)) {
951
+ this.data.filterableOptions.forEach((option) => this.dropdownItems.push({ ...option, arrange: 'equals' }));
952
+ }
953
+ }
954
+ isString(value) {
955
+ return typeof value === 'string';
956
+ }
957
+ // Métodos para controle do tooltip
958
+ onCellMouseEnter(event, row, col) {
959
+ // mostrar tooltip se a coluna tiver charLimit definido
960
+ if (!col.charLimit) {
961
+ return;
962
+ }
963
+ const fullValue = this.getDisplayValue(col, row, true);
964
+ // Só mostrar tooltip se o valor completo for maior que o limite
965
+ if (fullValue.length <= col.charLimit) {
966
+ return;
967
+ }
968
+ this.hoveredCell = { row, col };
969
+ this.tooltipContent = fullValue;
970
+ // Definir posição do tooltip
971
+ this.tooltipPosition = {
972
+ x: event.clientX + 10,
973
+ y: event.clientY - 10,
974
+ };
975
+ // Timeout para mostrar o tooltip
976
+ this.tooltipTimeout = setTimeout(() => {
977
+ if (this.hoveredCell &&
978
+ this.hoveredCell.row === row &&
979
+ this.hoveredCell.col === col) {
980
+ this.showTooltip = true;
981
+ }
982
+ }, 500);
983
+ }
984
+ onCellMouseLeave() {
985
+ if (this.tooltipTimeout) {
986
+ clearTimeout(this.tooltipTimeout);
987
+ this.tooltipTimeout = null;
988
+ }
989
+ this.showTooltip = false;
990
+ this.hoveredCell = null;
991
+ this.tooltipContent = '';
992
+ }
993
+ onCellMouseMove(event) {
994
+ if (this.showTooltip) {
995
+ this.tooltipPosition = {
996
+ x: event.clientX + 10,
997
+ y: event.clientY - 10,
998
+ };
999
+ }
1000
+ }
1001
+ // Métodos para inversão vertical dos tabs
1002
+ getTabGroups(tabs) {
1003
+ if (!tabs || tabs.length === 0)
1004
+ return [];
1005
+ const totalGroups = Math.ceil(tabs.length / 6);
1006
+ const groups = [];
1007
+ // Criar array de índices invertidos (último grupo primeiro)
1008
+ for (let i = totalGroups - 1; i >= 0; i--) {
1009
+ groups.push(i);
1010
+ }
1011
+ return groups;
1012
+ }
1013
+ getTabGroup(tabs, groupIndex) {
1014
+ if (!tabs || tabs.length === 0)
1015
+ return [];
1016
+ const startIndex = groupIndex * 6;
1017
+ const endIndex = Math.min(startIndex + 6, tabs.length);
1018
+ return tabs.slice(startIndex, endIndex);
1019
+ }
1020
+ getRealTabIndex(groupIndex, tabIndexInGroup) {
1021
+ if (!this.data.tabs?.tabsData)
1022
+ return 0;
1023
+ const totalGroups = Math.ceil(this.data.tabs.tabsData.length / 6);
1024
+ const realGroupIndex = totalGroups - 1 - groupIndex;
1025
+ return realGroupIndex * 6 + tabIndexInGroup;
1026
+ }
1027
+ onTableSelected(i, j) {
1028
+ if (!this.data.tabs?.tabsData || !this.data.tabs.method)
1029
+ return;
1030
+ this.selectedTab = this.getRealTabIndex(i, j);
1031
+ const tab = this.data.tabs.tabsData[this.selectedTab];
1032
+ if (tab) {
1033
+ this.data.tabs.method(tab, this.selectedTab);
1034
+ }
1035
+ }
1036
+ isTabSelected(originalIndex) {
1037
+ return this.selectedTab === originalIndex;
1038
+ }
1039
+ shouldShowActionButton() {
1040
+ if (!this.data?.actionButton) {
1041
+ return false;
1042
+ }
1043
+ if (!this.data.actionButton.condition) {
1044
+ return true;
1045
+ }
1046
+ try {
1047
+ return this.data.actionButton.condition(null) ?? true;
1048
+ }
1049
+ catch {
1050
+ return true;
1051
+ }
1052
+ }
1053
+ // Método para lidar com download (diferente para paginado e não paginado)
1054
+ handleDownload() {
1055
+ if (!this.downloadTable)
1056
+ return;
1057
+ // Se não há paginação, usar dados filtrados do dataSource
1058
+ if (this.data.pagination === false) {
1059
+ // Atualizar filteredItems com os dados filtrados do dataSource
1060
+ // (que já inclui filtro de texto aplicado via filterPredicate)
1061
+ if (this.dataSource && this.dataSource.filteredData) {
1062
+ this.filteredItems = [...this.dataSource.filteredData];
1063
+ }
1064
+ // Construir arrange com os filtros ativos (se houver)
1065
+ const arrange = this.buildArrangeFromFilters();
1066
+ this.downloadTable(arrange, this.data.conditions || []);
1067
+ }
1068
+ else {
1069
+ // Modo paginado: usar arrange existente (comportamento original)
1070
+ if (this.arrange) {
1071
+ this.downloadTable(this.arrange, this.data.conditions || []);
1072
+ }
1073
+ }
1074
+ }
1075
+ }
1076
+ 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.AngularFirestore }], target: i0.ɵɵFactoryTarget.Component });
1077
+ 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 }], ngImport: i0, template: "<div *ngIf=\"data\" class=\"card-body\">\n <div class=\"flex flex-col justify-between gap-6\">\n <!-- UNIFIED CONTROL PANEL: FILTERS, SORT & ACTIONS -->\n <div\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n *ngIf=\"\n data.pagination === true &&\n (dropdownItems.length > 0 ||\n sortableDropdownItems.length > 0 ||\n data.actionButton)\n \"\n >\n <!-- PANEL HEADER: Title, Custom Action, and Global Actions -->\n <div\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\"\n >\n <!-- Left Side: Title & Main Action Button -->\n <div class=\"flex flex-wrap items-center gap-4\">\n <div class=\"flex items-center gap-2\">\n <i class=\"fa fa-filter text-xl text-blue-500\"></i>\n <span class=\"text-lg font-semibold text-gray-700\"\n >Filtros e A\u00E7\u00F5es</span\n >\n </div>\n <button\n *ngIf=\"data.actionButton && data.actionButton.condition\"\n [ngClass]=\"\n (data.actionButton.colorClass || 'bg-blue-500') +\n ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\n \"\n [routerLink]=\"data.actionButton.routerLink\"\n (click)=\"\n data.actionButton.method ? data.actionButton.method($event) : null\n \"\n >\n <i\n *ngIf=\"data.actionButton.icon\"\n [class]=\"data.actionButton.icon\"\n ></i>\n {{ data.actionButton.label }}\n </button>\n </div>\n\n <!-- Right Side: Search, Reset, Export -->\n <div\n class=\"flex flex-wrap gap-3\"\n *ngIf=\"\n this.hasFilterableColumn === true || this.hasSortableColumn === true\n \"\n >\n <button\n (click)=\"search()\"\n type=\"button\"\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\"\n matTooltip=\"Aplicar filtros\"\n >\n <i class=\"fa fa-search\"></i>\n Pesquisar\n </button>\n\n <button\n (click)=\"resetFilter()\"\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\"\n matTooltip=\"Limpar filtros\"\n >\n <i class=\"fas fa-redo-alt\"></i>\n Resetar\n </button>\n\n <button\n *ngIf=\"data.download !== false && downloadTable\"\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\"\n matTooltipPosition=\"above\"\n matTooltip=\"Exportar Tabela\"\n [disabled]=\"\n this.dataSource && this.dataSource.filteredData.length <= 0\n \"\n (click)=\"\n $any(arrange) && downloadTable !== undefined\n ? downloadTable($any(arrange), data.conditions || [])\n : null\n \"\n >\n <i class=\"fa fa-download\"></i>\n Exportar\n </button>\n </div>\n </div>\n\n <!-- FILTERS CONTENT (WITH REFINEMENTS) -->\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\n <div\n [formGroup]=\"$any(filterGroup)\"\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\n class=\"flex flex-wrap items-center gap-3 rounded-lg border border-gray-200 p-2\"\n >\n <!-- FILTER TYPE SELECTOR -->\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>Tipo de filtro</mat-label>\n <mat-select\n placeholder=\"Selecione o tipo...\"\n formControlName=\"selectFilter\"\n (selectionChange)=\"onSelectFilterChange()\"\n >\n <mat-option *ngFor=\"let item of getAvailableFilterOptions()\" [value]=\"item\">\n <div class=\"flex items-center gap-2\">\n <i\n [class]=\"item.icon || 'fa fa-filter'\"\n class=\"text-sm text-blue-500\"\n ></i>\n <span>{{ item.title }}</span>\n </div>\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- TEXT FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-search text-gray-400\"></i>\n <span>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Filtrar\"\n }}</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n formControlName=\"typeFilter\"\n matInput\n placeholder=\"Digite para filtrar...\"\n #input\n />\n </mat-form-field>\n </div>\n\n <!-- DROPDOWN FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value &&\n $any(filterGroup)\n .get('selectFilter')\n ?.value.hasOwnProperty('items')\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Selecione\"\n }}</mat-label>\n <mat-select\n placeholder=\"Selecione...\"\n formControlName=\"selectItem\"\n multiple\n >\n <mat-option\n *ngFor=\"\n let item of $any(filterGroup).get('selectFilter')?.value\n .items\n \"\n [value]=\"item\"\n >\n {{ item.label }}\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- DATE FILTER -->\n <div\n class=\"min-w-[340px] flex-auto\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\n 'filterByDate'\n \"\n >\n <div\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\n >\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Inicial</span>\n </mat-label>\n <input\n matInput\n (keyup.enter)=\"search($event)\"\n formControlName=\"initialDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Final</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n matInput\n formControlName=\"finalDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n </div>\n </div>\n\n <!-- REMOVE FILTER BUTTON -->\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\n <button\n (click)=\"removeFilter(i)\"\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\n matTooltip=\"Remover filtro\"\n >\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\n </button>\n </div>\n </div>\n </div>\n\n <!-- PANEL FOOTER: Add Filter & Sort -->\n <div\n class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\n >\n <!-- Add Filter Button -->\n <div *ngIf=\"dropdownItems.length > 0\">\n <button\n (click)=\"addFilter()\"\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\"\n matTooltip=\"Adicionar novo filtro\"\n >\n <i class=\"fa fa-plus mr-2\"></i>\n Adicionar Filtro\n </button>\n </div>\n\n <!-- Sort Dropdown -->\n <div\n class=\"w-full sm:w-auto sm:min-w-[250px]\"\n *ngIf=\"sortableDropdownItems.length > 0\"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>Ordenar por</mat-label>\n <mat-select placeholder=\"Selecione...\" [formControl]=\"selectSort\">\n <mat-option\n *ngFor=\"let item of sortableDropdownItems\"\n [value]=\"item\"\n >\n <div class=\"flex items-center gap-2\">\n <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\n <span>{{ item.title }}</span>\n </div>\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n </div>\n </div>\n\n <!-- SIMPLE SEARCH (for non-paginated tables) -->\n <div\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n *ngIf=\"data.pagination === false && hasFilterableColumn === true\"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-search text-blue-500\"></i>\n Buscar\n </mat-label>\n <input\n matInput\n (keyup.enter)=\"search($event)\"\n (keyup)=\"applyFilter(filterInput.value)\"\n placeholder=\"Digite para filtrar...\"\n #filterInput\n />\n <mat-icon matSuffix class=\"text-gray-500\">search</mat-icon>\n </mat-form-field>\n <button\n *ngIf=\"data.actionButton\"\n [ngClass]=\"\n (data.actionButton.colorClass || 'bg-blue-500') +\n ' float-right flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\n \"\n [routerLink]=\"data.actionButton.routerLink\"\n (click)=\"\n data.actionButton.method ? data.actionButton.method($event) : null\n \"\n >\n <i *ngIf=\"data.actionButton.icon\" [class]=\"data.actionButton.icon\"></i>\n {{ data.actionButton.label }}\n </button>\n </div>\n\n <!-- FILTERS PANEL (for non-paginated tables) -->\n <div\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n *ngIf=\"data.pagination === false && dropdownItems.length > 0\"\n >\n <!-- FILTERS CONTENT -->\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\n <div\n [formGroup]=\"$any(filterGroup)\"\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\n class=\"flex flex-wrap items-center gap-3 rounded-lg border border-gray-200 p-2\"\n >\n <!-- FILTER TYPE SELECTOR -->\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>Tipo de filtro</mat-label>\n <mat-select\n placeholder=\"Selecione o tipo...\"\n formControlName=\"selectFilter\"\n (selectionChange)=\"onSelectFilterChange()\"\n >\n <mat-option *ngFor=\"let item of getAvailableFilterOptions()\" [value]=\"item\">\n <div class=\"flex items-center gap-2\">\n <i\n [class]=\"item.icon || 'fa fa-filter'\"\n class=\"text-sm text-blue-500\"\n ></i>\n <span>{{ item.title }}</span>\n </div>\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- TEXT FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-search text-gray-400\"></i>\n <span>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Filtrar\"\n }}</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n formControlName=\"typeFilter\"\n matInput\n placeholder=\"Digite para filtrar...\"\n #input\n />\n </mat-form-field>\n </div>\n\n <!-- DROPDOWN FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value &&\n $any(filterGroup)\n .get('selectFilter')\n ?.value.hasOwnProperty('items')\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Selecione\"\n }}</mat-label>\n <mat-select\n placeholder=\"Selecione...\"\n formControlName=\"selectItem\"\n multiple\n >\n <mat-option\n *ngFor=\"\n let item of $any(filterGroup).get('selectFilter')?.value\n .items\n \"\n [value]=\"item\"\n >\n {{ item.label }}\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- DATE FILTER -->\n <div\n class=\"min-w-[340px] flex-auto\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\n 'filterByDate'\n \"\n >\n <div\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\n >\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Inicial</span>\n </mat-label>\n <input\n matInput\n (keyup.enter)=\"search($event)\"\n (blur)=\"onDateFilterChange()\"\n formControlName=\"initialDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Final</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n (blur)=\"onDateFilterChange()\"\n matInput\n formControlName=\"finalDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n </div>\n </div>\n\n <!-- REMOVE FILTER BUTTON -->\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\n <button\n (click)=\"removeFilter(i)\"\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\n matTooltip=\"Remover filtro\"\n >\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\n </button>\n </div>\n </div>\n </div>\n\n <!-- PANEL FOOTER: Add Filter & Actions -->\n <div\n class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\n >\n <!-- Add Filter Button -->\n <div *ngIf=\"dropdownItems.length > 0\">\n <button\n (click)=\"addFilter()\"\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\"\n matTooltip=\"Adicionar novo filtro\"\n >\n <i class=\"fa fa-plus mr-2\"></i>\n Adicionar Filtro\n </button>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"flex flex-wrap gap-3\">\n <button\n (click)=\"search()\"\n type=\"button\"\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\"\n matTooltip=\"Aplicar filtros\"\n >\n <i class=\"fa fa-search\"></i>\n Aplicar\n </button>\n\n <button\n (click)=\"resetFilter()\"\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\"\n matTooltip=\"Limpar filtros\"\n >\n <i class=\"fas fa-redo-alt\"></i>\n Resetar\n </button>\n\n <button\n *ngIf=\"data.download !== false && downloadTable\"\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\"\n matTooltipPosition=\"above\"\n matTooltip=\"Exportar Tabela\"\n [disabled]=\"\n this.dataSource && this.dataSource.filteredData.length <= 0\n \"\n (click)=\"handleDownload()\"\n >\n <i class=\"fa fa-download\"></i>\n Exportar\n </button>\n </div>\n </div>\n </div>\n\n <div class=\"flex flex-col\">\n <div\n class=\"mx-auto flex flex-col\"\n *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\n >\n <!-- Calcular quantos grupos de 6 tabs existem -->\n <ng-container\n *ngFor=\"\n let groupIndex of getTabGroups(data.tabs.tabsData);\n let i = index\n \"\n >\n <div class=\"mx-auto flex flex-row\">\n <ng-container\n *ngFor=\"\n let tab of getTabGroup(data.tabs.tabsData, groupIndex);\n let j = index\n \"\n >\n <button\n class=\"border-2 border-gray-300 bg-gray-200 px-4 py-2 font-medium transition hover:brightness-95\"\n [ngClass]=\"\n isTabSelected(getRealTabIndex(i, j))\n ? 'border-b-0 brightness-110'\n : ''\n \"\n (click)=\"onTableSelected(i, j)\"\n >\n {{ tab.label }}\n <span\n *ngIf=\"tab.counter !== undefined\"\n class=\"ml-2 text-xs font-bold\"\n [ngClass]=\"tab.counterClass\"\n >\n {{ tab.counter }}\n </span>\n </button>\n </ng-container>\n </div>\n </ng-container>\n </div>\n <div class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl\">\n <table\n mat-table\n [dataSource]=\"dataSource\"\n matSort\n #sort=\"matSort\"\n matSortActive=\"createdAt\"\n matSortDirection=\"desc\"\n >\n <ng-container\n *ngFor=\"let col of data.displayedColumns\"\n matColumnDef=\"{{ col.property }}\"\n >\n <ng-container *matHeaderCellDef>\n <!-- IF THE COLUMN IS NOT SORTABLE, THEN DON'T SHOW THE SORT BUTTONS -->\n <th\n *ngIf=\"!col.isSortable || data.pagination === true\"\n mat-header-cell\n [ngClass]=\"\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\n (data.color?.text ? ' ' + $any(data.color).text : '')\n \"\n >\n {{ col.title }}\n </th>\n <!-- IF THE COLUMN IS SORTABLE, THEN SHOW THE SORT BUTTONS -->\n <th\n *ngIf=\"col.isSortable && data.pagination === false\"\n mat-header-cell\n mat-sort-header\n [ngClass]=\"\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\n (data.color?.text ? ' ' + $any(data.color).text : '')\n \"\n >\n {{ col.title }}\n </th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n (click)=\"col.method ? col.method(row) : null\"\n (mouseenter)=\"onCellMouseEnter($event, row, col)\"\n (mouseleave)=\"onCellMouseLeave()\"\n (mousemove)=\"onCellMouseMove($event)\"\n >\n <!-- CHECK IF THE COLUMN MUST BE DISPLAYED -->\n <span *ngIf=\"!col.image && !col.iconClass && !col.method\">\n <ng-container>\n <span\n *ngIf=\"\n col.charLimit &&\n row[col.property] &&\n row[col.property].length > col.charLimit;\n else withinLimit\n \"\n >\n <a\n *ngIf=\"col.hasLink === true\"\n [href]=\"row[col.property]\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row) }}\n </a>\n <a\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\n [href]=\"col.hasLink\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row) }}\n </a>\n <span\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\n >\n {{ getDisplayValue(col, row) }}\n </span>\n </span>\n </ng-container>\n <ng-template #withinLimit>\n <a\n *ngIf=\"col.hasLink === true\"\n [href]=\"row[col.property]\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row, true) }}\n </a>\n <a\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\n [href]=\"col.hasLink\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row, true) }}\n </a>\n <span\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\n >\n {{ getDisplayValue(col, row, true) }}\n </span>\n </ng-template>\n </span>\n <!------------------- IMAGE ------------------>\n <img\n *ngIf=\"\n col.image && col.image.path && !col.iconClass && !col.method\n \"\n [src]=\"col.image.path + '/' + row[col.property]\"\n [ngClass]=\"col.image.class\"\n alt=\"Imagem\"\n />\n <img\n *ngIf=\"\n col.image && col.image.url && !col.iconClass && !col.method\n \"\n [src]=\"row[col.property]\"\n [ngClass]=\"col.image.class\"\n alt=\"Imagem\"\n />\n <ng-container *ngIf=\"col.iconClass\">\n <button\n *ngFor=\"let iconClass of col.iconClass\"\n (click)=\"\n iconClass.buttonMethod\n ? iconClass.buttonMethod(row, $event)\n : $event.stopPropagation()\n \"\n >\n <span\n [ngClass]=\"iconClass.class\"\n *ngIf=\"\n iconClass.condition === undefined ||\n (iconClass.condition !== undefined &&\n $any(iconClass.condition)(row))\n \"\n >{{ iconClass.text }}</span\n >\n </button>\n </ng-container>\n </td>\n </ng-container>\n </ng-container>\n\n <tr mat-header-row *matHeaderRowDef=\"columnProperties\"></tr>\n <tr\n [ngClass]=\"{\n 'example-element-row': data.isNotClickable === true,\n 'example-element-row cursor-pointer': !data.isNotClickable\n }\"\n mat-row\n *matRowDef=\"let row; columns: columnProperties\"\n (click)=\"goToDetails(row)\"\n ></tr>\n\n <!-- ROW SHOWN WHEN THERE IS NO MATCHING DATA. -->\n <tr class=\"mat-row\" *matNoDataRow>\n <td *ngIf=\"!isLoading\" class=\"mat-cell p-4\" colspan=\"4\">\n Nenhum resultado encontrado para a busca\n </td>\n </tr>\n </table>\n\n <div class=\"flex justify-center\" *ngIf=\"isLoading\">\n <mat-spinner></mat-spinner>\n </div>\n\n <div class=\"paginator-container\">\n <mat-paginator\n #paginator\n [pageSizeOptions]=\"[25, 50, 100]\"\n [pageSize]=\"pageSize\"\n [length]=\"totalItems\"\n showFirstLastButtons\n aria-label=\"Select page of periodic elements\"\n (page)=\"onPageChange($event)\"\n [ngClass]=\"{\n 'hide-length':\n ['filter', 'filterByDate', 'equals'].includes(\n this.currentArrange\n ) || this.data.filterFn,\n 'hide-next-button': !hasNextPage && data.pagination === true,\n 'hide-last-button':\n (!hasNextPage && data.pagination === true) || this.data.filterFn\n }\"\n >\n </mat-paginator>\n <div\n *ngIf=\"\n !isLoading &&\n dataSource?.data &&\n dataSource.data.length > 0 &&\n data?.filterFn\n \"\n class=\"page-number-display\"\n >\n {{ currentPageNumber }}\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- TOOLTIP PERSONALIZADO -->\n <div\n *ngIf=\"showTooltip\"\n class=\"fixed z-50 max-w-md break-words rounded-lg bg-gray-800 px-3 py-2 text-sm text-white shadow-lg\"\n [style.left.px]=\"tooltipPosition.x\"\n [style.top.px]=\"tooltipPosition.y\"\n [style.pointer-events]=\"'none'\"\n >\n {{ tooltipContent }}\n </div>\n</div>\n", styles: ["::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}\n"], dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i5.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: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i5.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i5.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i5.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i5.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: i6.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i6.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i6.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i6.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { kind: "directive", type: i6.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i6.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i6.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i6.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i6.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i6.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: i6.MatNoDataRow, selector: "ng-template[matNoDataRow]" }, { kind: "component", type: i7.MatPaginator, selector: "mat-paginator", inputs: ["disabled"], exportAs: ["matPaginator"] }, { kind: "directive", type: i8.MatSort, selector: "[matSort]", inputs: ["matSortDisabled", "matSortActive", "matSortStart", "matSortDirection", "matSortDisableClear"], outputs: ["matSortChange"], exportAs: ["matSort"] }, { kind: "component", type: i8.MatSortHeader, selector: "[mat-sort-header]", inputs: ["disabled", "mat-sort-header", "arrowPosition", "start", "sortActionDescription", "disableClear"], exportAs: ["matSortHeader"] }, { kind: "component", type: i9.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i9.MatLabel, selector: "mat-label" }, { kind: "directive", type: i9.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "directive", type: i10.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: i11.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator"], exportAs: ["matSelect"] }, { kind: "component", type: i12.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i13.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "component", type: i14.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: i15.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i16.NgxMaskDirective, selector: "input[mask], textarea[mask]", inputs: ["mask", "specialCharacters", "patterns", "prefix", "suffix", "thousandSeparator", "decimalMarker", "dropSpecialCharacters", "hiddenInput", "showMaskTyped", "placeHolderCharacter", "shownMaskExpression", "showTemplate", "clearIfNotMatch", "validation", "separatorLimit", "allowNegativeNumbers", "leadZeroDateTime", "leadZero", "triggerOnMaskChange", "apm", "inputTransformFn", "outputTransformFn", "keepCharacterPositions"], outputs: ["maskFilled"], exportAs: ["mask", "ngxMask"] }] });
1078
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableComponent, decorators: [{
1079
+ type: Component,
1080
+ args: [{ selector: 'lib-table', template: "<div *ngIf=\"data\" class=\"card-body\">\n <div class=\"flex flex-col justify-between gap-6\">\n <!-- UNIFIED CONTROL PANEL: FILTERS, SORT & ACTIONS -->\n <div\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n *ngIf=\"\n data.pagination === true &&\n (dropdownItems.length > 0 ||\n sortableDropdownItems.length > 0 ||\n data.actionButton)\n \"\n >\n <!-- PANEL HEADER: Title, Custom Action, and Global Actions -->\n <div\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\"\n >\n <!-- Left Side: Title & Main Action Button -->\n <div class=\"flex flex-wrap items-center gap-4\">\n <div class=\"flex items-center gap-2\">\n <i class=\"fa fa-filter text-xl text-blue-500\"></i>\n <span class=\"text-lg font-semibold text-gray-700\"\n >Filtros e A\u00E7\u00F5es</span\n >\n </div>\n <button\n *ngIf=\"data.actionButton && data.actionButton.condition\"\n [ngClass]=\"\n (data.actionButton.colorClass || 'bg-blue-500') +\n ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\n \"\n [routerLink]=\"data.actionButton.routerLink\"\n (click)=\"\n data.actionButton.method ? data.actionButton.method($event) : null\n \"\n >\n <i\n *ngIf=\"data.actionButton.icon\"\n [class]=\"data.actionButton.icon\"\n ></i>\n {{ data.actionButton.label }}\n </button>\n </div>\n\n <!-- Right Side: Search, Reset, Export -->\n <div\n class=\"flex flex-wrap gap-3\"\n *ngIf=\"\n this.hasFilterableColumn === true || this.hasSortableColumn === true\n \"\n >\n <button\n (click)=\"search()\"\n type=\"button\"\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\"\n matTooltip=\"Aplicar filtros\"\n >\n <i class=\"fa fa-search\"></i>\n Pesquisar\n </button>\n\n <button\n (click)=\"resetFilter()\"\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\"\n matTooltip=\"Limpar filtros\"\n >\n <i class=\"fas fa-redo-alt\"></i>\n Resetar\n </button>\n\n <button\n *ngIf=\"data.download !== false && downloadTable\"\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\"\n matTooltipPosition=\"above\"\n matTooltip=\"Exportar Tabela\"\n [disabled]=\"\n this.dataSource && this.dataSource.filteredData.length <= 0\n \"\n (click)=\"\n $any(arrange) && downloadTable !== undefined\n ? downloadTable($any(arrange), data.conditions || [])\n : null\n \"\n >\n <i class=\"fa fa-download\"></i>\n Exportar\n </button>\n </div>\n </div>\n\n <!-- FILTERS CONTENT (WITH REFINEMENTS) -->\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\n <div\n [formGroup]=\"$any(filterGroup)\"\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\n class=\"flex flex-wrap items-center gap-3 rounded-lg border border-gray-200 p-2\"\n >\n <!-- FILTER TYPE SELECTOR -->\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>Tipo de filtro</mat-label>\n <mat-select\n placeholder=\"Selecione o tipo...\"\n formControlName=\"selectFilter\"\n (selectionChange)=\"onSelectFilterChange()\"\n >\n <mat-option *ngFor=\"let item of getAvailableFilterOptions()\" [value]=\"item\">\n <div class=\"flex items-center gap-2\">\n <i\n [class]=\"item.icon || 'fa fa-filter'\"\n class=\"text-sm text-blue-500\"\n ></i>\n <span>{{ item.title }}</span>\n </div>\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- TEXT FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-search text-gray-400\"></i>\n <span>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Filtrar\"\n }}</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n formControlName=\"typeFilter\"\n matInput\n placeholder=\"Digite para filtrar...\"\n #input\n />\n </mat-form-field>\n </div>\n\n <!-- DROPDOWN FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value &&\n $any(filterGroup)\n .get('selectFilter')\n ?.value.hasOwnProperty('items')\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Selecione\"\n }}</mat-label>\n <mat-select\n placeholder=\"Selecione...\"\n formControlName=\"selectItem\"\n multiple\n >\n <mat-option\n *ngFor=\"\n let item of $any(filterGroup).get('selectFilter')?.value\n .items\n \"\n [value]=\"item\"\n >\n {{ item.label }}\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- DATE FILTER -->\n <div\n class=\"min-w-[340px] flex-auto\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\n 'filterByDate'\n \"\n >\n <div\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\n >\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Inicial</span>\n </mat-label>\n <input\n matInput\n (keyup.enter)=\"search($event)\"\n formControlName=\"initialDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Final</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n matInput\n formControlName=\"finalDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n </div>\n </div>\n\n <!-- REMOVE FILTER BUTTON -->\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\n <button\n (click)=\"removeFilter(i)\"\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\n matTooltip=\"Remover filtro\"\n >\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\n </button>\n </div>\n </div>\n </div>\n\n <!-- PANEL FOOTER: Add Filter & Sort -->\n <div\n class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\n >\n <!-- Add Filter Button -->\n <div *ngIf=\"dropdownItems.length > 0\">\n <button\n (click)=\"addFilter()\"\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\"\n matTooltip=\"Adicionar novo filtro\"\n >\n <i class=\"fa fa-plus mr-2\"></i>\n Adicionar Filtro\n </button>\n </div>\n\n <!-- Sort Dropdown -->\n <div\n class=\"w-full sm:w-auto sm:min-w-[250px]\"\n *ngIf=\"sortableDropdownItems.length > 0\"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>Ordenar por</mat-label>\n <mat-select placeholder=\"Selecione...\" [formControl]=\"selectSort\">\n <mat-option\n *ngFor=\"let item of sortableDropdownItems\"\n [value]=\"item\"\n >\n <div class=\"flex items-center gap-2\">\n <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\n <span>{{ item.title }}</span>\n </div>\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n </div>\n </div>\n\n <!-- SIMPLE SEARCH (for non-paginated tables) -->\n <div\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n *ngIf=\"data.pagination === false && hasFilterableColumn === true\"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-search text-blue-500\"></i>\n Buscar\n </mat-label>\n <input\n matInput\n (keyup.enter)=\"search($event)\"\n (keyup)=\"applyFilter(filterInput.value)\"\n placeholder=\"Digite para filtrar...\"\n #filterInput\n />\n <mat-icon matSuffix class=\"text-gray-500\">search</mat-icon>\n </mat-form-field>\n <button\n *ngIf=\"data.actionButton\"\n [ngClass]=\"\n (data.actionButton.colorClass || 'bg-blue-500') +\n ' float-right flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\n \"\n [routerLink]=\"data.actionButton.routerLink\"\n (click)=\"\n data.actionButton.method ? data.actionButton.method($event) : null\n \"\n >\n <i *ngIf=\"data.actionButton.icon\" [class]=\"data.actionButton.icon\"></i>\n {{ data.actionButton.label }}\n </button>\n </div>\n\n <!-- FILTERS PANEL (for non-paginated tables) -->\n <div\n class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n *ngIf=\"data.pagination === false && dropdownItems.length > 0\"\n >\n <!-- FILTERS CONTENT -->\n <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\n <div\n [formGroup]=\"$any(filterGroup)\"\n *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\n class=\"flex flex-wrap items-center gap-3 rounded-lg border border-gray-200 p-2\"\n >\n <!-- FILTER TYPE SELECTOR -->\n <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>Tipo de filtro</mat-label>\n <mat-select\n placeholder=\"Selecione o tipo...\"\n formControlName=\"selectFilter\"\n (selectionChange)=\"onSelectFilterChange()\"\n >\n <mat-option *ngFor=\"let item of getAvailableFilterOptions()\" [value]=\"item\">\n <div class=\"flex items-center gap-2\">\n <i\n [class]=\"item.icon || 'fa fa-filter'\"\n class=\"text-sm text-blue-500\"\n ></i>\n <span>{{ item.title }}</span>\n </div>\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- TEXT FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-search text-gray-400\"></i>\n <span>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Filtrar\"\n }}</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n formControlName=\"typeFilter\"\n matInput\n placeholder=\"Digite para filtrar...\"\n #input\n />\n </mat-form-field>\n </div>\n\n <!-- DROPDOWN FILTER -->\n <div\n class=\"min-w-[200px] flex-1\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value &&\n $any(filterGroup)\n .get('selectFilter')\n ?.value.hasOwnProperty('items')\n \"\n >\n <mat-form-field appearance=\"outline\" class=\"w-full\">\n <mat-label>{{\n $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n \"Selecione\"\n }}</mat-label>\n <mat-select\n placeholder=\"Selecione...\"\n formControlName=\"selectItem\"\n multiple\n >\n <mat-option\n *ngFor=\"\n let item of $any(filterGroup).get('selectFilter')?.value\n .items\n \"\n [value]=\"item\"\n >\n {{ item.label }}\n </mat-option>\n </mat-select>\n </mat-form-field>\n </div>\n\n <!-- DATE FILTER -->\n <div\n class=\"min-w-[340px] flex-auto\"\n *ngIf=\"\n $any(filterGroup).get('selectFilter')?.value?.arrange ===\n 'filterByDate'\n \"\n >\n <div\n class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\n >\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Inicial</span>\n </mat-label>\n <input\n matInput\n (keyup.enter)=\"search($event)\"\n (blur)=\"onDateFilterChange()\"\n formControlName=\"initialDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n\n <mat-form-field appearance=\"outline\" class=\"flex-1\">\n <mat-label class=\"flex items-center gap-2\">\n <i class=\"fa fa-calendar text-gray-400\"></i>\n <span>Data Final</span>\n </mat-label>\n <input\n (keyup.enter)=\"search($event)\"\n (blur)=\"onDateFilterChange()\"\n matInput\n formControlName=\"finalDate\"\n [dropSpecialCharacters]=\"false\"\n mask=\"d0/M0/0000\"\n placeholder=\"DD/MM/AAAA\"\n maxlength=\"10\"\n />\n </mat-form-field>\n </div>\n </div>\n\n <!-- REMOVE FILTER BUTTON -->\n <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\n <button\n (click)=\"removeFilter(i)\"\n class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\n matTooltip=\"Remover filtro\"\n >\n <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\n </button>\n </div>\n </div>\n </div>\n\n <!-- PANEL FOOTER: Add Filter & Actions -->\n <div\n class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\n >\n <!-- Add Filter Button -->\n <div *ngIf=\"dropdownItems.length > 0\">\n <button\n (click)=\"addFilter()\"\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\"\n matTooltip=\"Adicionar novo filtro\"\n >\n <i class=\"fa fa-plus mr-2\"></i>\n Adicionar Filtro\n </button>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"flex flex-wrap gap-3\">\n <button\n (click)=\"search()\"\n type=\"button\"\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\"\n matTooltip=\"Aplicar filtros\"\n >\n <i class=\"fa fa-search\"></i>\n Aplicar\n </button>\n\n <button\n (click)=\"resetFilter()\"\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\"\n matTooltip=\"Limpar filtros\"\n >\n <i class=\"fas fa-redo-alt\"></i>\n Resetar\n </button>\n\n <button\n *ngIf=\"data.download !== false && downloadTable\"\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\"\n matTooltipPosition=\"above\"\n matTooltip=\"Exportar Tabela\"\n [disabled]=\"\n this.dataSource && this.dataSource.filteredData.length <= 0\n \"\n (click)=\"handleDownload()\"\n >\n <i class=\"fa fa-download\"></i>\n Exportar\n </button>\n </div>\n </div>\n </div>\n\n <div class=\"flex flex-col\">\n <div\n class=\"mx-auto flex flex-col\"\n *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\n >\n <!-- Calcular quantos grupos de 6 tabs existem -->\n <ng-container\n *ngFor=\"\n let groupIndex of getTabGroups(data.tabs.tabsData);\n let i = index\n \"\n >\n <div class=\"mx-auto flex flex-row\">\n <ng-container\n *ngFor=\"\n let tab of getTabGroup(data.tabs.tabsData, groupIndex);\n let j = index\n \"\n >\n <button\n class=\"border-2 border-gray-300 bg-gray-200 px-4 py-2 font-medium transition hover:brightness-95\"\n [ngClass]=\"\n isTabSelected(getRealTabIndex(i, j))\n ? 'border-b-0 brightness-110'\n : ''\n \"\n (click)=\"onTableSelected(i, j)\"\n >\n {{ tab.label }}\n <span\n *ngIf=\"tab.counter !== undefined\"\n class=\"ml-2 text-xs font-bold\"\n [ngClass]=\"tab.counterClass\"\n >\n {{ tab.counter }}\n </span>\n </button>\n </ng-container>\n </div>\n </ng-container>\n </div>\n <div class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl\">\n <table\n mat-table\n [dataSource]=\"dataSource\"\n matSort\n #sort=\"matSort\"\n matSortActive=\"createdAt\"\n matSortDirection=\"desc\"\n >\n <ng-container\n *ngFor=\"let col of data.displayedColumns\"\n matColumnDef=\"{{ col.property }}\"\n >\n <ng-container *matHeaderCellDef>\n <!-- IF THE COLUMN IS NOT SORTABLE, THEN DON'T SHOW THE SORT BUTTONS -->\n <th\n *ngIf=\"!col.isSortable || data.pagination === true\"\n mat-header-cell\n [ngClass]=\"\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\n (data.color?.text ? ' ' + $any(data.color).text : '')\n \"\n >\n {{ col.title }}\n </th>\n <!-- IF THE COLUMN IS SORTABLE, THEN SHOW THE SORT BUTTONS -->\n <th\n *ngIf=\"col.isSortable && data.pagination === false\"\n mat-header-cell\n mat-sort-header\n [ngClass]=\"\n (data.color?.bg ? ' ' + $any(data.color).bg : '') +\n (data.color?.text ? ' ' + $any(data.color).text : '')\n \"\n >\n {{ col.title }}\n </th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n (click)=\"col.method ? col.method(row) : null\"\n (mouseenter)=\"onCellMouseEnter($event, row, col)\"\n (mouseleave)=\"onCellMouseLeave()\"\n (mousemove)=\"onCellMouseMove($event)\"\n >\n <!-- CHECK IF THE COLUMN MUST BE DISPLAYED -->\n <span *ngIf=\"!col.image && !col.iconClass && !col.method\">\n <ng-container>\n <span\n *ngIf=\"\n col.charLimit &&\n row[col.property] &&\n row[col.property].length > col.charLimit;\n else withinLimit\n \"\n >\n <a\n *ngIf=\"col.hasLink === true\"\n [href]=\"row[col.property]\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row) }}\n </a>\n <a\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\n [href]=\"col.hasLink\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row) }}\n </a>\n <span\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\n >\n {{ getDisplayValue(col, row) }}\n </span>\n </span>\n </ng-container>\n <ng-template #withinLimit>\n <a\n *ngIf=\"col.hasLink === true\"\n [href]=\"row[col.property]\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row, true) }}\n </a>\n <a\n *ngIf=\"col.hasLink && isString(col.hasLink)\"\n [href]=\"col.hasLink\"\n target=\"_blank\"\n >\n {{ getDisplayValue(col, row, true) }}\n </a>\n <span\n *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\n >\n {{ getDisplayValue(col, row, true) }}\n </span>\n </ng-template>\n </span>\n <!------------------- IMAGE ------------------>\n <img\n *ngIf=\"\n col.image && col.image.path && !col.iconClass && !col.method\n \"\n [src]=\"col.image.path + '/' + row[col.property]\"\n [ngClass]=\"col.image.class\"\n alt=\"Imagem\"\n />\n <img\n *ngIf=\"\n col.image && col.image.url && !col.iconClass && !col.method\n \"\n [src]=\"row[col.property]\"\n [ngClass]=\"col.image.class\"\n alt=\"Imagem\"\n />\n <ng-container *ngIf=\"col.iconClass\">\n <button\n *ngFor=\"let iconClass of col.iconClass\"\n (click)=\"\n iconClass.buttonMethod\n ? iconClass.buttonMethod(row, $event)\n : $event.stopPropagation()\n \"\n >\n <span\n [ngClass]=\"iconClass.class\"\n *ngIf=\"\n iconClass.condition === undefined ||\n (iconClass.condition !== undefined &&\n $any(iconClass.condition)(row))\n \"\n >{{ iconClass.text }}</span\n >\n </button>\n </ng-container>\n </td>\n </ng-container>\n </ng-container>\n\n <tr mat-header-row *matHeaderRowDef=\"columnProperties\"></tr>\n <tr\n [ngClass]=\"{\n 'example-element-row': data.isNotClickable === true,\n 'example-element-row cursor-pointer': !data.isNotClickable\n }\"\n mat-row\n *matRowDef=\"let row; columns: columnProperties\"\n (click)=\"goToDetails(row)\"\n ></tr>\n\n <!-- ROW SHOWN WHEN THERE IS NO MATCHING DATA. -->\n <tr class=\"mat-row\" *matNoDataRow>\n <td *ngIf=\"!isLoading\" class=\"mat-cell p-4\" colspan=\"4\">\n Nenhum resultado encontrado para a busca\n </td>\n </tr>\n </table>\n\n <div class=\"flex justify-center\" *ngIf=\"isLoading\">\n <mat-spinner></mat-spinner>\n </div>\n\n <div class=\"paginator-container\">\n <mat-paginator\n #paginator\n [pageSizeOptions]=\"[25, 50, 100]\"\n [pageSize]=\"pageSize\"\n [length]=\"totalItems\"\n showFirstLastButtons\n aria-label=\"Select page of periodic elements\"\n (page)=\"onPageChange($event)\"\n [ngClass]=\"{\n 'hide-length':\n ['filter', 'filterByDate', 'equals'].includes(\n this.currentArrange\n ) || this.data.filterFn,\n 'hide-next-button': !hasNextPage && data.pagination === true,\n 'hide-last-button':\n (!hasNextPage && data.pagination === true) || this.data.filterFn\n }\"\n >\n </mat-paginator>\n <div\n *ngIf=\"\n !isLoading &&\n dataSource?.data &&\n dataSource.data.length > 0 &&\n data?.filterFn\n \"\n class=\"page-number-display\"\n >\n {{ currentPageNumber }}\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- TOOLTIP PERSONALIZADO -->\n <div\n *ngIf=\"showTooltip\"\n class=\"fixed z-50 max-w-md break-words rounded-lg bg-gray-800 px-3 py-2 text-sm text-white shadow-lg\"\n [style.left.px]=\"tooltipPosition.x\"\n [style.top.px]=\"tooltipPosition.y\"\n [style.pointer-events]=\"'none'\"\n >\n {{ tooltipContent }}\n </div>\n</div>\n", styles: ["::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}\n"] }]
1081
+ }], ctorParameters: function () { return [{ type: i1.Router }, { type: i2.TableService }, { type: i3.AngularFirestore }]; }, propDecorators: { data: [{
1082
+ type: Input
1083
+ }], downloadTable: [{
1084
+ type: Input
1085
+ }], paginator: [{
1086
+ type: ViewChild,
1087
+ args: [MatPaginator]
1088
+ }], sort: [{
1089
+ type: ViewChild,
1090
+ args: [MatSort]
1091
+ }] } });
1092
+ //# 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":"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","sourcesContent":["import { Arrange, Condition, Tab, TabData, TableData } from '../../types/Table';\nimport { Component, Input, OnInit, ViewChild } from '@angular/core';\nimport {\n  AngularFirestore,\n  QueryDocumentSnapshot,\n} from '@angular/fire/compat/firestore';\nimport { MatPaginator, PageEvent } from '@angular/material/paginator';\nimport { MatSort } from '@angular/material/sort';\nimport { MatTableDataSource } from '@angular/material/table';\nimport { Router } from '@angular/router';\nimport { TableService } from '../../services/table.service';\nimport { debounceTime, firstValueFrom, Subject } from 'rxjs';\nimport { OrderByDirection } from 'firebase/firestore';\nimport firebase from 'firebase/compat';\nimport WhereFilterOp = firebase.firestore.WhereFilterOp;\nimport firestore = firebase.firestore;\nimport { FormControl, FormArray, FormGroup } from '@angular/forms';\n\n@Component({\n  selector: 'lib-table',\n  templateUrl: './table.component.html',\n  styleUrls: ['./table.component.scss'],\n})\nexport class TableComponent implements OnInit {\n  // INPUTS\n  @Input() data!: TableData;\n  @Input() downloadTable?: (arrange: Arrange, conditions: Condition[]) => void;\n  arrange: Arrange | null = null;\n  // VARIABLES\n  dataSource!: MatTableDataSource<any>;\n  currentPageNumber = 1;\n  currentClientPageIndex = 0;\n\n  public items: any[] = [];\n  public filteredItems: any[] = []; // Dados filtrados para modo não paginado (público para acesso externo)\n  public isLoading = false;\n  private lastDoc: QueryDocumentSnapshot<any> | null = null;\n  private firstDoc: QueryDocumentSnapshot<any> | null = null;\n  private sortBy: { field: string; order: OrderByDirection } = {\n    field: 'createdAt',\n    order: 'desc',\n  };\n  public columnProperties: string[] = [];\n\n  filtersForm: FormArray;\n\n  selectSort: FormControl = new FormControl('');\n  currentArrange = '';\n\n  hasNextPage: boolean = false;\n\n  dropdownItems: any[] = [];\n  sortableDropdownItems: any[] = [];\n  pageSize = 25;\n  totalItems = 0;\n  pageEvent?: PageEvent;\n  filterValue: string | null = null;\n\n  hasFilterableColumn: boolean = false;\n  hasSortableColumn: boolean = false;\n  filterPredicate: ((data: any, filter: string) => boolean) | undefined;\n  private filterSubject = new Subject<string>();\n  private readonly debounceTimeMs = 500;\n\n  selectedTab: number = 0;\n\n  @ViewChild(MatPaginator) paginator!: MatPaginator;\n  @ViewChild(MatSort) sort!: MatSort;\n\n  // Propriedades para controle do tooltip\n  hoveredCell: { row: any; col: any } | null = null;\n  showTooltip: boolean = false;\n  tooltipTimeout: any;\n  tooltipContent: string = '';\n  tooltipPosition: { x: number; y: number } = { x: 0, y: 0 };\n\n  // CONSTRUCTOR\n  constructor(\n    private router: Router,\n    private tableService: TableService,\n    private firestore: AngularFirestore\n  ) {\n    this.filtersForm = new FormArray([this.createFilterGroup()]);\n  }\n\n  createFilterGroup(): FormGroup {\n    return new FormGroup({\n      selectFilter: new FormControl(''),\n      typeFilter: new FormControl(''),\n      selectItem: new FormControl(''),\n      initialDate: new FormControl('', [\n        this.tableService.dateFormatValidator(),\n      ]),\n      finalDate: new FormControl('', [this.tableService.dateFormatValidator()]),\n    });\n  }\n\n  addFilter(filterData?: {\n    selectFilter: any;\n    typeFilter?: string;\n    selectItem?: any[];\n    initialDate?: string;\n    finalDate?: string;\n  }): void {\n    if (\n      this.data.pagination === true &&\n      this.hasActiveDateFilter() &&\n      filterData?.selectFilter?.arrange === 'filterByDate'\n    ) {\n      // Não permitir adicionar novo filtro de data quando já existe um ativo\n      return;\n    }\n\n    const newFilterGroup = this.createFilterGroup();\n\n    if (filterData) {\n      if (filterData.selectFilter) {\n        newFilterGroup.get('selectFilter')?.setValue(filterData.selectFilter);\n      }\n      if (filterData.typeFilter) {\n        newFilterGroup.get('typeFilter')?.setValue(filterData.typeFilter);\n      }\n      if (filterData.selectItem) {\n        newFilterGroup.get('selectItem')?.setValue(filterData.selectItem);\n      }\n      if (filterData.initialDate) {\n        newFilterGroup.get('initialDate')?.setValue(filterData.initialDate);\n      }\n      if (filterData.finalDate) {\n        newFilterGroup.get('finalDate')?.setValue(filterData.finalDate);\n      }\n    }\n\n    this.filtersForm.push(newFilterGroup);\n  }\n\n  hasActiveDateFilter(): boolean {\n    return this.filtersForm.controls.some((control) => {\n      const group = control as FormGroup;\n      const selectedFilter = group.get('selectFilter')?.value;\n      return selectedFilter?.arrange === 'filterByDate';\n    });\n  }\n\n  getAvailableFilterOptions(): any[] {\n    if (this.data.pagination === true && this.hasActiveDateFilter()) {\n      return this.dropdownItems.filter(\n        (item) => item.arrange !== 'filterByDate'\n      );\n    }\n    return this.dropdownItems;\n  }\n\n  onSelectFilterChange() {\n    const lastIndex = this.filtersForm.length - 1;\n    const lastFilter = this.filtersForm.at(lastIndex);\n    const selectedFilter = lastFilter.get('selectFilter')?.value;\n\n    if (selectedFilter) {\n      if (\n        selectedFilter.arrange === 'filterByDate' &&\n        this.data.pagination === true\n      ) {\n        return;\n      }\n\n      this.addFilter();\n    }\n  }\n\n  removeFilter(index: number): void {\n    this.filtersForm.removeAt(index);\n    if (this.filtersForm.length === 0) {\n      this.addFilter();\n    }\n  }\n\n  removeAllFilters(): void {\n    this.filtersForm.clear();\n    this.addFilter();\n    this.resetFilter();\n  }\n\n  // METHODS\n  public async ngOnInit() {\n    if (!this.data.color)\n      this.data.color = { bg: 'bg-primary', text: 'text-black' };\n    this.columnProperties = this.data.displayedColumns.map((column) => {\n      return column.property;\n    });\n\n    if (this.data.actionButton && !this.data.actionButton.condition) {\n      this.data.actionButton.condition = (_row?: any) => true;\n    }\n\n    this.data.displayedColumns.forEach((col) => {\n      if (col.isFilterable) {\n        if (this.hasFilterableColumn === false) this.hasFilterableColumn = true;\n\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filter',\n          title: col.title,\n        });\n      }\n      if (col.isSortable) {\n        if (this.hasSortableColumn === false) this.hasSortableColumn = true;\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'ascending',\n          title: col.title + ': crescente',\n        });\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'descending',\n          title: col.title + ': decrescente',\n        });\n      }\n      if (col.isFilterableByDate) {\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filterByDate',\n          title: col.title + ': filtro por data',\n        });\n      }\n    });\n    if (\n      this.data.filterableOptions &&\n      Array.isArray(this.data.filterableOptions) &&\n      this.data.filterableOptions.length > 0\n    ) {\n      this.data.filterableOptions.forEach((option) =>\n        this.dropdownItems.push({ ...option, arrange: 'equals' })\n      );\n    }\n\n    // Sem paginação\n    if (this.data.pagination === false) {\n      await this.loadItems();\n    }\n    // Com paginação\n    if (this.data.pagination === true) {\n      if (this.data.sortBy)\n        this.sortBy = {\n          field: this.data.sortBy.field,\n          order: this.data.sortBy.order,\n        };\n      this.filterSubject\n        .pipe(debounceTime(this.debounceTimeMs))\n        .subscribe(() => {\n          this.loadItemsPaginated('reload', true);\n        });\n\n      this.isLoading = true;\n      await this.loadItemsPaginated('reload', true);\n\n      this.sort.active = 'createdAt';\n      this.sort.direction = 'desc';\n      this.dataSource.paginator = this.paginator;\n      this.dataSource.sort = this.sort;\n      this.totalItems = 0;\n      if (this.data.totalRef) {\n        for (const totalRef of this.data.totalRef) {\n          const totalRefDoc = await totalRef.ref.get();\n          const docData = totalRefDoc.data() as any;\n          if (docData && docData[totalRef.field])\n            this.totalItems = (this.totalItems +\n              docData[totalRef.field]) as number;\n        }\n      }\n      this.isLoading = false;\n    }\n  }\n\n  getDisplayValue(col: any, row: any, withinLimit: boolean = false): string {\n    let value: any;\n\n    if (col.calculateValue) {\n      value = col.calculateValue(row);\n    } else {\n      value = this.getNestedValue(row, col.property);\n    }\n\n    if (Array.isArray(value) && col.arrayField) {\n      value = this.formatArrayValue(value, col.arrayField);\n    }\n\n    if (col.queryLength && row[col.property]) {\n      value = row[col.property];\n    }\n\n    value = col.pipe ? col.pipe.transform(value) : value;\n\n    if (value === null || value === undefined) {\n      value = '';\n    } else {\n      value = String(value);\n    }\n\n    if (typeof value !== 'string') {\n      value = '';\n    }\n\n    if (withinLimit || !col.charLimit) {\n      return value;\n    }\n\n    return value.substring(0, col.charLimit) + '...';\n  }\n\n  getNestedValue(obj: any, path: string): any {\n    if (!path) return undefined;\n    const properties = path.split('.');\n    return properties.reduce(\n      (acc, currentPart) => acc && acc[currentPart],\n      obj\n    );\n  }\n\n  private formatArrayValue(array: any[], field: string): string {\n    if (!Array.isArray(array) || array.length === 0) {\n      return '';\n    }\n\n    const values = array\n      .map((item) => {\n        if (typeof item === 'object' && item !== null) {\n          return item[field] || '';\n        }\n        return String(item);\n      })\n      .filter((value) => value !== '' && value !== null && value !== undefined);\n\n    return values.join(', ');\n  }\n\n  // Métodos sem paginação\n  private async loadItems() {\n    this.items = await this.tableService.getItems(this.data.collectionRef);\n    if (this.data.conditions) {\n      this.filterItems();\n    }\n    await this.loadRelations();\n    await this.loadQueryLengths();\n    this.totalItems = this.items.length;\n\n    // Inicializar arrange para tabelas não paginadas\n    this.arrange = {\n      filters: [],\n      sortBy: this.data.sortBy || {\n        field: 'createdAt',\n        order: 'desc',\n      },\n    };\n    this.sortBy = this.data.sortBy || {\n      field: 'createdAt',\n      order: 'desc',\n    };\n\n    // Aplicar filtros client-side se existirem\n    let itemsToDisplay = [...this.items];\n    itemsToDisplay = this.applyClientSideFilters(itemsToDisplay);\n    this.filteredItems = itemsToDisplay; // Armazenar dados filtrados\n\n    this.dataSource = new MatTableDataSource(itemsToDisplay);\n    this.dataSource.paginator = this.paginator;\n    this.dataSource.sort = this.sort;\n    if (this.data.sortBy) {\n      this.dataSource.sort.active = this.data.sortBy.field;\n      this.dataSource.sort.direction = this.data.sortBy.order;\n      this.dataSource.sort.sortChange.emit();\n    }\n\n    this.dataSource.filterPredicate = (data, filter: string) => {\n      return this.data.displayedColumns.some((col): boolean => {\n        if (col.filterPredicates) {\n          return col.filterPredicates.some((predicate): boolean => {\n            const propertyValue = data[col.property];\n            if (!propertyValue || typeof propertyValue !== 'object') {\n              return false;\n            }\n\n            const predicateValue = propertyValue[predicate];\n            if (predicateValue === null || predicateValue === undefined) {\n              return false;\n            }\n\n            return String(predicateValue)\n              .trim()\n              .toLocaleLowerCase()\n              .includes(filter);\n          });\n        }\n\n        if (col.property && col.isFilterable) {\n          const propertyValue = data[col.property];\n          if (propertyValue === null || propertyValue === undefined) {\n            return false;\n          }\n\n          return String(propertyValue)\n            .trim()\n            .toLocaleLowerCase()\n            .includes(filter);\n        }\n        return false;\n      });\n    };\n    this.filterPredicate = this.dataSource.filterPredicate;\n  }\n\n  // Aplicar filtros client-side (filtros por data)\n  private applyClientSideFilters(items: any[]): any[] {\n    let filteredItems = [...items];\n\n    // Agrupar filtros de data por propriedade\n    const dateFiltersByProperty: Record<\n      string,\n      Array<{ initial: Date; final: Date }>\n    > = {};\n    const otherFilters: Array<{\n      group: FormGroup;\n      selectedFilter: any;\n      arrange: string;\n    }> = [];\n\n    this.filtersForm.controls.forEach((control) => {\n      const group = control as FormGroup;\n      const selectedFilter = group.get('selectFilter')?.value;\n      if (!selectedFilter) return;\n\n      const arrange = selectedFilter.arrange;\n\n      if (arrange === 'filterByDate') {\n        const initial = group.get('initialDate')?.value;\n        const final = group.get('finalDate')?.value;\n\n        // Só aplicar filtro se ambas as datas estiverem preenchidas\n        if (initial && final && initial.trim() && final.trim()) {\n          try {\n            // Validar formato da data (DD/MM/AAAA)\n            const datePattern = /^(\\d{1,2})\\/(\\d{1,2})\\/(\\d{4})$/;\n            if (!datePattern.test(initial) || !datePattern.test(final)) {\n              return; // Ignorar se formato inválido\n            }\n\n            const [dayI, monthI, yearI] = initial.split('/');\n            const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\n            const [dayF, monthF, yearF] = final.split('/');\n            const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\n            finalDate.setHours(23, 59, 59);\n\n            // Validar se as datas são válidas\n            if (isNaN(initialDate.getTime()) || isNaN(finalDate.getTime())) {\n              return; // Ignorar se datas inválidas\n            }\n\n            // Usar o campo da coluna ou o sortBy.field como padrão\n            const dateField = selectedFilter.property || this.sortBy.field;\n\n            if (!dateFiltersByProperty[dateField]) {\n              dateFiltersByProperty[dateField] = [];\n            }\n\n            dateFiltersByProperty[dateField].push({\n              initial: initialDate,\n              final: finalDate,\n            });\n          } catch (error) {\n            console.warn('Erro ao processar datas do filtro:', error);\n          }\n        }\n      } else {\n        otherFilters.push({ group, selectedFilter, arrange });\n      }\n    });\n\n    Object.keys(dateFiltersByProperty).forEach((dateField) => {\n      const intervals = dateFiltersByProperty[dateField];\n\n      filteredItems = filteredItems.filter((item: any) => {\n        try {\n          const fieldValue = item[dateField];\n\n          if (!fieldValue) {\n            return false;\n          }\n\n          let itemDate: Date;\n          if (typeof fieldValue.toDate === 'function') {\n            itemDate = fieldValue.toDate();\n          } else if (fieldValue instanceof Date) {\n            itemDate = fieldValue;\n          } else if (typeof fieldValue === 'string') {\n            itemDate = new Date(fieldValue);\n            if (isNaN(itemDate.getTime())) {\n              return false;\n            }\n          } else if (typeof fieldValue === 'number') {\n            itemDate = new Date(fieldValue);\n          } else {\n            return false;\n          }\n\n          return intervals.some(\n            (interval) =>\n              itemDate >= interval.initial && itemDate <= interval.final\n          );\n        } catch (error) {\n          console.warn(\n            'Erro ao processar filtro de data para o item:',\n            item.id,\n            error\n          );\n          return false;\n        }\n      });\n    });\n\n    otherFilters.forEach(({ group, selectedFilter, arrange }) => {\n      if (arrange === 'filter') {\n        const filterValue = group.get('typeFilter')?.value;\n        if (filterValue && filterValue.trim()) {\n          const property = selectedFilter.property;\n          filteredItems = filteredItems.filter((item: any) => {\n            const value = String(item[property] || '')\n              .trim()\n              .toLocaleLowerCase();\n            return value.includes(filterValue.trim().toLocaleLowerCase());\n          });\n        }\n      } else if (\n        selectedFilter.hasOwnProperty('items') &&\n        arrange === 'equals'\n      ) {\n        const selectedItems = group.get('selectItem')?.value;\n        if (Array.isArray(selectedItems) && selectedItems.length > 0) {\n          const values = selectedItems.map((item: any) => item.value);\n          filteredItems = filteredItems.filter((item: any) => {\n            return selectedItems.some((selectedItem: any) => {\n              const itemValue = item[selectedItem.property];\n              return itemValue === selectedItem.value;\n            });\n          });\n        }\n      }\n    });\n\n    return filteredItems;\n  }\n\n  // Handler para mudanças de data no filtro\n  onDateFilterChange(): void {\n    if (this.data.pagination === false) {\n      // Atualizar arrange com os filtros ativos\n      this.arrange = this.buildArrangeFromFilters();\n      // Aplicar filtros (ou remover se campos estiverem vazios)\n      this.applyFiltersToDataSource();\n    }\n  }\n\n  // Aplicar filtros ao dataSource quando não há paginação\n  private applyFiltersToDataSource(): void {\n    if (!this.dataSource) return;\n\n    let filteredItems = this.applyClientSideFilters([...this.items]);\n    this.dataSource.data = filteredItems;\n    // Atualizar filteredItems com os dados filtrados (será atualizado novamente no handleDownload com filteredData)\n    this.filteredItems = filteredItems;\n    this.totalItems = filteredItems.length;\n  }\n\n  // Métodos com paginação\n  private async loadItemsPaginated(\n    navigation: string = 'reload',\n    reset: boolean = false\n  ) {\n    if (reset && ['forward', 'reload'].includes(navigation)) {\n      this.lastDoc = null;\n      this.currentClientPageIndex = 0;\n    }\n    if (reset && ['backward', 'reload'].includes(navigation)) {\n      this.firstDoc = null;\n      this.currentClientPageIndex = 0;\n    }\n\n    const activeFilters = this.filtersForm.controls\n      .flatMap((control) => {\n        const group = control as FormGroup;\n        const selectedFilter = group.get('selectFilter')?.value;\n        if (!selectedFilter) return [];\n\n        const arrange = selectedFilter.arrange;\n\n        if (arrange === 'filter') {\n          const filterValue = group.get('typeFilter')?.value;\n          if (!filterValue) return [];\n          return {\n            arrange,\n            filter: {\n              property: selectedFilter.property,\n              filtering: filterValue,\n            },\n            dateFilter: undefined,\n          };\n        }\n\n        if (arrange === 'filterByDate') {\n          const initial = group.get('initialDate')?.value;\n          const final = group.get('finalDate')?.value;\n          if (initial && final) {\n            const [dayI, monthI, yearI] = initial.split('/');\n            const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\n            const [dayF, monthF, yearF] = final.split('/');\n            const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\n            finalDate.setHours(23, 59, 59);\n            return {\n              arrange,\n              filter: undefined,\n              dateFilter: {\n                initial: initialDate,\n                final: finalDate,\n              },\n            };\n          }\n          return [];\n        }\n\n        if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {\n          const selectedItems = group.get('selectItem')?.value;\n          if (Array.isArray(selectedItems) && selectedItems.length > 0) {\n            return selectedItems.map((item) => ({\n              arrange,\n              filter: {\n                property: item.property,\n                filtering: item.value,\n              },\n              dateFilter: undefined,\n            }));\n          }\n        }\n\n        return [];\n      })\n      .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));\n\n    this.arrange = {\n      filters: activeFilters,\n      sortBy: this.sortBy,\n    };\n\n    const paginated: any = {\n      batchSize: this.pageSize,\n      collection: this.data.collection,\n      doc: { lastDoc: this.lastDoc, firstDoc: this.firstDoc },\n      navigation,\n      arrange: this.arrange,\n      conditions: this.data.conditions,\n      size: this.totalItems,\n      filterFn: this.data.filterFn,\n      clientPageIndex: this.currentClientPageIndex,\n    };\n\n    const result = await this.tableService.getPaginated(paginated);\n    this.items = result.items;\n    await this.loadRelations();\n    await this.loadQueryLengths();\n    this.lastDoc = result.lastDoc as any;\n    this.firstDoc = result.firstDoc as any;\n\n    // Atualizar currentClientPageIndex se retornado pelo fallback\n    if (result.currentClientPageIndex !== undefined) {\n      this.currentClientPageIndex = result.currentClientPageIndex;\n    }\n\n    let sum = 0;\n    if (this.data.totalRef) {\n      for (const totalRef of this.data.totalRef) {\n        const totalRefDoc = await totalRef.ref.get();\n        const docData = totalRefDoc.data() as any;\n        if (docData || result.filterLength) {\n          sum =\n            result.filterLength ??\n            ((sum + (docData ? docData[totalRef.field] : 0)) as number);\n        }\n      }\n      this.totalItems = sum;\n    }\n\n    this.hasNextPage = result.hasNextPage;\n    this.dataSource = new MatTableDataSource(this.items);\n    this.filterPredicate = this.dataSource.filterPredicate;\n  }\n\n  async onPageChange(event?: PageEvent) {\n    if (this.data.pagination === true && event) {\n      this.isLoading = true;\n\n      const previousPageIndex = event.previousPageIndex ?? 0;\n      const pageIndex = event.pageIndex;\n      const currentComponentPageSize = this.pageSize;\n      const eventPageSize = event.pageSize;\n      const totalItems = this.totalItems;\n\n      let navigationDirection: string;\n      let resetDocs = false;\n      let originalPageSize: number | null = null;\n\n      const lastPageIndex = Math.max(\n        0,\n        Math.ceil(totalItems / eventPageSize) - 1\n      );\n\n      // Atualizar currentClientPageIndex sempre para o fallback\n      this.currentClientPageIndex = pageIndex;\n\n      if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {\n        this.currentPageNumber++;\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex < previousPageIndex\n      ) {\n        this.currentPageNumber = Math.max(1, this.currentPageNumber - 1);\n      }\n\n      if (eventPageSize !== currentComponentPageSize) {\n        console.log('Alterou a quantidade de elementos exibidos por página');\n        this.pageSize = eventPageSize;\n        navigationDirection = 'forward';\n        resetDocs = true;\n        this.currentClientPageIndex = 0;\n      } else if (\n        pageIndex === 0 &&\n        previousPageIndex !== undefined &&\n        pageIndex < previousPageIndex\n      ) {\n        console.log('Pulou para a primeira página');\n        navigationDirection = 'forward';\n        this.currentPageNumber = 1;\n        this.currentClientPageIndex = 0;\n        resetDocs = true;\n      } else if (\n        pageIndex === lastPageIndex &&\n        previousPageIndex !== undefined &&\n        pageIndex > previousPageIndex &&\n        pageIndex - previousPageIndex > 1\n      ) {\n        console.log('Pulou para a ultima página');\n        navigationDirection = 'backward';\n        resetDocs = true;\n\n        const itemsExpectedInLastPage =\n          totalItems - lastPageIndex * eventPageSize;\n\n        if (\n          itemsExpectedInLastPage > 0 &&\n          itemsExpectedInLastPage < eventPageSize\n        ) {\n          originalPageSize = this.pageSize;\n          this.pageSize = itemsExpectedInLastPage;\n        }\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex > previousPageIndex\n      ) {\n        console.log('Procedendo');\n        navigationDirection = 'forward';\n        resetDocs = false;\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex < previousPageIndex\n      ) {\n        console.log('Retrocedendo.');\n        navigationDirection = 'backward';\n        resetDocs = false;\n      } else if (\n        previousPageIndex !== undefined &&\n        pageIndex === previousPageIndex\n      ) {\n        console.log('Recarregando.');\n        navigationDirection = 'reload';\n        resetDocs = false;\n      } else if (previousPageIndex === undefined && pageIndex === 0) {\n        console.log(\n          'Evento inicial do paginador para pág 0. ngOnInit carregou.'\n        );\n        this.isLoading = false;\n        if (event) this.pageEvent = event;\n        return;\n      } else {\n        console.warn('INESPERADO! Condição de navegação não tratada:', event);\n        this.isLoading = false;\n        if (event) this.pageEvent = event;\n        return;\n      }\n\n      if (navigationDirection) {\n        try {\n          await this.loadItemsPaginated(navigationDirection, resetDocs);\n        } catch (error) {\n          console.error('Erro ao carregar itens paginados:', error);\n        } finally {\n          if (originalPageSize !== null) {\n            this.pageSize = originalPageSize;\n          }\n        }\n      }\n\n      if (event) this.pageEvent = event;\n      this.isLoading = false;\n    }\n  }\n\n  // Outros métodos\n\n  applyFilter(value: string) {\n    // Sem paginação\n    if (this.data.pagination === false) {\n      this.dataSource.filter = String(value).trim().toLowerCase();\n    }\n    // Com paginação\n    if (this.data.pagination === true) {\n      this.filterValue = value;\n      this.filterSubject.next(this.filterValue);\n    }\n  }\n\n  goToDetails(row: any): void {\n    if (this.data.isNotClickable) {\n      return;\n    }\n    const urlPath = this.data.url || this.data.name;\n    const url = this.router.serializeUrl(\n      this.router.createUrlTree([`/${urlPath}`, row.id])\n    );\n    window.open(url, '_blank');\n  }\n\n  async getRelation(params: {\n    id: any;\n    collection: string;\n    newProperty: string;\n  }): Promise<string> {\n    try {\n      let snapshot: firestore.DocumentSnapshot<unknown> | undefined;\n      if (\n        params.id !== '' &&\n        params.id !== undefined &&\n        params.collection !== undefined &&\n        params.collection !== ''\n      ) {\n        snapshot = await firstValueFrom(\n          this.firestore.collection(params.collection).doc(params.id).get()\n        );\n      }\n      if (snapshot && snapshot.exists) {\n        const data = snapshot.data() as any;\n        return data?.[params.newProperty] ?? '';\n      }\n      return '';\n    } catch (e) {\n      console.log(e);\n      return '';\n    }\n  }\n\n  private async loadRelations() {\n    const relationPromises = this.data.displayedColumns\n      .filter((col) => col.relation)\n      .flatMap((col) =>\n        this.items.map(async (item) => {\n          if (col.relation) {\n            item[col.property] = await this.getRelation({\n              id: item[col.relation.property],\n              collection: col.relation.collection,\n              newProperty: col.relation.newProperty,\n            });\n          }\n        })\n      );\n\n    await Promise.all(relationPromises);\n  }\n\n  async getQueryLength(params: {\n    item: any;\n    relation: {\n      collection: string;\n      property: string;\n      operator: WhereFilterOp;\n      value: string;\n    };\n  }) {\n    const snapshot = await this.firestore\n      .collection(params.relation.collection)\n      .ref.where(\n        params.relation.property,\n        params.relation.operator,\n        params.item[params.relation.value]\n      )\n      .get();\n    return snapshot.size;\n  }\n\n  private async loadQueryLengths() {\n    const lengthPromises = this.data.displayedColumns\n      .filter((col) => col.queryLength)\n      .flatMap((col) =>\n        this.items.map(async (item) => {\n          if (col.queryLength) {\n            item[col.property] = await this.getQueryLength({\n              item: item,\n              relation: col.queryLength,\n            });\n          }\n        })\n      );\n    await Promise.all(lengthPromises);\n  }\n\n  private filterItems() {\n    if (this.data.conditions) {\n      this.data.conditions.forEach((cond) => {\n        this.items = this.items.filter((item) => {\n          const operatorFunction = (this.tableService.operators as any)[\n            cond.operator\n          ];\n          if (operatorFunction) {\n            return operatorFunction(\n              item[cond.firestoreProperty],\n              cond.dashProperty\n            );\n          }\n          return false;\n        });\n      });\n    }\n  }\n\n  private buildArrangeFromFilters(): Arrange {\n    const activeFilters = this.filtersForm.controls\n      .flatMap((control) => {\n        const group = control as FormGroup;\n        const selectedFilter = group.get('selectFilter')?.value;\n        if (!selectedFilter) return [];\n\n        const arrange = selectedFilter.arrange;\n\n        if (arrange === 'filter') {\n          const filterValue = group.get('typeFilter')?.value;\n          if (!filterValue) return [];\n          return {\n            arrange,\n            filter: {\n              property: selectedFilter.property,\n              filtering: filterValue,\n            },\n            dateFilter: undefined,\n          };\n        }\n\n        if (arrange === 'filterByDate') {\n          const initial = group.get('initialDate')?.value;\n          const final = group.get('finalDate')?.value;\n          if (initial && final) {\n            try {\n              const [dayI, monthI, yearI] = initial.split('/');\n              const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\n              const [dayF, monthF, yearF] = final.split('/');\n              const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\n              finalDate.setHours(23, 59, 59);\n              return {\n                arrange,\n                filter: undefined,\n                dateFilter: {\n                  initial: initialDate,\n                  final: finalDate,\n                },\n              };\n            } catch (error) {\n              return [];\n            }\n          }\n          return [];\n        }\n\n        if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {\n          const selectedItems = group.get('selectItem')?.value;\n          if (Array.isArray(selectedItems) && selectedItems.length > 0) {\n            return selectedItems.map((item) => ({\n              arrange,\n              filter: {\n                property: item.property,\n                filtering: item.value,\n              },\n              dateFilter: undefined,\n            }));\n          }\n        }\n\n        return [];\n      })\n      .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));\n\n    return {\n      filters: activeFilters,\n      sortBy: this.sortBy,\n    };\n  }\n\n  // Filtro de data\n  async search(event?: Event) {\n    // Prevenir comportamento padrão do formulário ao pressionar Enter\n    if (event) {\n      event.preventDefault();\n      event.stopPropagation();\n    }\n\n    if (this.selectSort.value) {\n      if (this.selectSort.value.arrange === 'ascending') {\n        this.sortBy = {\n          field: this.selectSort.value.property,\n          order: 'asc',\n        };\n      }\n      if (this.selectSort.value.arrange === 'descending') {\n        this.sortBy = {\n          field: this.selectSort.value.property,\n          order: 'desc',\n        };\n      }\n    }\n\n    // Sem paginação: aplicar filtros client-side\n    if (this.data.pagination === false) {\n      // Atualizar arrange com os filtros ativos\n      this.arrange = this.buildArrangeFromFilters();\n      this.applyFiltersToDataSource();\n      this.currentArrange =\n        this.filtersForm.length > 0\n          ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''\n          : '';\n    } else {\n      // Com paginação: comportamento original\n      await this.loadItemsPaginated('reload', true);\n      this.currentArrange =\n        this.filtersForm.length > 0\n          ? this.filtersForm.at(0).get('selectFilter')?.value?.arrange ?? ''\n          : '';\n      this.paginator.firstPage();\n    }\n  }\n\n  async resetFilter() {\n    this.dataSource.filter = '';\n    if (this.filterPredicate) {\n      this.dataSource.filterPredicate = this.filterPredicate;\n    }\n\n    this.filtersForm.clear();\n    this.addFilter();\n    this.selectSort.patchValue('');\n    this.sortBy = {\n      order: this.data.sortBy ? this.data.sortBy.order : 'desc',\n      field: this.data.sortBy ? this.data.sortBy.field : 'createdAt',\n    };\n\n    // Sem paginação: recarregar itens originais\n    if (this.data.pagination === false) {\n      // Resetar arrange para valores padrão\n      this.arrange = {\n        filters: [],\n        sortBy: this.sortBy,\n      };\n      this.dataSource.data = [...this.items];\n      this.totalItems = this.items.length;\n      this.currentArrange = '';\n    } else {\n      // Com paginação: comportamento original\n      await this.loadItemsPaginated('reload', true);\n      this.currentArrange = '';\n      this.paginator.firstPage();\n    }\n  }\n\n  // Método público para recarregar a tabela\n  async reloadTable() {\n    if (this.data.pagination) {\n      await this.loadItemsPaginated('reload', true);\n      this.paginator.firstPage();\n    } else {\n      await this.loadItems();\n    }\n  }\n\n  updateDisplayedColumns() {\n    if (this.dataSource) {\n      this.dataSource = new MatTableDataSource<any>([]);\n    }\n\n    this.columnProperties = this.data.displayedColumns.map((column) => {\n      return column.property;\n    });\n\n    this.dropdownItems = [];\n    this.sortableDropdownItems = [];\n\n    this.data.displayedColumns.forEach((col) => {\n      if (col.isFilterable) {\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filter',\n          title: col.title,\n        });\n      }\n      if (col.isSortable) {\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'ascending',\n          title: col.title + ': crescente',\n        });\n        this.sortableDropdownItems.push({\n          ...col,\n          arrange: 'descending',\n          title: col.title + ': decrescente',\n        });\n      }\n      if (col.isFilterableByDate) {\n        this.dropdownItems.push({\n          ...col,\n          arrange: 'filterByDate',\n          title: col.title + ': filtro por data',\n        });\n      }\n    });\n\n    if (\n      this.data.filterableOptions &&\n      Array.isArray(this.data.filterableOptions)\n    ) {\n      this.data.filterableOptions.forEach((option) =>\n        this.dropdownItems.push({ ...option, arrange: 'equals' })\n      );\n    }\n  }\n\n  isString(value: any): boolean {\n    return typeof value === 'string';\n  }\n\n  // Métodos para controle do tooltip\n  onCellMouseEnter(event: MouseEvent, row: any, col: any): void {\n    // Só mostrar tooltip se a coluna tiver charLimit definido\n    if (!col.charLimit) {\n      return;\n    }\n\n    const fullValue = this.getDisplayValue(col, row, true);\n\n    // Só mostrar tooltip se o valor completo for maior que o limite\n    if (fullValue.length <= col.charLimit) {\n      return;\n    }\n\n    this.hoveredCell = { row, col };\n    this.tooltipContent = fullValue;\n\n    // Definir posição do tooltip\n    this.tooltipPosition = {\n      x: event.clientX + 10,\n      y: event.clientY - 10,\n    };\n\n    // Timeout para mostrar o tooltip\n    this.tooltipTimeout = setTimeout(() => {\n      if (\n        this.hoveredCell &&\n        this.hoveredCell.row === row &&\n        this.hoveredCell.col === col\n      ) {\n        this.showTooltip = true;\n      }\n    }, 500);\n  }\n\n  onCellMouseLeave(): void {\n    if (this.tooltipTimeout) {\n      clearTimeout(this.tooltipTimeout);\n      this.tooltipTimeout = null;\n    }\n\n    this.showTooltip = false;\n    this.hoveredCell = null;\n    this.tooltipContent = '';\n  }\n\n  onCellMouseMove(event: MouseEvent): void {\n    if (this.showTooltip) {\n      this.tooltipPosition = {\n        x: event.clientX + 10,\n        y: event.clientY - 10,\n      };\n    }\n  }\n\n  // Métodos para inversão vertical dos tabs\n  getTabGroups(tabs: any[]): number[] {\n    if (!tabs || tabs.length === 0) return [];\n\n    const totalGroups = Math.ceil(tabs.length / 6);\n    const groups: number[] = [];\n\n    // Criar array de índices invertidos (último grupo primeiro)\n    for (let i = totalGroups - 1; i >= 0; i--) {\n      groups.push(i);\n    }\n\n    return groups;\n  }\n\n  getTabGroup(tabs: any[], groupIndex: number): any[] {\n    if (!tabs || tabs.length === 0) return [];\n\n    const startIndex = groupIndex * 6;\n    const endIndex = Math.min(startIndex + 6, tabs.length);\n\n    return tabs.slice(startIndex, endIndex);\n  }\n\n  getRealTabIndex(groupIndex: number, tabIndexInGroup: number): number {\n    if (!this.data.tabs?.tabsData) return 0;\n    const totalGroups = Math.ceil(this.data.tabs.tabsData.length / 6);\n    const realGroupIndex = totalGroups - 1 - groupIndex;\n    return realGroupIndex * 6 + tabIndexInGroup;\n  }\n\n  onTableSelected(i: number, j: number) {\n    if (!this.data.tabs?.tabsData || !this.data.tabs.method) return;\n    this.selectedTab = this.getRealTabIndex(i, j);\n    const tab = this.data.tabs.tabsData[this.selectedTab];\n    if (tab) {\n      this.data.tabs.method(tab, this.selectedTab);\n    }\n  }\n\n  isTabSelected(originalIndex: number): boolean {\n    return this.selectedTab === originalIndex;\n  }\n\n  shouldShowActionButton(): boolean {\n    if (!this.data?.actionButton) {\n      return false;\n    }\n\n    if (!this.data.actionButton.condition) {\n      return true;\n    }\n\n    try {\n      return this.data.actionButton.condition(null) ?? true;\n    } catch {\n      return true;\n    }\n  }\n\n  // Método para lidar com download (diferente para paginado e não paginado)\n  handleDownload(): void {\n    if (!this.downloadTable) return;\n\n    // Se não há paginação, usar dados filtrados do dataSource\n    if (this.data.pagination === false) {\n      // Atualizar filteredItems com os dados filtrados do dataSource\n      // (que já inclui filtro de texto aplicado via filterPredicate)\n      if (this.dataSource && this.dataSource.filteredData) {\n        this.filteredItems = [...this.dataSource.filteredData];\n      }\n\n      // Construir arrange com os filtros ativos (se houver)\n      const arrange = this.buildArrangeFromFilters();\n      this.downloadTable(arrange, this.data.conditions || []);\n    } else {\n      // Modo paginado: usar arrange existente (comportamento original)\n      if (this.arrange) {\n        this.downloadTable(this.arrange, this.data.conditions || []);\n      }\n    }\n  }\n}\n","<div *ngIf=\"data\" class=\"card-body\">\n  <div class=\"flex flex-col justify-between gap-6\">\n    <!-- UNIFIED CONTROL PANEL: FILTERS, SORT & ACTIONS -->\n    <div\n      class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n      *ngIf=\"\n        data.pagination === true &&\n        (dropdownItems.length > 0 ||\n          sortableDropdownItems.length > 0 ||\n          data.actionButton)\n      \"\n    >\n      <!-- PANEL HEADER: Title, Custom Action, and Global Actions -->\n      <div\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\"\n      >\n        <!-- Left Side: Title & Main Action Button -->\n        <div class=\"flex flex-wrap items-center gap-4\">\n          <div class=\"flex items-center gap-2\">\n            <i class=\"fa fa-filter text-xl text-blue-500\"></i>\n            <span class=\"text-lg font-semibold text-gray-700\"\n              >Filtros e Ações</span\n            >\n          </div>\n          <button\n            *ngIf=\"data.actionButton && data.actionButton.condition\"\n            [ngClass]=\"\n              (data.actionButton.colorClass || 'bg-blue-500') +\n              ' flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\n            \"\n            [routerLink]=\"data.actionButton.routerLink\"\n            (click)=\"\n              data.actionButton.method ? data.actionButton.method($event) : null\n            \"\n          >\n            <i\n              *ngIf=\"data.actionButton.icon\"\n              [class]=\"data.actionButton.icon\"\n            ></i>\n            {{ data.actionButton.label }}\n          </button>\n        </div>\n\n        <!-- Right Side: Search, Reset, Export -->\n        <div\n          class=\"flex flex-wrap gap-3\"\n          *ngIf=\"\n            this.hasFilterableColumn === true || this.hasSortableColumn === true\n          \"\n        >\n          <button\n            (click)=\"search()\"\n            type=\"button\"\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\"\n            matTooltip=\"Aplicar filtros\"\n          >\n            <i class=\"fa fa-search\"></i>\n            Pesquisar\n          </button>\n\n          <button\n            (click)=\"resetFilter()\"\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\"\n            matTooltip=\"Limpar filtros\"\n          >\n            <i class=\"fas fa-redo-alt\"></i>\n            Resetar\n          </button>\n\n          <button\n            *ngIf=\"data.download !== false && downloadTable\"\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\"\n            matTooltipPosition=\"above\"\n            matTooltip=\"Exportar Tabela\"\n            [disabled]=\"\n              this.dataSource && this.dataSource.filteredData.length <= 0\n            \"\n            (click)=\"\n              $any(arrange) && downloadTable !== undefined\n                ? downloadTable($any(arrange), data.conditions || [])\n                : null\n            \"\n          >\n            <i class=\"fa fa-download\"></i>\n            Exportar\n          </button>\n        </div>\n      </div>\n\n      <!-- FILTERS CONTENT (WITH REFINEMENTS) -->\n      <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\n        <div\n          [formGroup]=\"$any(filterGroup)\"\n          *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\n          class=\"flex flex-wrap items-center gap-3 rounded-lg border border-gray-200 p-2\"\n        >\n          <!-- FILTER TYPE SELECTOR -->\n          <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\n              <mat-label>Tipo de filtro</mat-label>\n              <mat-select\n                placeholder=\"Selecione o tipo...\"\n                formControlName=\"selectFilter\"\n                (selectionChange)=\"onSelectFilterChange()\"\n              >\n                <mat-option *ngFor=\"let item of getAvailableFilterOptions()\" [value]=\"item\">\n                  <div class=\"flex items-center gap-2\">\n                    <i\n                      [class]=\"item.icon || 'fa fa-filter'\"\n                      class=\"text-sm text-blue-500\"\n                    ></i>\n                    <span>{{ item.title }}</span>\n                  </div>\n                </mat-option>\n              </mat-select>\n            </mat-form-field>\n          </div>\n\n          <!-- TEXT FILTER -->\n          <div\n            class=\"min-w-[200px] flex-1\"\n            *ngIf=\"\n              $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\n            \"\n          >\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\n              <mat-label class=\"flex items-center gap-2\">\n                <i class=\"fa fa-search text-gray-400\"></i>\n                <span>{{\n                  $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n                    \"Filtrar\"\n                }}</span>\n              </mat-label>\n              <input\n                (keyup.enter)=\"search($event)\"\n                formControlName=\"typeFilter\"\n                matInput\n                placeholder=\"Digite para filtrar...\"\n                #input\n              />\n            </mat-form-field>\n          </div>\n\n          <!-- DROPDOWN FILTER -->\n          <div\n            class=\"min-w-[200px] flex-1\"\n            *ngIf=\"\n              $any(filterGroup).get('selectFilter')?.value &&\n              $any(filterGroup)\n                .get('selectFilter')\n                ?.value.hasOwnProperty('items')\n            \"\n          >\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\n              <mat-label>{{\n                $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n                  \"Selecione\"\n              }}</mat-label>\n              <mat-select\n                placeholder=\"Selecione...\"\n                formControlName=\"selectItem\"\n                multiple\n              >\n                <mat-option\n                  *ngFor=\"\n                    let item of $any(filterGroup).get('selectFilter')?.value\n                      .items\n                  \"\n                  [value]=\"item\"\n                >\n                  {{ item.label }}\n                </mat-option>\n              </mat-select>\n            </mat-form-field>\n          </div>\n\n          <!-- DATE FILTER -->\n          <div\n            class=\"min-w-[340px] flex-auto\"\n            *ngIf=\"\n              $any(filterGroup).get('selectFilter')?.value?.arrange ===\n              'filterByDate'\n            \"\n          >\n            <div\n              class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\n            >\n              <mat-form-field appearance=\"outline\" class=\"flex-1\">\n                <mat-label class=\"flex items-center gap-2\">\n                  <i class=\"fa fa-calendar text-gray-400\"></i>\n                  <span>Data Inicial</span>\n                </mat-label>\n                <input\n                  matInput\n                  (keyup.enter)=\"search($event)\"\n                  formControlName=\"initialDate\"\n                  [dropSpecialCharacters]=\"false\"\n                  mask=\"d0/M0/0000\"\n                  placeholder=\"DD/MM/AAAA\"\n                  maxlength=\"10\"\n                />\n              </mat-form-field>\n\n              <mat-form-field appearance=\"outline\" class=\"flex-1\">\n                <mat-label class=\"flex items-center gap-2\">\n                  <i class=\"fa fa-calendar text-gray-400\"></i>\n                  <span>Data Final</span>\n                </mat-label>\n                <input\n                  (keyup.enter)=\"search($event)\"\n                  matInput\n                  formControlName=\"finalDate\"\n                  [dropSpecialCharacters]=\"false\"\n                  mask=\"d0/M0/0000\"\n                  placeholder=\"DD/MM/AAAA\"\n                  maxlength=\"10\"\n                />\n              </mat-form-field>\n            </div>\n          </div>\n\n          <!-- REMOVE FILTER BUTTON -->\n          <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\n            <button\n              (click)=\"removeFilter(i)\"\n              class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\n              matTooltip=\"Remover filtro\"\n            >\n              <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\n            </button>\n          </div>\n        </div>\n      </div>\n\n      <!-- PANEL FOOTER: Add Filter & Sort -->\n      <div\n        class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\n      >\n        <!-- Add Filter Button -->\n        <div *ngIf=\"dropdownItems.length > 0\">\n          <button\n            (click)=\"addFilter()\"\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\"\n            matTooltip=\"Adicionar novo filtro\"\n          >\n            <i class=\"fa fa-plus mr-2\"></i>\n            Adicionar Filtro\n          </button>\n        </div>\n\n        <!-- Sort Dropdown -->\n        <div\n          class=\"w-full sm:w-auto sm:min-w-[250px]\"\n          *ngIf=\"sortableDropdownItems.length > 0\"\n        >\n          <mat-form-field appearance=\"outline\" class=\"w-full\">\n            <mat-label>Ordenar por</mat-label>\n            <mat-select placeholder=\"Selecione...\" [formControl]=\"selectSort\">\n              <mat-option\n                *ngFor=\"let item of sortableDropdownItems\"\n                [value]=\"item\"\n              >\n                <div class=\"flex items-center gap-2\">\n                  <i class=\"fa fa-sort-alpha-down text-cyan-600\"></i>\n                  <span>{{ item.title }}</span>\n                </div>\n              </mat-option>\n            </mat-select>\n          </mat-form-field>\n        </div>\n      </div>\n    </div>\n\n    <!-- SIMPLE SEARCH (for non-paginated tables) -->\n    <div\n      class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n      *ngIf=\"data.pagination === false && hasFilterableColumn === true\"\n    >\n      <mat-form-field appearance=\"outline\" class=\"w-full\">\n        <mat-label class=\"flex items-center gap-2\">\n          <i class=\"fa fa-search text-blue-500\"></i>\n          Buscar\n        </mat-label>\n        <input\n          matInput\n          (keyup.enter)=\"search($event)\"\n          (keyup)=\"applyFilter(filterInput.value)\"\n          placeholder=\"Digite para filtrar...\"\n          #filterInput\n        />\n        <mat-icon matSuffix class=\"text-gray-500\">search</mat-icon>\n      </mat-form-field>\n      <button\n        *ngIf=\"data.actionButton\"\n        [ngClass]=\"\n          (data.actionButton.colorClass || 'bg-blue-500') +\n          ' float-right flex items-center gap-2 rounded-lg px-4 py-2 text-sm font-medium text-white hover:opacity-70'\n        \"\n        [routerLink]=\"data.actionButton.routerLink\"\n        (click)=\"\n          data.actionButton.method ? data.actionButton.method($event) : null\n        \"\n      >\n        <i *ngIf=\"data.actionButton.icon\" [class]=\"data.actionButton.icon\"></i>\n        {{ data.actionButton.label }}\n      </button>\n    </div>\n\n    <!-- FILTERS PANEL (for non-paginated tables) -->\n    <div\n      class=\"rounded-xl border border-gray-200 bg-white p-4 shadow-lg\"\n      *ngIf=\"data.pagination === false && dropdownItems.length > 0\"\n    >\n      <!-- FILTERS CONTENT -->\n      <div class=\"mb-4 space-y-3\" *ngIf=\"filtersForm.controls.length > 0\">\n        <div\n          [formGroup]=\"$any(filterGroup)\"\n          *ngFor=\"let filterGroup of filtersForm.controls; let i = index\"\n          class=\"flex flex-wrap items-center gap-3 rounded-lg border border-gray-200 p-2\"\n        >\n          <!-- FILTER TYPE SELECTOR -->\n          <div class=\"min-w-[200px] flex-1\" *ngIf=\"dropdownItems.length > 0\">\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\n              <mat-label>Tipo de filtro</mat-label>\n              <mat-select\n                placeholder=\"Selecione o tipo...\"\n                formControlName=\"selectFilter\"\n                (selectionChange)=\"onSelectFilterChange()\"\n              >\n                <mat-option *ngFor=\"let item of getAvailableFilterOptions()\" [value]=\"item\">\n                  <div class=\"flex items-center gap-2\">\n                    <i\n                      [class]=\"item.icon || 'fa fa-filter'\"\n                      class=\"text-sm text-blue-500\"\n                    ></i>\n                    <span>{{ item.title }}</span>\n                  </div>\n                </mat-option>\n              </mat-select>\n            </mat-form-field>\n          </div>\n\n          <!-- TEXT FILTER -->\n          <div\n            class=\"min-w-[200px] flex-1\"\n            *ngIf=\"\n              $any(filterGroup).get('selectFilter')?.value?.arrange === 'filter'\n            \"\n          >\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\n              <mat-label class=\"flex items-center gap-2\">\n                <i class=\"fa fa-search text-gray-400\"></i>\n                <span>{{\n                  $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n                    \"Filtrar\"\n                }}</span>\n              </mat-label>\n              <input\n                (keyup.enter)=\"search($event)\"\n                formControlName=\"typeFilter\"\n                matInput\n                placeholder=\"Digite para filtrar...\"\n                #input\n              />\n            </mat-form-field>\n          </div>\n\n          <!-- DROPDOWN FILTER -->\n          <div\n            class=\"min-w-[200px] flex-1\"\n            *ngIf=\"\n              $any(filterGroup).get('selectFilter')?.value &&\n              $any(filterGroup)\n                .get('selectFilter')\n                ?.value.hasOwnProperty('items')\n            \"\n          >\n            <mat-form-field appearance=\"outline\" class=\"w-full\">\n              <mat-label>{{\n                $any(filterGroup).get(\"selectFilter\")?.value?.title ||\n                  \"Selecione\"\n              }}</mat-label>\n              <mat-select\n                placeholder=\"Selecione...\"\n                formControlName=\"selectItem\"\n                multiple\n              >\n                <mat-option\n                  *ngFor=\"\n                    let item of $any(filterGroup).get('selectFilter')?.value\n                      .items\n                  \"\n                  [value]=\"item\"\n                >\n                  {{ item.label }}\n                </mat-option>\n              </mat-select>\n            </mat-form-field>\n          </div>\n\n          <!-- DATE FILTER -->\n          <div\n            class=\"min-w-[340px] flex-auto\"\n            *ngIf=\"\n              $any(filterGroup).get('selectFilter')?.value?.arrange ===\n              'filterByDate'\n            \"\n          >\n            <div\n              class=\"flex flex-col items-stretch gap-3 sm:flex-row sm:items-center\"\n            >\n              <mat-form-field appearance=\"outline\" class=\"flex-1\">\n                <mat-label class=\"flex items-center gap-2\">\n                  <i class=\"fa fa-calendar text-gray-400\"></i>\n                  <span>Data Inicial</span>\n                </mat-label>\n                <input\n                  matInput\n                  (keyup.enter)=\"search($event)\"\n                  (blur)=\"onDateFilterChange()\"\n                  formControlName=\"initialDate\"\n                  [dropSpecialCharacters]=\"false\"\n                  mask=\"d0/M0/0000\"\n                  placeholder=\"DD/MM/AAAA\"\n                  maxlength=\"10\"\n                />\n              </mat-form-field>\n\n              <mat-form-field appearance=\"outline\" class=\"flex-1\">\n                <mat-label class=\"flex items-center gap-2\">\n                  <i class=\"fa fa-calendar text-gray-400\"></i>\n                  <span>Data Final</span>\n                </mat-label>\n                <input\n                  (keyup.enter)=\"search($event)\"\n                  (blur)=\"onDateFilterChange()\"\n                  matInput\n                  formControlName=\"finalDate\"\n                  [dropSpecialCharacters]=\"false\"\n                  mask=\"d0/M0/0000\"\n                  placeholder=\"DD/MM/AAAA\"\n                  maxlength=\"10\"\n                />\n              </mat-form-field>\n            </div>\n          </div>\n\n          <!-- REMOVE FILTER BUTTON -->\n          <div *ngIf=\"filtersForm.length > 1\" class=\"ml-auto flex-shrink-0\">\n            <button\n              (click)=\"removeFilter(i)\"\n              class=\"flex h-10 w-10 items-center justify-center rounded-full transition-colors duration-300 hover:bg-red-100\"\n              matTooltip=\"Remover filtro\"\n            >\n              <i class=\"fa fa-trash text-red-500 hover:text-red-600\"></i>\n            </button>\n          </div>\n        </div>\n      </div>\n\n      <!-- PANEL FOOTER: Add Filter & Actions -->\n      <div\n        class=\"-mb-2 flex flex-col items-center justify-between gap-4 border-t border-gray-200 pt-4 sm:flex-row\"\n      >\n        <!-- Add Filter Button -->\n        <div *ngIf=\"dropdownItems.length > 0\">\n          <button\n            (click)=\"addFilter()\"\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\"\n            matTooltip=\"Adicionar novo filtro\"\n          >\n            <i class=\"fa fa-plus mr-2\"></i>\n            Adicionar Filtro\n          </button>\n        </div>\n\n        <!-- Action Buttons -->\n        <div class=\"flex flex-wrap gap-3\">\n          <button\n            (click)=\"search()\"\n            type=\"button\"\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\"\n            matTooltip=\"Aplicar filtros\"\n          >\n            <i class=\"fa fa-search\"></i>\n            Aplicar\n          </button>\n\n          <button\n            (click)=\"resetFilter()\"\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\"\n            matTooltip=\"Limpar filtros\"\n          >\n            <i class=\"fas fa-redo-alt\"></i>\n            Resetar\n          </button>\n\n          <button\n            *ngIf=\"data.download !== false && downloadTable\"\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\"\n            matTooltipPosition=\"above\"\n            matTooltip=\"Exportar Tabela\"\n            [disabled]=\"\n              this.dataSource && this.dataSource.filteredData.length <= 0\n            \"\n            (click)=\"handleDownload()\"\n          >\n            <i class=\"fa fa-download\"></i>\n            Exportar\n          </button>\n        </div>\n      </div>\n    </div>\n\n    <div class=\"flex flex-col\">\n      <div\n        class=\"mx-auto flex flex-col\"\n        *ngIf=\"data.tabs && data.tabs.tabsData && data.tabs.tabsData.length > 0\"\n      >\n        <!-- Calcular quantos grupos de 6 tabs existem -->\n        <ng-container\n          *ngFor=\"\n            let groupIndex of getTabGroups(data.tabs.tabsData);\n            let i = index\n          \"\n        >\n          <div class=\"mx-auto flex flex-row\">\n            <ng-container\n              *ngFor=\"\n                let tab of getTabGroup(data.tabs.tabsData, groupIndex);\n                let j = index\n              \"\n            >\n              <button\n                class=\"border-2 border-gray-300 bg-gray-200 px-4 py-2 font-medium transition hover:brightness-95\"\n                [ngClass]=\"\n                  isTabSelected(getRealTabIndex(i, j))\n                    ? 'border-b-0 brightness-110'\n                    : ''\n                \"\n                (click)=\"onTableSelected(i, j)\"\n              >\n                {{ tab.label }}\n                <span\n                  *ngIf=\"tab.counter !== undefined\"\n                  class=\"ml-2 text-xs font-bold\"\n                  [ngClass]=\"tab.counterClass\"\n                >\n                  {{ tab.counter }}\n                </span>\n              </button>\n            </ng-container>\n          </div>\n        </ng-container>\n      </div>\n      <div class=\"mat-elevation-z8 w-full overflow-x-auto rounded-xl\">\n        <table\n          mat-table\n          [dataSource]=\"dataSource\"\n          matSort\n          #sort=\"matSort\"\n          matSortActive=\"createdAt\"\n          matSortDirection=\"desc\"\n        >\n          <ng-container\n            *ngFor=\"let col of data.displayedColumns\"\n            matColumnDef=\"{{ col.property }}\"\n          >\n            <ng-container *matHeaderCellDef>\n              <!-- IF THE COLUMN IS NOT SORTABLE, THEN DON'T SHOW THE SORT BUTTONS -->\n              <th\n                *ngIf=\"!col.isSortable || data.pagination === true\"\n                mat-header-cell\n                [ngClass]=\"\n                  (data.color?.bg ? ' ' + $any(data.color).bg : '') +\n                  (data.color?.text ? ' ' + $any(data.color).text : '')\n                \"\n              >\n                {{ col.title }}\n              </th>\n              <!-- IF THE COLUMN IS SORTABLE, THEN SHOW THE SORT BUTTONS -->\n              <th\n                *ngIf=\"col.isSortable && data.pagination === false\"\n                mat-header-cell\n                mat-sort-header\n                [ngClass]=\"\n                  (data.color?.bg ? ' ' + $any(data.color).bg : '') +\n                  (data.color?.text ? ' ' + $any(data.color).text : '')\n                \"\n              >\n                {{ col.title }}\n              </th>\n              <td\n                mat-cell\n                *matCellDef=\"let row\"\n                (click)=\"col.method ? col.method(row) : null\"\n                (mouseenter)=\"onCellMouseEnter($event, row, col)\"\n                (mouseleave)=\"onCellMouseLeave()\"\n                (mousemove)=\"onCellMouseMove($event)\"\n              >\n                <!-- CHECK IF THE COLUMN MUST BE DISPLAYED -->\n                <span *ngIf=\"!col.image && !col.iconClass && !col.method\">\n                  <ng-container>\n                    <span\n                      *ngIf=\"\n                        col.charLimit &&\n                          row[col.property] &&\n                          row[col.property].length > col.charLimit;\n                        else withinLimit\n                      \"\n                    >\n                      <a\n                        *ngIf=\"col.hasLink === true\"\n                        [href]=\"row[col.property]\"\n                        target=\"_blank\"\n                      >\n                        {{ getDisplayValue(col, row) }}\n                      </a>\n                      <a\n                        *ngIf=\"col.hasLink && isString(col.hasLink)\"\n                        [href]=\"col.hasLink\"\n                        target=\"_blank\"\n                      >\n                        {{ getDisplayValue(col, row) }}\n                      </a>\n                      <span\n                        *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\n                      >\n                        {{ getDisplayValue(col, row) }}\n                      </span>\n                    </span>\n                  </ng-container>\n                  <ng-template #withinLimit>\n                    <a\n                      *ngIf=\"col.hasLink === true\"\n                      [href]=\"row[col.property]\"\n                      target=\"_blank\"\n                    >\n                      {{ getDisplayValue(col, row, true) }}\n                    </a>\n                    <a\n                      *ngIf=\"col.hasLink && isString(col.hasLink)\"\n                      [href]=\"col.hasLink\"\n                      target=\"_blank\"\n                    >\n                      {{ getDisplayValue(col, row, true) }}\n                    </a>\n                    <span\n                      *ngIf=\"col.hasLink !== true && !isString(col.hasLink)\"\n                    >\n                      {{ getDisplayValue(col, row, true) }}\n                    </span>\n                  </ng-template>\n                </span>\n                <!------------------- IMAGE ------------------>\n                <img\n                  *ngIf=\"\n                    col.image && col.image.path && !col.iconClass && !col.method\n                  \"\n                  [src]=\"col.image.path + '/' + row[col.property]\"\n                  [ngClass]=\"col.image.class\"\n                  alt=\"Imagem\"\n                />\n                <img\n                  *ngIf=\"\n                    col.image && col.image.url && !col.iconClass && !col.method\n                  \"\n                  [src]=\"row[col.property]\"\n                  [ngClass]=\"col.image.class\"\n                  alt=\"Imagem\"\n                />\n                <ng-container *ngIf=\"col.iconClass\">\n                  <button\n                    *ngFor=\"let iconClass of col.iconClass\"\n                    (click)=\"\n                      iconClass.buttonMethod\n                        ? iconClass.buttonMethod(row, $event)\n                        : $event.stopPropagation()\n                    \"\n                  >\n                    <span\n                      [ngClass]=\"iconClass.class\"\n                      *ngIf=\"\n                        iconClass.condition === undefined ||\n                        (iconClass.condition !== undefined &&\n                          $any(iconClass.condition)(row))\n                      \"\n                      >{{ iconClass.text }}</span\n                    >\n                  </button>\n                </ng-container>\n              </td>\n            </ng-container>\n          </ng-container>\n\n          <tr mat-header-row *matHeaderRowDef=\"columnProperties\"></tr>\n          <tr\n            [ngClass]=\"{\n              'example-element-row': data.isNotClickable === true,\n              'example-element-row cursor-pointer': !data.isNotClickable\n            }\"\n            mat-row\n            *matRowDef=\"let row; columns: columnProperties\"\n            (click)=\"goToDetails(row)\"\n          ></tr>\n\n          <!-- ROW SHOWN WHEN THERE IS NO MATCHING DATA. -->\n          <tr class=\"mat-row\" *matNoDataRow>\n            <td *ngIf=\"!isLoading\" class=\"mat-cell p-4\" colspan=\"4\">\n              Nenhum resultado encontrado para a busca\n            </td>\n          </tr>\n        </table>\n\n        <div class=\"flex justify-center\" *ngIf=\"isLoading\">\n          <mat-spinner></mat-spinner>\n        </div>\n\n        <div class=\"paginator-container\">\n          <mat-paginator\n            #paginator\n            [pageSizeOptions]=\"[25, 50, 100]\"\n            [pageSize]=\"pageSize\"\n            [length]=\"totalItems\"\n            showFirstLastButtons\n            aria-label=\"Select page of periodic elements\"\n            (page)=\"onPageChange($event)\"\n            [ngClass]=\"{\n              'hide-length':\n                ['filter', 'filterByDate', 'equals'].includes(\n                  this.currentArrange\n                ) || this.data.filterFn,\n              'hide-next-button': !hasNextPage && data.pagination === true,\n              'hide-last-button':\n                (!hasNextPage && data.pagination === true) || this.data.filterFn\n            }\"\n          >\n          </mat-paginator>\n          <div\n            *ngIf=\"\n              !isLoading &&\n              dataSource?.data &&\n              dataSource.data.length > 0 &&\n              data?.filterFn\n            \"\n            class=\"page-number-display\"\n          >\n            {{ currentPageNumber }}\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <!-- TOOLTIP PERSONALIZADO -->\n  <div\n    *ngIf=\"showTooltip\"\n    class=\"fixed z-50 max-w-md break-words rounded-lg bg-gray-800 px-3 py-2 text-sm text-white shadow-lg\"\n    [style.left.px]=\"tooltipPosition.x\"\n    [style.top.px]=\"tooltipPosition.y\"\n    [style.pointer-events]=\"'none'\"\n  >\n    {{ tooltipContent }}\n  </div>\n</div>\n"]}