ng-firebase-table-kxp 1.2.4 → 1.2.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/components/table/table.component.mjs +1557 -0
- package/esm2020/lib/components/table-tabs/table-tabs.component.mjs +129 -0
- package/esm2020/lib/components/table-tooltip/table-tooltip.component.mjs +49 -0
- package/esm2020/lib/ng-firebase-table-kxp.component.mjs +15 -0
- package/esm2020/lib/ng-firebase-table-kxp.module.mjs +103 -0
- package/esm2020/lib/ng-firebase-table-kxp.service.mjs +14 -0
- package/esm2020/lib/services/filter.service.mjs +416 -0
- package/esm2020/lib/services/pagination.service.mjs +115 -0
- package/esm2020/lib/services/table.service.mjs +1140 -0
- package/esm2020/lib/services/tooltip.service.mjs +141 -0
- package/esm2020/lib/types/Table.mjs +9 -0
- package/esm2020/lib/utils/table.utils.mjs +75 -0
- package/esm2020/ng-firebase-table-kxp.mjs +5 -0
- package/esm2020/public-api.mjs +22 -0
- package/fesm2015/ng-firebase-table-kxp.mjs +4026 -0
- package/fesm2015/ng-firebase-table-kxp.mjs.map +1 -0
- package/{fesm2022 → fesm2020}/ng-firebase-table-kxp.mjs +3666 -3659
- package/fesm2020/ng-firebase-table-kxp.mjs.map +1 -0
- package/index.d.ts +5 -666
- package/lib/components/table/table.component.d.ts +135 -0
- package/lib/components/table-tabs/table-tabs.component.d.ts +34 -0
- package/lib/components/table-tooltip/table-tooltip.component.d.ts +18 -0
- package/lib/ng-firebase-table-kxp.component.d.ts +5 -0
- package/lib/ng-firebase-table-kxp.module.d.ts +24 -0
- package/lib/ng-firebase-table-kxp.service.d.ts +6 -0
- package/lib/services/filter.service.d.ts +88 -0
- package/lib/services/pagination.service.d.ts +34 -0
- package/lib/services/table.service.d.ts +80 -0
- package/lib/services/tooltip.service.d.ts +73 -0
- package/lib/types/Table.d.ts +162 -0
- package/lib/utils/table.utils.d.ts +25 -0
- package/package.json +11 -3
- package/public-api.d.ts +12 -0
- package/fesm2022/ng-firebase-table-kxp.mjs.map +0 -1
|
@@ -0,0 +1,416 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "./table.service";
|
|
5
|
+
export class FilterService {
|
|
6
|
+
constructor(tableService) {
|
|
7
|
+
this.tableService = tableService;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Cria um novo FormGroup para um filtro
|
|
11
|
+
*/
|
|
12
|
+
createFilterGroup() {
|
|
13
|
+
const dateValidator = this.tableService.dateFormatValidator();
|
|
14
|
+
return new FormGroup({
|
|
15
|
+
selectFilter: new FormControl(''),
|
|
16
|
+
typeFilter: new FormControl(''),
|
|
17
|
+
selectItem: new FormControl(''),
|
|
18
|
+
initialDate: new FormControl('', [dateValidator]),
|
|
19
|
+
finalDate: new FormControl('', [dateValidator]),
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Adiciona um novo filtro ao FormArray
|
|
24
|
+
*/
|
|
25
|
+
addFilter(filtersForm, isPaginated, filterData) {
|
|
26
|
+
if (isPaginated &&
|
|
27
|
+
this.hasActiveDateFilter(filtersForm) &&
|
|
28
|
+
filterData?.selectFilter?.arrange === 'filterByDate') {
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
const newFilterGroup = this.createFilterGroup();
|
|
32
|
+
if (filterData) {
|
|
33
|
+
if (filterData.selectFilter) {
|
|
34
|
+
newFilterGroup.get('selectFilter')?.setValue(filterData.selectFilter);
|
|
35
|
+
}
|
|
36
|
+
if (filterData.typeFilter) {
|
|
37
|
+
newFilterGroup.get('typeFilter')?.setValue(filterData.typeFilter);
|
|
38
|
+
}
|
|
39
|
+
if (filterData.selectItem) {
|
|
40
|
+
newFilterGroup.get('selectItem')?.setValue(filterData.selectItem);
|
|
41
|
+
}
|
|
42
|
+
if (filterData.initialDate) {
|
|
43
|
+
newFilterGroup.get('initialDate')?.setValue(filterData.initialDate);
|
|
44
|
+
}
|
|
45
|
+
if (filterData.finalDate) {
|
|
46
|
+
newFilterGroup.get('finalDate')?.setValue(filterData.finalDate);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
filtersForm.push(newFilterGroup);
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Verifica se há um filtro de data ativo
|
|
53
|
+
*/
|
|
54
|
+
hasActiveDateFilter(filtersForm) {
|
|
55
|
+
return filtersForm.controls.some((control) => {
|
|
56
|
+
const group = control;
|
|
57
|
+
const selectedFilter = group.get('selectFilter')?.value;
|
|
58
|
+
return selectedFilter?.arrange === 'filterByDate';
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Retorna as opções de filtro disponíveis
|
|
63
|
+
*/
|
|
64
|
+
getAvailableFilterOptions(dropdownItems, isPaginated, filtersForm) {
|
|
65
|
+
if (isPaginated && this.hasActiveDateFilter(filtersForm)) {
|
|
66
|
+
return dropdownItems.filter((item) => item.arrange !== 'filterByDate');
|
|
67
|
+
}
|
|
68
|
+
return dropdownItems;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Remove um filtro do FormArray
|
|
72
|
+
*/
|
|
73
|
+
removeFilter(filtersForm, index) {
|
|
74
|
+
filtersForm.removeAt(index);
|
|
75
|
+
if (filtersForm.length === 0) {
|
|
76
|
+
this.addFilter(filtersForm, false);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Remove todos os filtros do FormArray
|
|
81
|
+
*/
|
|
82
|
+
removeAllFilters(filtersForm) {
|
|
83
|
+
filtersForm.clear();
|
|
84
|
+
this.addFilter(filtersForm, false);
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Aplica máscara de data (DD/MM/AAAA)
|
|
88
|
+
*/
|
|
89
|
+
applyDateMask(event, controlName, filterGroup) {
|
|
90
|
+
const input = event.target;
|
|
91
|
+
let value = input.value.replace(/\D/g, '');
|
|
92
|
+
if (value.length > 8) {
|
|
93
|
+
value = value.slice(0, 8);
|
|
94
|
+
}
|
|
95
|
+
if (value.length <= 2) {
|
|
96
|
+
// mantém como está
|
|
97
|
+
}
|
|
98
|
+
else if (value.length <= 4) {
|
|
99
|
+
value = value.slice(0, 2) + '/' + value.slice(2);
|
|
100
|
+
}
|
|
101
|
+
else {
|
|
102
|
+
value =
|
|
103
|
+
value.slice(0, 2) + '/' + value.slice(2, 4) + '/' + value.slice(4, 8);
|
|
104
|
+
}
|
|
105
|
+
filterGroup.get(controlName)?.setValue(value, { emitEvent: false });
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Aplica filtros client-side aos itens
|
|
109
|
+
*/
|
|
110
|
+
applyClientSideFilters(items, filtersForm, sortBy) {
|
|
111
|
+
let filteredItems = [...items];
|
|
112
|
+
const dateFiltersByProperty = {};
|
|
113
|
+
const otherFilters = [];
|
|
114
|
+
filtersForm.controls.forEach((control) => {
|
|
115
|
+
const group = control;
|
|
116
|
+
const selectedFilter = group.get('selectFilter')?.value;
|
|
117
|
+
if (!selectedFilter)
|
|
118
|
+
return;
|
|
119
|
+
const arrange = selectedFilter.arrange;
|
|
120
|
+
if (arrange === 'filterByDate') {
|
|
121
|
+
const initial = group.get('initialDate')?.value;
|
|
122
|
+
const final = group.get('finalDate')?.value;
|
|
123
|
+
if (initial && final && initial.trim() && final.trim()) {
|
|
124
|
+
try {
|
|
125
|
+
const datePattern = /^(\d{1,2})\/(\d{1,2})\/(\d{4})$/;
|
|
126
|
+
if (!datePattern.test(initial) || !datePattern.test(final)) {
|
|
127
|
+
return;
|
|
128
|
+
}
|
|
129
|
+
const [dayI, monthI, yearI] = initial.split('/');
|
|
130
|
+
const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
|
|
131
|
+
const [dayF, monthF, yearF] = final.split('/');
|
|
132
|
+
const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
|
|
133
|
+
finalDate.setHours(23, 59, 59);
|
|
134
|
+
if (isNaN(initialDate.getTime()) || isNaN(finalDate.getTime())) {
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
const dateField = selectedFilter.property || sortBy.field;
|
|
138
|
+
if (!dateFiltersByProperty[dateField]) {
|
|
139
|
+
dateFiltersByProperty[dateField] = [];
|
|
140
|
+
}
|
|
141
|
+
dateFiltersByProperty[dateField].push({
|
|
142
|
+
initial: initialDate,
|
|
143
|
+
final: finalDate,
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
catch (error) {
|
|
147
|
+
console.warn('Erro ao processar datas do filtro:', error);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
else {
|
|
152
|
+
otherFilters.push({ group, selectedFilter, arrange });
|
|
153
|
+
}
|
|
154
|
+
});
|
|
155
|
+
// Aplicar filtros de data
|
|
156
|
+
Object.keys(dateFiltersByProperty).forEach((dateField) => {
|
|
157
|
+
const intervals = dateFiltersByProperty[dateField];
|
|
158
|
+
filteredItems = filteredItems.filter((item) => {
|
|
159
|
+
try {
|
|
160
|
+
const fieldValue = item[dateField];
|
|
161
|
+
if (!fieldValue) {
|
|
162
|
+
return false;
|
|
163
|
+
}
|
|
164
|
+
let itemDate;
|
|
165
|
+
if (typeof fieldValue.toDate === 'function') {
|
|
166
|
+
itemDate = fieldValue.toDate();
|
|
167
|
+
}
|
|
168
|
+
else if (fieldValue instanceof Date) {
|
|
169
|
+
itemDate = fieldValue;
|
|
170
|
+
}
|
|
171
|
+
else if (typeof fieldValue === 'string') {
|
|
172
|
+
itemDate = new Date(fieldValue);
|
|
173
|
+
if (isNaN(itemDate.getTime())) {
|
|
174
|
+
return false;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
else if (typeof fieldValue === 'number') {
|
|
178
|
+
itemDate = new Date(fieldValue);
|
|
179
|
+
}
|
|
180
|
+
else {
|
|
181
|
+
return false;
|
|
182
|
+
}
|
|
183
|
+
return intervals.some((interval) => itemDate >= interval.initial && itemDate <= interval.final);
|
|
184
|
+
}
|
|
185
|
+
catch (error) {
|
|
186
|
+
console.warn('Erro ao processar filtro de data para o item:', item.id, error);
|
|
187
|
+
return false;
|
|
188
|
+
}
|
|
189
|
+
});
|
|
190
|
+
});
|
|
191
|
+
// Aplicar outros filtros
|
|
192
|
+
otherFilters.forEach(({ group, selectedFilter, arrange }) => {
|
|
193
|
+
if (arrange === 'filter') {
|
|
194
|
+
const filterValue = group.get('typeFilter')?.value;
|
|
195
|
+
if (filterValue && filterValue.trim()) {
|
|
196
|
+
const property = selectedFilter.property;
|
|
197
|
+
filteredItems = filteredItems.filter((item) => {
|
|
198
|
+
const value = String(item[property] || '')
|
|
199
|
+
.trim()
|
|
200
|
+
.toLocaleLowerCase();
|
|
201
|
+
return value.includes(filterValue.trim().toLocaleLowerCase());
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
else if (selectedFilter.hasOwnProperty('items') &&
|
|
206
|
+
arrange === 'equals') {
|
|
207
|
+
const selectedItems = group.get('selectItem')?.value;
|
|
208
|
+
if (Array.isArray(selectedItems) && selectedItems.length > 0) {
|
|
209
|
+
filteredItems = filteredItems.filter((item) => {
|
|
210
|
+
return selectedItems.some((selectedItem) => {
|
|
211
|
+
const itemValue = item[selectedItem.property];
|
|
212
|
+
return itemValue === selectedItem.value;
|
|
213
|
+
});
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
});
|
|
218
|
+
return filteredItems;
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Constrói o objeto Arrange a partir dos filtros ativos
|
|
222
|
+
*/
|
|
223
|
+
buildArrangeFromFilters(filtersForm, sortBy) {
|
|
224
|
+
const activeFilters = filtersForm.controls
|
|
225
|
+
.flatMap((control) => {
|
|
226
|
+
const group = control;
|
|
227
|
+
const selectedFilter = group.get('selectFilter')?.value;
|
|
228
|
+
if (!selectedFilter)
|
|
229
|
+
return [];
|
|
230
|
+
const arrange = selectedFilter.arrange;
|
|
231
|
+
if (arrange === 'filter') {
|
|
232
|
+
const filterValue = group.get('typeFilter')?.value;
|
|
233
|
+
if (!filterValue)
|
|
234
|
+
return [];
|
|
235
|
+
return {
|
|
236
|
+
arrange,
|
|
237
|
+
filter: {
|
|
238
|
+
property: selectedFilter.property,
|
|
239
|
+
filtering: filterValue,
|
|
240
|
+
},
|
|
241
|
+
dateFilter: undefined,
|
|
242
|
+
};
|
|
243
|
+
}
|
|
244
|
+
if (arrange === 'filterByDate') {
|
|
245
|
+
const initial = group.get('initialDate')?.value;
|
|
246
|
+
const final = group.get('finalDate')?.value;
|
|
247
|
+
if (initial && final) {
|
|
248
|
+
try {
|
|
249
|
+
const [dayI, monthI, yearI] = initial.split('/');
|
|
250
|
+
const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
|
|
251
|
+
const [dayF, monthF, yearF] = final.split('/');
|
|
252
|
+
const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
|
|
253
|
+
finalDate.setHours(23, 59, 59);
|
|
254
|
+
return {
|
|
255
|
+
arrange,
|
|
256
|
+
filter: undefined,
|
|
257
|
+
dateFilter: {
|
|
258
|
+
initial: initialDate,
|
|
259
|
+
final: finalDate,
|
|
260
|
+
},
|
|
261
|
+
};
|
|
262
|
+
}
|
|
263
|
+
catch (error) {
|
|
264
|
+
return [];
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
return [];
|
|
268
|
+
}
|
|
269
|
+
if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {
|
|
270
|
+
const selectedItems = group.get('selectItem')?.value;
|
|
271
|
+
if (Array.isArray(selectedItems) && selectedItems.length > 0) {
|
|
272
|
+
return selectedItems.map((item) => ({
|
|
273
|
+
arrange,
|
|
274
|
+
filter: {
|
|
275
|
+
property: item.property,
|
|
276
|
+
filtering: item.value,
|
|
277
|
+
},
|
|
278
|
+
dateFilter: undefined,
|
|
279
|
+
}));
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
return [];
|
|
283
|
+
})
|
|
284
|
+
.filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));
|
|
285
|
+
return {
|
|
286
|
+
filters: activeFilters,
|
|
287
|
+
sortBy: sortBy,
|
|
288
|
+
};
|
|
289
|
+
}
|
|
290
|
+
/**
|
|
291
|
+
* Extrai filtros ativos para paginação
|
|
292
|
+
*/
|
|
293
|
+
extractActiveFilters(filtersForm) {
|
|
294
|
+
return filtersForm.controls
|
|
295
|
+
.flatMap((control) => {
|
|
296
|
+
const group = control;
|
|
297
|
+
const selectedFilter = group.get('selectFilter')?.value;
|
|
298
|
+
if (!selectedFilter)
|
|
299
|
+
return [];
|
|
300
|
+
const arrange = selectedFilter.arrange;
|
|
301
|
+
if (arrange === 'filter') {
|
|
302
|
+
const filterValue = group.get('typeFilter')?.value;
|
|
303
|
+
if (!filterValue)
|
|
304
|
+
return [];
|
|
305
|
+
return {
|
|
306
|
+
arrange,
|
|
307
|
+
filter: {
|
|
308
|
+
property: selectedFilter.property,
|
|
309
|
+
filtering: filterValue,
|
|
310
|
+
},
|
|
311
|
+
dateFilter: undefined,
|
|
312
|
+
};
|
|
313
|
+
}
|
|
314
|
+
if (arrange === 'filterByDate') {
|
|
315
|
+
const initial = group.get('initialDate')?.value;
|
|
316
|
+
const final = group.get('finalDate')?.value;
|
|
317
|
+
if (initial && final) {
|
|
318
|
+
const [dayI, monthI, yearI] = initial.split('/');
|
|
319
|
+
const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);
|
|
320
|
+
const [dayF, monthF, yearF] = final.split('/');
|
|
321
|
+
const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);
|
|
322
|
+
finalDate.setHours(23, 59, 59);
|
|
323
|
+
return {
|
|
324
|
+
arrange,
|
|
325
|
+
filter: undefined,
|
|
326
|
+
dateFilter: {
|
|
327
|
+
initial: initialDate,
|
|
328
|
+
final: finalDate,
|
|
329
|
+
},
|
|
330
|
+
};
|
|
331
|
+
}
|
|
332
|
+
return [];
|
|
333
|
+
}
|
|
334
|
+
if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {
|
|
335
|
+
const selectedItems = group.get('selectItem')?.value;
|
|
336
|
+
if (Array.isArray(selectedItems) && selectedItems.length > 0) {
|
|
337
|
+
return selectedItems.map((item) => ({
|
|
338
|
+
arrange,
|
|
339
|
+
filter: {
|
|
340
|
+
property: item.property,
|
|
341
|
+
filtering: item.value,
|
|
342
|
+
},
|
|
343
|
+
dateFilter: undefined,
|
|
344
|
+
}));
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
return [];
|
|
348
|
+
})
|
|
349
|
+
.filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));
|
|
350
|
+
}
|
|
351
|
+
/**
|
|
352
|
+
* Inicializa os dropdownItems baseado nas colunas
|
|
353
|
+
*/
|
|
354
|
+
initializeDropdownItems(displayedColumns) {
|
|
355
|
+
const dropdownItems = [];
|
|
356
|
+
const sortableDropdownItems = [];
|
|
357
|
+
let hasFilterableColumn = false;
|
|
358
|
+
let hasSortableColumn = false;
|
|
359
|
+
displayedColumns.forEach((col) => {
|
|
360
|
+
if (col.isFilterable) {
|
|
361
|
+
hasFilterableColumn = true;
|
|
362
|
+
dropdownItems.push({
|
|
363
|
+
...col,
|
|
364
|
+
arrange: 'filter',
|
|
365
|
+
title: col.title,
|
|
366
|
+
});
|
|
367
|
+
}
|
|
368
|
+
if (col.isSortable) {
|
|
369
|
+
hasSortableColumn = true;
|
|
370
|
+
sortableDropdownItems.push({
|
|
371
|
+
...col,
|
|
372
|
+
arrange: 'ascending',
|
|
373
|
+
title: col.title + ': crescente',
|
|
374
|
+
});
|
|
375
|
+
sortableDropdownItems.push({
|
|
376
|
+
...col,
|
|
377
|
+
arrange: 'descending',
|
|
378
|
+
title: col.title + ': decrescente',
|
|
379
|
+
});
|
|
380
|
+
}
|
|
381
|
+
if (col.isFilterableByDate) {
|
|
382
|
+
dropdownItems.push({
|
|
383
|
+
...col,
|
|
384
|
+
arrange: 'filterByDate',
|
|
385
|
+
title: col.title + ': filtro por data',
|
|
386
|
+
});
|
|
387
|
+
}
|
|
388
|
+
});
|
|
389
|
+
return {
|
|
390
|
+
dropdownItems,
|
|
391
|
+
sortableDropdownItems,
|
|
392
|
+
hasFilterableColumn,
|
|
393
|
+
hasSortableColumn,
|
|
394
|
+
};
|
|
395
|
+
}
|
|
396
|
+
/**
|
|
397
|
+
* Adiciona opções de filtro customizadas
|
|
398
|
+
*/
|
|
399
|
+
addFilterableOptions(dropdownItems, filterableOptions) {
|
|
400
|
+
if (filterableOptions &&
|
|
401
|
+
Array.isArray(filterableOptions) &&
|
|
402
|
+
filterableOptions.length > 0) {
|
|
403
|
+
filterableOptions.forEach((option) => dropdownItems.push({ ...option, arrange: 'equals' }));
|
|
404
|
+
}
|
|
405
|
+
return dropdownItems;
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
FilterService.ɵfac = function FilterService_Factory(t) { return new (t || FilterService)(i0.ɵɵinject(i1.TableService)); };
|
|
409
|
+
FilterService.ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: FilterService, factory: FilterService.ɵfac, providedIn: 'root' });
|
|
410
|
+
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(FilterService, [{
|
|
411
|
+
type: Injectable,
|
|
412
|
+
args: [{
|
|
413
|
+
providedIn: 'root',
|
|
414
|
+
}]
|
|
415
|
+
}], function () { return [{ type: i1.TableService }]; }, null); })();
|
|
416
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter.service.js","sourceRoot":"","sources":["../../../../../projects/ng-firebase-table-kxp/src/lib/services/filter.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { FormArray, FormControl, FormGroup } from '@angular/forms';\r\nimport { Arrange, Column, TABLE_DEFAULTS } from '../types/Table';\r\nimport { TableService } from './table.service';\r\nimport { OrderByDirection } from 'firebase/firestore';\r\n\r\nexport interface FilterConfig {\r\n  selectFilter: any;\r\n  typeFilter?: string;\r\n  selectItem?: any[];\r\n  initialDate?: string;\r\n  finalDate?: string;\r\n}\r\n\r\nexport interface DateFiltersByProperty {\r\n  [property: string]: Array<{ initial: Date; final: Date }>;\r\n}\r\n\r\nexport interface ProcessedFilter {\r\n  group: FormGroup;\r\n  selectedFilter: any;\r\n  arrange: string;\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class FilterService {\r\n  constructor(private tableService: TableService) {}\r\n\r\n  /**\r\n   * Cria um novo FormGroup para um filtro\r\n   */\r\n  createFilterGroup(): FormGroup {\r\n    const dateValidator = this.tableService.dateFormatValidator();\r\n\r\n    return new FormGroup({\r\n      selectFilter: new FormControl(''),\r\n      typeFilter: new FormControl(''),\r\n      selectItem: new FormControl(''),\r\n      initialDate: new FormControl('', [dateValidator]),\r\n      finalDate: new FormControl('', [dateValidator]),\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Adiciona um novo filtro ao FormArray\r\n   */\r\n  addFilter(\r\n    filtersForm: FormArray,\r\n    isPaginated: boolean,\r\n    filterData?: FilterConfig\r\n  ): void {\r\n    if (\r\n      isPaginated &&\r\n      this.hasActiveDateFilter(filtersForm) &&\r\n      filterData?.selectFilter?.arrange === 'filterByDate'\r\n    ) {\r\n      return;\r\n    }\r\n\r\n    const newFilterGroup = this.createFilterGroup();\r\n\r\n    if (filterData) {\r\n      if (filterData.selectFilter) {\r\n        newFilterGroup.get('selectFilter')?.setValue(filterData.selectFilter);\r\n      }\r\n      if (filterData.typeFilter) {\r\n        newFilterGroup.get('typeFilter')?.setValue(filterData.typeFilter);\r\n      }\r\n      if (filterData.selectItem) {\r\n        newFilterGroup.get('selectItem')?.setValue(filterData.selectItem);\r\n      }\r\n      if (filterData.initialDate) {\r\n        newFilterGroup.get('initialDate')?.setValue(filterData.initialDate);\r\n      }\r\n      if (filterData.finalDate) {\r\n        newFilterGroup.get('finalDate')?.setValue(filterData.finalDate);\r\n      }\r\n    }\r\n\r\n    filtersForm.push(newFilterGroup);\r\n  }\r\n\r\n  /**\r\n   * Verifica se há um filtro de data ativo\r\n   */\r\n  hasActiveDateFilter(filtersForm: FormArray): boolean {\r\n    return filtersForm.controls.some((control) => {\r\n      const group = control as FormGroup;\r\n      const selectedFilter = group.get('selectFilter')?.value;\r\n      return selectedFilter?.arrange === 'filterByDate';\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Retorna as opções de filtro disponíveis\r\n   */\r\n  getAvailableFilterOptions(\r\n    dropdownItems: any[],\r\n    isPaginated: boolean,\r\n    filtersForm: FormArray\r\n  ): any[] {\r\n    if (isPaginated && this.hasActiveDateFilter(filtersForm)) {\r\n      return dropdownItems.filter((item) => item.arrange !== 'filterByDate');\r\n    }\r\n    return dropdownItems;\r\n  }\r\n\r\n  /**\r\n   * Remove um filtro do FormArray\r\n   */\r\n  removeFilter(filtersForm: FormArray, index: number): void {\r\n    filtersForm.removeAt(index);\r\n    if (filtersForm.length === 0) {\r\n      this.addFilter(filtersForm, false);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Remove todos os filtros do FormArray\r\n   */\r\n  removeAllFilters(filtersForm: FormArray): void {\r\n    filtersForm.clear();\r\n    this.addFilter(filtersForm, false);\r\n  }\r\n\r\n  /**\r\n   * Aplica máscara de data (DD/MM/AAAA)\r\n   */\r\n  applyDateMask(\r\n    event: any,\r\n    controlName: 'initialDate' | 'finalDate',\r\n    filterGroup: FormGroup\r\n  ): void {\r\n    const input = event.target;\r\n    let value = input.value.replace(/\\D/g, '');\r\n\r\n    if (value.length > 8) {\r\n      value = value.slice(0, 8);\r\n    }\r\n\r\n    if (value.length <= 2) {\r\n      // mantém como está\r\n    } else if (value.length <= 4) {\r\n      value = value.slice(0, 2) + '/' + value.slice(2);\r\n    } else {\r\n      value =\r\n        value.slice(0, 2) + '/' + value.slice(2, 4) + '/' + value.slice(4, 8);\r\n    }\r\n\r\n    filterGroup.get(controlName)?.setValue(value, { emitEvent: false });\r\n  }\r\n\r\n  /**\r\n   * Aplica filtros client-side aos itens\r\n   */\r\n  applyClientSideFilters<T = any>(\r\n    items: T[],\r\n    filtersForm: FormArray,\r\n    sortBy: { field: string; order: OrderByDirection }\r\n  ): T[] {\r\n    let filteredItems = [...items];\r\n\r\n    const dateFiltersByProperty: DateFiltersByProperty = {};\r\n    const otherFilters: ProcessedFilter[] = [];\r\n\r\n    filtersForm.controls.forEach((control) => {\r\n      const group = control as FormGroup;\r\n      const selectedFilter = group.get('selectFilter')?.value;\r\n      if (!selectedFilter) return;\r\n\r\n      const arrange = selectedFilter.arrange;\r\n\r\n      if (arrange === 'filterByDate') {\r\n        const initial = group.get('initialDate')?.value;\r\n        const final = group.get('finalDate')?.value;\r\n\r\n        if (initial && final && initial.trim() && final.trim()) {\r\n          try {\r\n            const datePattern = /^(\\d{1,2})\\/(\\d{1,2})\\/(\\d{4})$/;\r\n            if (!datePattern.test(initial) || !datePattern.test(final)) {\r\n              return;\r\n            }\r\n\r\n            const [dayI, monthI, yearI] = initial.split('/');\r\n            const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\r\n            const [dayF, monthF, yearF] = final.split('/');\r\n            const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\r\n            finalDate.setHours(23, 59, 59);\r\n\r\n            if (isNaN(initialDate.getTime()) || isNaN(finalDate.getTime())) {\r\n              return;\r\n            }\r\n\r\n            const dateField = selectedFilter.property || sortBy.field;\r\n\r\n            if (!dateFiltersByProperty[dateField]) {\r\n              dateFiltersByProperty[dateField] = [];\r\n            }\r\n\r\n            dateFiltersByProperty[dateField].push({\r\n              initial: initialDate,\r\n              final: finalDate,\r\n            });\r\n          } catch (error) {\r\n            console.warn('Erro ao processar datas do filtro:', error);\r\n          }\r\n        }\r\n      } else {\r\n        otherFilters.push({ group, selectedFilter, arrange });\r\n      }\r\n    });\r\n\r\n    // Aplicar filtros de data\r\n    Object.keys(dateFiltersByProperty).forEach((dateField) => {\r\n      const intervals = dateFiltersByProperty[dateField];\r\n\r\n      filteredItems = filteredItems.filter((item: any) => {\r\n        try {\r\n          const fieldValue = item[dateField];\r\n\r\n          if (!fieldValue) {\r\n            return false;\r\n          }\r\n\r\n          let itemDate: Date;\r\n          if (typeof fieldValue.toDate === 'function') {\r\n            itemDate = fieldValue.toDate();\r\n          } else if (fieldValue instanceof Date) {\r\n            itemDate = fieldValue;\r\n          } else if (typeof fieldValue === 'string') {\r\n            itemDate = new Date(fieldValue);\r\n            if (isNaN(itemDate.getTime())) {\r\n              return false;\r\n            }\r\n          } else if (typeof fieldValue === 'number') {\r\n            itemDate = new Date(fieldValue);\r\n          } else {\r\n            return false;\r\n          }\r\n\r\n          return intervals.some(\r\n            (interval) =>\r\n              itemDate >= interval.initial && itemDate <= interval.final\r\n          );\r\n        } catch (error) {\r\n          console.warn(\r\n            'Erro ao processar filtro de data para o item:',\r\n            item.id,\r\n            error\r\n          );\r\n          return false;\r\n        }\r\n      });\r\n    });\r\n\r\n    // Aplicar outros filtros\r\n    otherFilters.forEach(({ group, selectedFilter, arrange }) => {\r\n      if (arrange === 'filter') {\r\n        const filterValue = group.get('typeFilter')?.value;\r\n        if (filterValue && filterValue.trim()) {\r\n          const property = selectedFilter.property;\r\n          filteredItems = filteredItems.filter((item: any) => {\r\n            const value = String(item[property] || '')\r\n              .trim()\r\n              .toLocaleLowerCase();\r\n            return value.includes(filterValue.trim().toLocaleLowerCase());\r\n          });\r\n        }\r\n      } else if (\r\n        selectedFilter.hasOwnProperty('items') &&\r\n        arrange === 'equals'\r\n      ) {\r\n        const selectedItems = group.get('selectItem')?.value;\r\n        if (Array.isArray(selectedItems) && selectedItems.length > 0) {\r\n          filteredItems = filteredItems.filter((item: any) => {\r\n            return selectedItems.some((selectedItem: any) => {\r\n              const itemValue = item[selectedItem.property];\r\n              return itemValue === selectedItem.value;\r\n            });\r\n          });\r\n        }\r\n      }\r\n    });\r\n\r\n    return filteredItems;\r\n  }\r\n\r\n  /**\r\n   * Constrói o objeto Arrange a partir dos filtros ativos\r\n   */\r\n  buildArrangeFromFilters(\r\n    filtersForm: FormArray,\r\n    sortBy: { field: string; order: OrderByDirection }\r\n  ): Arrange {\r\n    const activeFilters = filtersForm.controls\r\n      .flatMap((control) => {\r\n        const group = control as FormGroup;\r\n        const selectedFilter = group.get('selectFilter')?.value;\r\n        if (!selectedFilter) return [];\r\n\r\n        const arrange = selectedFilter.arrange;\r\n\r\n        if (arrange === 'filter') {\r\n          const filterValue = group.get('typeFilter')?.value;\r\n          if (!filterValue) return [];\r\n          return {\r\n            arrange,\r\n            filter: {\r\n              property: selectedFilter.property,\r\n              filtering: filterValue,\r\n            },\r\n            dateFilter: undefined,\r\n          };\r\n        }\r\n\r\n        if (arrange === 'filterByDate') {\r\n          const initial = group.get('initialDate')?.value;\r\n          const final = group.get('finalDate')?.value;\r\n          if (initial && final) {\r\n            try {\r\n              const [dayI, monthI, yearI] = initial.split('/');\r\n              const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\r\n              const [dayF, monthF, yearF] = final.split('/');\r\n              const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\r\n              finalDate.setHours(23, 59, 59);\r\n              return {\r\n                arrange,\r\n                filter: undefined,\r\n                dateFilter: {\r\n                  initial: initialDate,\r\n                  final: finalDate,\r\n                },\r\n              };\r\n            } catch (error) {\r\n              return [];\r\n            }\r\n          }\r\n          return [];\r\n        }\r\n\r\n        if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {\r\n          const selectedItems = group.get('selectItem')?.value;\r\n          if (Array.isArray(selectedItems) && selectedItems.length > 0) {\r\n            return selectedItems.map((item) => ({\r\n              arrange,\r\n              filter: {\r\n                property: item.property,\r\n                filtering: item.value,\r\n              },\r\n              dateFilter: undefined,\r\n            }));\r\n          }\r\n        }\r\n\r\n        return [];\r\n      })\r\n      .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));\r\n\r\n    return {\r\n      filters: activeFilters,\r\n      sortBy: sortBy,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Extrai filtros ativos para paginação\r\n   */\r\n  extractActiveFilters(filtersForm: FormArray): any[] {\r\n    return filtersForm.controls\r\n      .flatMap((control) => {\r\n        const group = control as FormGroup;\r\n        const selectedFilter = group.get('selectFilter')?.value;\r\n        if (!selectedFilter) return [];\r\n\r\n        const arrange = selectedFilter.arrange;\r\n\r\n        if (arrange === 'filter') {\r\n          const filterValue = group.get('typeFilter')?.value;\r\n          if (!filterValue) return [];\r\n          return {\r\n            arrange,\r\n            filter: {\r\n              property: selectedFilter.property,\r\n              filtering: filterValue,\r\n            },\r\n            dateFilter: undefined,\r\n          };\r\n        }\r\n\r\n        if (arrange === 'filterByDate') {\r\n          const initial = group.get('initialDate')?.value;\r\n          const final = group.get('finalDate')?.value;\r\n          if (initial && final) {\r\n            const [dayI, monthI, yearI] = initial.split('/');\r\n            const initialDate = new Date(`${monthI}/${dayI}/${yearI}`);\r\n            const [dayF, monthF, yearF] = final.split('/');\r\n            const finalDate = new Date(`${monthF}/${dayF}/${yearF}`);\r\n            finalDate.setHours(23, 59, 59);\r\n            return {\r\n              arrange,\r\n              filter: undefined,\r\n              dateFilter: {\r\n                initial: initialDate,\r\n                final: finalDate,\r\n              },\r\n            };\r\n          }\r\n          return [];\r\n        }\r\n\r\n        if (selectedFilter.hasOwnProperty('items') && arrange === 'equals') {\r\n          const selectedItems = group.get('selectItem')?.value;\r\n          if (Array.isArray(selectedItems) && selectedItems.length > 0) {\r\n            return selectedItems.map((item) => ({\r\n              arrange,\r\n              filter: {\r\n                property: item.property,\r\n                filtering: item.value,\r\n              },\r\n              dateFilter: undefined,\r\n            }));\r\n          }\r\n        }\r\n\r\n        return [];\r\n      })\r\n      .filter((f) => f && (f.filter?.filtering !== undefined || f.dateFilter));\r\n  }\r\n\r\n  /**\r\n   * Inicializa os dropdownItems baseado nas colunas\r\n   */\r\n  initializeDropdownItems(displayedColumns: Column[]): {\r\n    dropdownItems: any[];\r\n    sortableDropdownItems: any[];\r\n    hasFilterableColumn: boolean;\r\n    hasSortableColumn: boolean;\r\n  } {\r\n    const dropdownItems: any[] = [];\r\n    const sortableDropdownItems: any[] = [];\r\n    let hasFilterableColumn = false;\r\n    let hasSortableColumn = false;\r\n\r\n    displayedColumns.forEach((col) => {\r\n      if (col.isFilterable) {\r\n        hasFilterableColumn = true;\r\n        dropdownItems.push({\r\n          ...col,\r\n          arrange: 'filter',\r\n          title: col.title,\r\n        });\r\n      }\r\n      if (col.isSortable) {\r\n        hasSortableColumn = true;\r\n        sortableDropdownItems.push({\r\n          ...col,\r\n          arrange: 'ascending',\r\n          title: col.title + ': crescente',\r\n        });\r\n        sortableDropdownItems.push({\r\n          ...col,\r\n          arrange: 'descending',\r\n          title: col.title + ': decrescente',\r\n        });\r\n      }\r\n      if (col.isFilterableByDate) {\r\n        dropdownItems.push({\r\n          ...col,\r\n          arrange: 'filterByDate',\r\n          title: col.title + ': filtro por data',\r\n        });\r\n      }\r\n    });\r\n\r\n    return {\r\n      dropdownItems,\r\n      sortableDropdownItems,\r\n      hasFilterableColumn,\r\n      hasSortableColumn,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Adiciona opções de filtro customizadas\r\n   */\r\n  addFilterableOptions(dropdownItems: any[], filterableOptions: any[]): any[] {\r\n    if (\r\n      filterableOptions &&\r\n      Array.isArray(filterableOptions) &&\r\n      filterableOptions.length > 0\r\n    ) {\r\n      filterableOptions.forEach((option) =>\r\n        dropdownItems.push({ ...option, arrange: 'equals' })\r\n      );\r\n    }\r\n    return dropdownItems;\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class PaginationService {
|
|
4
|
+
calculateNavigation(event, currentState) {
|
|
5
|
+
const previousPageIndex = event.previousPageIndex ?? 0;
|
|
6
|
+
const pageIndex = event.pageIndex;
|
|
7
|
+
const eventPageSize = event.pageSize;
|
|
8
|
+
const totalItems = currentState.totalItems;
|
|
9
|
+
const currentComponentPageSize = currentState.pageSize;
|
|
10
|
+
let navigationDirection = 'reload';
|
|
11
|
+
let resetDocs = false;
|
|
12
|
+
let originalPageSize = null;
|
|
13
|
+
let shouldNavigate = true;
|
|
14
|
+
let newClientPageIndex = pageIndex;
|
|
15
|
+
let newPageNumber = currentState.currentPageNumber;
|
|
16
|
+
const lastPageIndex = Math.max(0, Math.ceil(totalItems / eventPageSize) - 1);
|
|
17
|
+
// Atualizar número da página
|
|
18
|
+
if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {
|
|
19
|
+
newPageNumber++;
|
|
20
|
+
}
|
|
21
|
+
else if (previousPageIndex !== undefined &&
|
|
22
|
+
pageIndex < previousPageIndex) {
|
|
23
|
+
newPageNumber = Math.max(1, newPageNumber - 1);
|
|
24
|
+
}
|
|
25
|
+
// Mudança de tamanho de página
|
|
26
|
+
if (eventPageSize !== currentComponentPageSize) {
|
|
27
|
+
navigationDirection = 'forward';
|
|
28
|
+
resetDocs = true;
|
|
29
|
+
newClientPageIndex = 0;
|
|
30
|
+
}
|
|
31
|
+
// Indo para primeira página
|
|
32
|
+
else if (pageIndex === 0 &&
|
|
33
|
+
previousPageIndex !== undefined &&
|
|
34
|
+
pageIndex < previousPageIndex) {
|
|
35
|
+
navigationDirection = 'forward';
|
|
36
|
+
newPageNumber = 1;
|
|
37
|
+
newClientPageIndex = 0;
|
|
38
|
+
resetDocs = true;
|
|
39
|
+
}
|
|
40
|
+
// Indo para última página (salto)
|
|
41
|
+
else if (pageIndex === lastPageIndex &&
|
|
42
|
+
previousPageIndex !== undefined &&
|
|
43
|
+
pageIndex > previousPageIndex &&
|
|
44
|
+
pageIndex - previousPageIndex > 1) {
|
|
45
|
+
navigationDirection = 'backward';
|
|
46
|
+
resetDocs = true;
|
|
47
|
+
const itemsExpectedInLastPage = totalItems - lastPageIndex * eventPageSize;
|
|
48
|
+
if (itemsExpectedInLastPage > 0 &&
|
|
49
|
+
itemsExpectedInLastPage < eventPageSize) {
|
|
50
|
+
originalPageSize = currentComponentPageSize;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
// Próxima página
|
|
54
|
+
else if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {
|
|
55
|
+
navigationDirection = 'forward';
|
|
56
|
+
resetDocs = false;
|
|
57
|
+
}
|
|
58
|
+
// Página anterior
|
|
59
|
+
else if (previousPageIndex !== undefined && pageIndex < previousPageIndex) {
|
|
60
|
+
navigationDirection = 'backward';
|
|
61
|
+
resetDocs = false;
|
|
62
|
+
}
|
|
63
|
+
// Mesma página (reload)
|
|
64
|
+
else if (previousPageIndex !== undefined &&
|
|
65
|
+
pageIndex === previousPageIndex) {
|
|
66
|
+
navigationDirection = 'reload';
|
|
67
|
+
resetDocs = false;
|
|
68
|
+
}
|
|
69
|
+
// Caso inicial (primeira carga)
|
|
70
|
+
else if (previousPageIndex === undefined && pageIndex === 0) {
|
|
71
|
+
shouldNavigate = false;
|
|
72
|
+
}
|
|
73
|
+
// Caso não tratado
|
|
74
|
+
else {
|
|
75
|
+
console.warn('Condição de navegação não tratada:', event);
|
|
76
|
+
shouldNavigate = false;
|
|
77
|
+
}
|
|
78
|
+
return {
|
|
79
|
+
direction: navigationDirection,
|
|
80
|
+
resetDocs,
|
|
81
|
+
originalPageSize,
|
|
82
|
+
shouldNavigate,
|
|
83
|
+
newClientPageIndex,
|
|
84
|
+
newPageNumber,
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Calcula o número de itens esperados na última página
|
|
89
|
+
*/
|
|
90
|
+
calculateLastPageItemCount(totalItems, pageSize) {
|
|
91
|
+
const lastPageIndex = Math.max(0, Math.ceil(totalItems / pageSize) - 1);
|
|
92
|
+
return totalItems - lastPageIndex * pageSize;
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Calcula o índice da última página
|
|
96
|
+
*/
|
|
97
|
+
calculateLastPageIndex(totalItems, pageSize) {
|
|
98
|
+
return Math.max(0, Math.ceil(totalItems / pageSize) - 1);
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Verifica se deve mostrar o botão de próxima página
|
|
102
|
+
*/
|
|
103
|
+
shouldShowNextButton(hasNextPage, isPaginated) {
|
|
104
|
+
return hasNextPage || !isPaginated;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
PaginationService.ɵfac = function PaginationService_Factory(t) { return new (t || PaginationService)(); };
|
|
108
|
+
PaginationService.ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: PaginationService, factory: PaginationService.ɵfac, providedIn: 'root' });
|
|
109
|
+
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(PaginationService, [{
|
|
110
|
+
type: Injectable,
|
|
111
|
+
args: [{
|
|
112
|
+
providedIn: 'root',
|
|
113
|
+
}]
|
|
114
|
+
}], null, null); })();
|
|
115
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.service.js","sourceRoot":"","sources":["../../../../../projects/ng-firebase-table-kxp/src/lib/services/pagination.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { PageEvent } from '@angular/material/paginator';\r\n\r\nexport interface PaginationState {\r\n  currentPageNumber: number;\r\n  currentClientPageIndex: number;\r\n  pageSize: number;\r\n  totalItems: number;\r\n  hasNextPage: boolean;\r\n}\r\n\r\nexport interface NavigationResult {\r\n  direction: 'forward' | 'backward' | 'reload';\r\n  resetDocs: boolean;\r\n  originalPageSize: number | null;\r\n  shouldNavigate: boolean;\r\n  newClientPageIndex: number;\r\n  newPageNumber: number;\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class PaginationService {\r\n  calculateNavigation(\r\n    event: PageEvent,\r\n    currentState: PaginationState\r\n  ): NavigationResult {\r\n    const previousPageIndex = event.previousPageIndex ?? 0;\r\n    const pageIndex = event.pageIndex;\r\n    const eventPageSize = event.pageSize;\r\n    const totalItems = currentState.totalItems;\r\n    const currentComponentPageSize = currentState.pageSize;\r\n\r\n    let navigationDirection: 'forward' | 'backward' | 'reload' = 'reload';\r\n    let resetDocs = false;\r\n    let originalPageSize: number | null = null;\r\n    let shouldNavigate = true;\r\n    let newClientPageIndex = pageIndex;\r\n    let newPageNumber = currentState.currentPageNumber;\r\n\r\n    const lastPageIndex = Math.max(\r\n      0,\r\n      Math.ceil(totalItems / eventPageSize) - 1\r\n    );\r\n\r\n    // Atualizar número da página\r\n    if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {\r\n      newPageNumber++;\r\n    } else if (\r\n      previousPageIndex !== undefined &&\r\n      pageIndex < previousPageIndex\r\n    ) {\r\n      newPageNumber = Math.max(1, newPageNumber - 1);\r\n    }\r\n\r\n    // Mudança de tamanho de página\r\n    if (eventPageSize !== currentComponentPageSize) {\r\n      navigationDirection = 'forward';\r\n      resetDocs = true;\r\n      newClientPageIndex = 0;\r\n    }\r\n    // Indo para primeira página\r\n    else if (\r\n      pageIndex === 0 &&\r\n      previousPageIndex !== undefined &&\r\n      pageIndex < previousPageIndex\r\n    ) {\r\n      navigationDirection = 'forward';\r\n      newPageNumber = 1;\r\n      newClientPageIndex = 0;\r\n      resetDocs = true;\r\n    }\r\n    // Indo para última página (salto)\r\n    else if (\r\n      pageIndex === lastPageIndex &&\r\n      previousPageIndex !== undefined &&\r\n      pageIndex > previousPageIndex &&\r\n      pageIndex - previousPageIndex > 1\r\n    ) {\r\n      navigationDirection = 'backward';\r\n      resetDocs = true;\r\n\r\n      const itemsExpectedInLastPage =\r\n        totalItems - lastPageIndex * eventPageSize;\r\n\r\n      if (\r\n        itemsExpectedInLastPage > 0 &&\r\n        itemsExpectedInLastPage < eventPageSize\r\n      ) {\r\n        originalPageSize = currentComponentPageSize;\r\n      }\r\n    }\r\n    // Próxima página\r\n    else if (previousPageIndex !== undefined && pageIndex > previousPageIndex) {\r\n      navigationDirection = 'forward';\r\n      resetDocs = false;\r\n    }\r\n    // Página anterior\r\n    else if (previousPageIndex !== undefined && pageIndex < previousPageIndex) {\r\n      navigationDirection = 'backward';\r\n      resetDocs = false;\r\n    }\r\n    // Mesma página (reload)\r\n    else if (\r\n      previousPageIndex !== undefined &&\r\n      pageIndex === previousPageIndex\r\n    ) {\r\n      navigationDirection = 'reload';\r\n      resetDocs = false;\r\n    }\r\n    // Caso inicial (primeira carga)\r\n    else if (previousPageIndex === undefined && pageIndex === 0) {\r\n      shouldNavigate = false;\r\n    }\r\n    // Caso não tratado\r\n    else {\r\n      console.warn('Condição de navegação não tratada:', event);\r\n      shouldNavigate = false;\r\n    }\r\n\r\n    return {\r\n      direction: navigationDirection,\r\n      resetDocs,\r\n      originalPageSize,\r\n      shouldNavigate,\r\n      newClientPageIndex,\r\n      newPageNumber,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Calcula o número de itens esperados na última página\r\n   */\r\n  calculateLastPageItemCount(totalItems: number, pageSize: number): number {\r\n    const lastPageIndex = Math.max(0, Math.ceil(totalItems / pageSize) - 1);\r\n    return totalItems - lastPageIndex * pageSize;\r\n  }\r\n\r\n  /**\r\n   * Calcula o índice da última página\r\n   */\r\n  calculateLastPageIndex(totalItems: number, pageSize: number): number {\r\n    return Math.max(0, Math.ceil(totalItems / pageSize) - 1);\r\n  }\r\n\r\n  /**\r\n   * Verifica se deve mostrar o botão de próxima página\r\n   */\r\n  shouldShowNextButton(hasNextPage: boolean, isPaginated: boolean): boolean {\r\n    return hasNextPage || !isPaginated;\r\n  }\r\n}\r\n"]}
|