ng-firebase-table-kxp 1.2.0 → 1.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/{fesm2020 → fesm2022}/ng-firebase-table-kxp.mjs +2762 -2744
  2. package/fesm2022/ng-firebase-table-kxp.mjs.map +1 -0
  3. package/index.d.ts +665 -5
  4. package/package.json +3 -11
  5. package/esm2020/lib/components/table/table.component.mjs +0 -710
  6. package/esm2020/lib/components/table-tabs/table-tabs.component.mjs +0 -73
  7. package/esm2020/lib/components/table-tooltip/table-tooltip.component.mjs +0 -34
  8. package/esm2020/lib/ng-firebase-table-kxp.component.mjs +0 -11
  9. package/esm2020/lib/ng-firebase-table-kxp.module.mjs +0 -98
  10. package/esm2020/lib/ng-firebase-table-kxp.service.mjs +0 -14
  11. package/esm2020/lib/services/filter.service.mjs +0 -416
  12. package/esm2020/lib/services/pagination.service.mjs +0 -115
  13. package/esm2020/lib/services/table.service.mjs +0 -1140
  14. package/esm2020/lib/services/tooltip.service.mjs +0 -141
  15. package/esm2020/lib/types/Table.mjs +0 -9
  16. package/esm2020/lib/utils/table.utils.mjs +0 -75
  17. package/esm2020/ng-firebase-table-kxp.mjs +0 -5
  18. package/esm2020/public-api.mjs +0 -22
  19. package/fesm2015/ng-firebase-table-kxp.mjs +0 -2869
  20. package/fesm2015/ng-firebase-table-kxp.mjs.map +0 -1
  21. package/fesm2020/ng-firebase-table-kxp.mjs.map +0 -1
  22. package/lib/components/table/table.component.d.ts +0 -132
  23. package/lib/components/table/table.component.d.ts.map +0 -1
  24. package/lib/components/table-tabs/table-tabs.component.d.ts +0 -34
  25. package/lib/components/table-tabs/table-tabs.component.d.ts.map +0 -1
  26. package/lib/components/table-tooltip/table-tooltip.component.d.ts +0 -18
  27. package/lib/components/table-tooltip/table-tooltip.component.d.ts.map +0 -1
  28. package/lib/ng-firebase-table-kxp.component.d.ts +0 -5
  29. package/lib/ng-firebase-table-kxp.component.d.ts.map +0 -1
  30. package/lib/ng-firebase-table-kxp.module.d.ts +0 -23
  31. package/lib/ng-firebase-table-kxp.module.d.ts.map +0 -1
  32. package/lib/ng-firebase-table-kxp.service.d.ts +0 -6
  33. package/lib/ng-firebase-table-kxp.service.d.ts.map +0 -1
  34. package/lib/services/filter.service.d.ts +0 -88
  35. package/lib/services/filter.service.d.ts.map +0 -1
  36. package/lib/services/pagination.service.d.ts +0 -34
  37. package/lib/services/pagination.service.d.ts.map +0 -1
  38. package/lib/services/table.service.d.ts +0 -80
  39. package/lib/services/table.service.d.ts.map +0 -1
  40. package/lib/services/tooltip.service.d.ts +0 -73
  41. package/lib/services/tooltip.service.d.ts.map +0 -1
  42. package/lib/types/Table.d.ts +0 -162
  43. package/lib/types/Table.d.ts.map +0 -1
  44. package/lib/utils/table.utils.d.ts +0 -25
  45. package/lib/utils/table.utils.d.ts.map +0 -1
  46. package/ng-firebase-table-kxp.d.ts.map +0 -1
  47. package/public-api.d.ts +0 -12
  48. package/public-api.d.ts.map +0 -1
@@ -1,1140 +0,0 @@
1
- import { Injectable, Optional } from '@angular/core';
2
- import firebase from 'firebase/compat/app';
3
- import { firstValueFrom } from 'rxjs';
4
- import * as moment from 'moment';
5
- import * as i0 from "@angular/core";
6
- import * as i1 from "@angular/fire/compat/firestore";
7
- import * as i2 from "@angular/material/dialog";
8
- import * as i3 from "ngx-toastr";
9
- export class TableService {
10
- constructor(ngFire, dialog, toastr) {
11
- this.ngFire = ngFire;
12
- this.dialog = dialog;
13
- this.toastr = toastr;
14
- this.operators = {
15
- '==': (a, b) => a === b,
16
- '!=': (a, b) => a !== b,
17
- '>': (a, b) => a > b,
18
- '<': (a, b) => a < b,
19
- '>=': (a, b) => a >= b,
20
- '<=': (a, b) => a <= b,
21
- in: (a, b) => Array.isArray(b) && b.includes(a),
22
- 'not-in': (a, b) => Array.isArray(b) && !b.includes(a),
23
- 'array-contains': (a, b) => Array.isArray(a) && a.includes(b),
24
- 'array-contains-any': (a, b) => Array.isArray(a) &&
25
- Array.isArray(b) &&
26
- b.some((item) => a.includes(item)),
27
- includes: (a, b) => {
28
- if (typeof a === 'string' && typeof b === 'string') {
29
- return a.includes(b);
30
- }
31
- if (Array.isArray(a)) {
32
- return a.includes(b);
33
- }
34
- return false;
35
- },
36
- };
37
- }
38
- async getItems(collection) {
39
- try {
40
- const querySnapshot = await collection.get();
41
- return querySnapshot.docs.map((doc) => {
42
- const data = doc.data();
43
- return { ...(data ?? {}), id: doc.id };
44
- });
45
- }
46
- catch (error) {
47
- console.warn('Collection não encontrada:', error);
48
- return [];
49
- }
50
- }
51
- async executeQuery(params) {
52
- if (params.filterFn) {
53
- // Lógica com filtro no cliente (filterFn)
54
- const BATCH_FETCH_SIZE = params.batchSize;
55
- const GOAL_SIZE = params.batchSize + 1;
56
- if (params.navigation === 'forward' || params.navigation === 'reload') {
57
- if (params.navigation === 'reload' && params.doc) {
58
- params.doc.lastDoc = null;
59
- }
60
- let lastDocCursor = params.doc ? params.doc.lastDoc : null;
61
- let pageResults = [];
62
- let allFetchedDocs = [];
63
- let hasMoreDocsInDb = true;
64
- while (pageResults.length < GOAL_SIZE && hasMoreDocsInDb) {
65
- let query = this.ngFire.collection(params.collection).ref;
66
- query = this.applyFilters(query, params.arrange, params.conditions);
67
- if (lastDocCursor) {
68
- query = query.startAfter(lastDocCursor);
69
- }
70
- query = query.limit(BATCH_FETCH_SIZE);
71
- const snapshot = await query.get();
72
- if (snapshot.empty) {
73
- hasMoreDocsInDb = false;
74
- break;
75
- }
76
- lastDocCursor = snapshot.docs[snapshot.docs.length - 1];
77
- allFetchedDocs.push(...snapshot.docs);
78
- const batchUsers = snapshot.docs
79
- .map((doc) => {
80
- const data = doc.data();
81
- return { id: doc.id, ...(data ?? {}) };
82
- })
83
- .filter(params.filterFn);
84
- pageResults.push(...batchUsers);
85
- if (snapshot.size < BATCH_FETCH_SIZE) {
86
- hasMoreDocsInDb = false;
87
- }
88
- }
89
- const hasNextPage = pageResults.length > params.batchSize;
90
- const finalItems = pageResults.slice(0, params.batchSize);
91
- const firstDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[0]?.id) || null;
92
- const lastDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[finalItems.length - 1]?.id) || null;
93
- return {
94
- items: finalItems,
95
- filterLength: null,
96
- firstDoc: firstDocOfPage,
97
- lastDoc: lastDocOfPage,
98
- hasNextPage: hasNextPage,
99
- hasPreviousPage: !!(params.doc && params.doc.lastDoc) &&
100
- params.navigation !== 'reload',
101
- currentClientPageIndex: undefined,
102
- };
103
- }
104
- // Lógica para trás (backward)
105
- else if (params.navigation === 'backward') {
106
- if (!params.doc || !params.doc.firstDoc) {
107
- return {
108
- items: [],
109
- filterLength: null,
110
- firstDoc: null,
111
- lastDoc: null,
112
- hasNextPage: true,
113
- hasPreviousPage: false,
114
- currentClientPageIndex: undefined,
115
- };
116
- }
117
- let pageResults = [];
118
- let allFetchedDocs = [];
119
- let hasMoreDocsInDb = true;
120
- let boundaryDoc = params.doc.firstDoc;
121
- while (pageResults.length < GOAL_SIZE && hasMoreDocsInDb) {
122
- if (!boundaryDoc)
123
- break;
124
- let query = this.ngFire.collection(params.collection).ref;
125
- query = this.applyFilters(query, params.arrange, params.conditions);
126
- query = query.endBefore(boundaryDoc);
127
- query = query.limitToLast(BATCH_FETCH_SIZE);
128
- const snapshot = await query.get();
129
- if (snapshot.empty) {
130
- hasMoreDocsInDb = false;
131
- break;
132
- }
133
- boundaryDoc = snapshot
134
- .docs[0];
135
- allFetchedDocs = [...snapshot.docs, ...allFetchedDocs];
136
- const batchUsers = snapshot.docs
137
- .map((doc) => {
138
- const data = doc.data();
139
- return { id: doc.id, ...(data ?? {}) };
140
- })
141
- .filter(params.filterFn);
142
- pageResults = [...batchUsers, ...pageResults];
143
- if (snapshot.size < BATCH_FETCH_SIZE) {
144
- hasMoreDocsInDb = false;
145
- }
146
- }
147
- const finalItems = pageResults.slice(0, params.batchSize);
148
- const firstDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[0]?.id) || null;
149
- const lastDocOfPage = allFetchedDocs.find((doc) => doc.id === finalItems[finalItems.length - 1]?.id) || null;
150
- return {
151
- items: finalItems,
152
- filterLength: null,
153
- firstDoc: firstDocOfPage,
154
- lastDoc: lastDocOfPage,
155
- hasNextPage: true,
156
- currentClientPageIndex: undefined,
157
- };
158
- }
159
- }
160
- else {
161
- let items = [];
162
- let docs = [];
163
- let hasNextPage = false;
164
- let filterLength = null;
165
- let query = this.ngFire.collection(params.collection).ref;
166
- if (params.conditions) {
167
- params.conditions.forEach((c) => {
168
- if (c.operator === '!=') {
169
- query = query.orderBy(c.firestoreProperty);
170
- }
171
- });
172
- }
173
- query = this.applyFilters(query, params.arrange, params.conditions);
174
- if (params.navigation === 'reload') {
175
- query = query.limit(params.batchSize + 1);
176
- if (params.doc && params.doc.firstDoc) {
177
- query = query.startAt(params.doc.firstDoc);
178
- }
179
- }
180
- else if (params.navigation === 'forward') {
181
- query = query.limit(params.batchSize + 1);
182
- if (params.doc && params.doc.lastDoc) {
183
- query = query.startAfter(params.doc.lastDoc);
184
- }
185
- }
186
- else {
187
- // backward
188
- query = query.limitToLast(params.batchSize + 1);
189
- if (params.doc && params.doc.firstDoc) {
190
- query = query.endBefore(params.doc.firstDoc);
191
- }
192
- }
193
- const itemCol = await query.get();
194
- itemCol.docs.forEach((doc) => docs.push(doc));
195
- const itemPromises = docs.map(async (item) => {
196
- const itemData = item.data();
197
- items.push({ id: item.id, ...(itemData ?? {}) });
198
- });
199
- let lastDoc = docs[docs.length - 1] || null;
200
- let firstDoc = docs[0];
201
- if ((items.length > params.batchSize && params.navigation === 'forward') ||
202
- (params.navigation === 'reload' && items.length > params.batchSize)) {
203
- lastDoc = docs[docs.length - 2] || null;
204
- items.pop();
205
- hasNextPage = true;
206
- }
207
- if (items.length > params.batchSize && params.navigation === 'backward') {
208
- firstDoc = docs[1];
209
- items.shift();
210
- hasNextPage = true;
211
- }
212
- await Promise.all(itemPromises);
213
- return {
214
- items,
215
- filterLength,
216
- lastDoc,
217
- firstDoc,
218
- hasNextPage,
219
- currentClientPageIndex: undefined,
220
- };
221
- }
222
- // Fallback para garantir que sempre retornamos algo
223
- return {
224
- items: [],
225
- filterLength: null,
226
- firstDoc: null,
227
- lastDoc: null,
228
- hasNextPage: false,
229
- currentClientPageIndex: undefined,
230
- };
231
- }
232
- applyFilters(query, arrange, conditions) {
233
- if (conditions) {
234
- conditions.map((cond) => {
235
- query = query.where(cond.firestoreProperty, cond.operator, cond.dashProperty);
236
- });
237
- }
238
- let hasFilterSpecificOrderBy = false;
239
- let appliedOrderByField = null;
240
- const equalsFilters = arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
241
- const otherFilters = arrange.filters.filter((f) => f.arrange !== 'equals');
242
- const equalsGroupedByProperty = equalsFilters.reduce((acc, current) => {
243
- const prop = current.filter.property;
244
- if (!acc[prop]) {
245
- acc[prop] = [];
246
- }
247
- acc[prop].push(current.filter.filtering);
248
- return acc;
249
- }, {});
250
- for (const prop in equalsGroupedByProperty) {
251
- const values = equalsGroupedByProperty[prop];
252
- if (values.length > 0) {
253
- query = query.where(prop, 'in', values);
254
- }
255
- }
256
- otherFilters.forEach((filterItem) => {
257
- // Aplicar filtragem por busca
258
- if (filterItem.filter?.filtering &&
259
- filterItem.filter?.property !== '' &&
260
- filterItem.arrange === 'filter') {
261
- query = query
262
- .where(filterItem.filter.property, '>=', filterItem.filter.filtering.trim().toUpperCase())
263
- .where(filterItem.filter.property, '<=', filterItem.filter.filtering.trim().toUpperCase() + '\uf8ff');
264
- if (!hasFilterSpecificOrderBy) {
265
- query = query.orderBy(filterItem.filter.property);
266
- hasFilterSpecificOrderBy = true;
267
- appliedOrderByField = filterItem.filter.property;
268
- }
269
- }
270
- // Aplicar filtro do tipo "filterByDate"
271
- if (filterItem.dateFilter && filterItem.arrange === 'filterByDate') {
272
- query = query
273
- .where(arrange.sortBy.field, '>=', filterItem.dateFilter.initial)
274
- .where(arrange.sortBy.field, '<=', filterItem.dateFilter.final);
275
- if (!hasFilterSpecificOrderBy) {
276
- query = query.orderBy(arrange.sortBy.field);
277
- hasFilterSpecificOrderBy = true;
278
- appliedOrderByField = arrange.sortBy.field;
279
- }
280
- }
281
- });
282
- // Aplicar sortBy
283
- if (arrange.sortBy && arrange.sortBy.field && arrange.sortBy.order) {
284
- if (appliedOrderByField !== arrange.sortBy.field) {
285
- query = query.orderBy(arrange.sortBy.field, arrange.sortBy.order);
286
- }
287
- }
288
- return query;
289
- }
290
- getIdFilter(params) {
291
- if (!params.arrange?.filters)
292
- return null;
293
- const idFilter = params.arrange.filters.find((f) => f.arrange === 'filter' &&
294
- !!f.filter &&
295
- f.filter.property === 'id' &&
296
- !!f.filter.filtering);
297
- return idFilter?.filter?.filtering?.trim() || null;
298
- }
299
- async getDocumentById(collection, docId) {
300
- try {
301
- const docRef = this.ngFire.collection(collection).doc(docId);
302
- const docSnapshot = await docRef.get().toPromise();
303
- if (docSnapshot && docSnapshot.exists) {
304
- const data = docSnapshot.data();
305
- return {
306
- id: docSnapshot.id,
307
- ...(data ?? {}),
308
- };
309
- }
310
- return null;
311
- }
312
- catch (error) {
313
- console.warn('Erro ao buscar documento por ID:', error);
314
- return null;
315
- }
316
- }
317
- async searchByIdPartial(params, searchTerm) {
318
- const exactMatch = await this.getDocumentById(params.collection, searchTerm);
319
- if (exactMatch) {
320
- if (params.conditions) {
321
- const operators = this.operators;
322
- const passesConditions = params.conditions.every((cond) => {
323
- const operatorFn = operators[cond.operator];
324
- return operatorFn
325
- ? operatorFn(exactMatch[cond.firestoreProperty], cond.dashProperty)
326
- : false;
327
- });
328
- if (!passesConditions) {
329
- return {
330
- items: [],
331
- filterLength: 0,
332
- firstDoc: null,
333
- lastDoc: null,
334
- hasNextPage: false,
335
- hasPreviousPage: false,
336
- currentClientPageIndex: 0,
337
- totalPages: 0,
338
- };
339
- }
340
- }
341
- if (params.filterFn && !params.filterFn(exactMatch)) {
342
- return {
343
- items: [],
344
- filterLength: 0,
345
- firstDoc: null,
346
- lastDoc: null,
347
- hasNextPage: false,
348
- hasPreviousPage: false,
349
- currentClientPageIndex: 0,
350
- totalPages: 0,
351
- };
352
- }
353
- return {
354
- items: [exactMatch],
355
- filterLength: 1,
356
- firstDoc: null,
357
- lastDoc: null,
358
- hasNextPage: false,
359
- hasPreviousPage: false,
360
- currentClientPageIndex: 0,
361
- totalPages: 1,
362
- };
363
- }
364
- const searchTermLower = searchTerm.toLowerCase();
365
- const paramsWithoutIdFilter = {
366
- ...params,
367
- arrange: {
368
- ...params.arrange,
369
- filters: params.arrange.filters.filter((f) => !(f.arrange === 'filter' && f.filter?.property === 'id')),
370
- },
371
- };
372
- let query = this.ngFire.collection(params.collection).ref;
373
- // Aplicar conditions
374
- if (params.conditions) {
375
- params.conditions.forEach((cond) => {
376
- query = query.where(cond.firestoreProperty, cond.operator, cond.dashProperty);
377
- });
378
- }
379
- // Aplicar sortBy
380
- if (params.arrange?.sortBy?.field && params.arrange?.sortBy?.order) {
381
- query = query.orderBy(params.arrange.sortBy.field, params.arrange.sortBy.order);
382
- }
383
- const snapshot = await query.get();
384
- let items = snapshot.docs
385
- .map((doc) => {
386
- const data = doc.data();
387
- return { id: doc.id, ...(data ?? {}) };
388
- })
389
- .filter((item) => item.id.toLowerCase().includes(searchTermLower));
390
- // Separar equals filters e outros filtros
391
- const equalsFilters = paramsWithoutIdFilter.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
392
- const otherFilters = paramsWithoutIdFilter.arrange.filters.filter((f) => f.arrange !== 'equals' &&
393
- (f.arrange !== 'filter' || f.filter?.property !== 'id'));
394
- // Aplicar equals filters com lógica OR dentro de cada propriedade
395
- if (equalsFilters.length > 0) {
396
- // Agrupar por propriedade para aplicar OR
397
- const groupedByProperty = equalsFilters.reduce((acc, f) => {
398
- const prop = f.filter.property;
399
- if (!acc[prop]) {
400
- acc[prop] = [];
401
- }
402
- acc[prop].push(f.filter.filtering);
403
- return acc;
404
- }, {});
405
- // Filtrar: item deve ter pelo menos um valor de CADA propriedade
406
- items = items.filter((item) => {
407
- return Object.entries(groupedByProperty).every(([prop, values]) => {
408
- const itemValue = item[prop];
409
- return values.includes(itemValue);
410
- });
411
- });
412
- }
413
- // Aplicar outros filtros
414
- otherFilters.forEach((filterItem) => {
415
- if (filterItem.arrange === 'filter' &&
416
- filterItem.filter?.filtering &&
417
- filterItem.filter?.property) {
418
- const filterValue = String(filterItem.filter.filtering)
419
- .trim()
420
- .toLowerCase();
421
- items = items.filter((item) => {
422
- const itemValue = String(item[filterItem.filter.property] ?? '').toLowerCase();
423
- return itemValue.includes(filterValue);
424
- });
425
- }
426
- });
427
- // Aplicar filterFn se existir
428
- if (params.filterFn) {
429
- items = items.filter(params.filterFn);
430
- }
431
- // Paginação
432
- const pageSize = params.batchSize;
433
- let currentClientPageIndex = 0;
434
- if (params.navigation === 'reload') {
435
- currentClientPageIndex = 0;
436
- }
437
- else {
438
- // Usar o índice passado pelo componente sem incrementar/decrementar
439
- currentClientPageIndex = params.clientPageIndex || 0;
440
- }
441
- const startIndex = currentClientPageIndex * pageSize;
442
- const endIndex = startIndex + pageSize;
443
- const paginatedItems = items.slice(startIndex, endIndex);
444
- const totalPages = Math.ceil(items.length / pageSize);
445
- const hasNextPage = currentClientPageIndex < totalPages - 1;
446
- const hasPreviousPage = currentClientPageIndex > 0;
447
- return {
448
- items: paginatedItems,
449
- filterLength: items.length,
450
- firstDoc: null,
451
- lastDoc: null,
452
- hasNextPage,
453
- hasPreviousPage,
454
- currentClientPageIndex,
455
- totalPages,
456
- };
457
- }
458
- shouldUseClientSideFallback(params) {
459
- const hasConditions = params.conditions && params.conditions.length > 0;
460
- const hasArrangeFilters = params.arrange?.filters && params.arrange.filters.length > 0;
461
- const hasSortBy = params.arrange?.sortBy?.field;
462
- if (params.filterFn) {
463
- return false;
464
- }
465
- if (hasArrangeFilters) {
466
- const equalsFilters = params.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
467
- if (equalsFilters.length > 0) {
468
- const propertiesSet = new Set(equalsFilters.map((f) => f.filter.property));
469
- if (propertiesSet.size > 1) {
470
- return true;
471
- }
472
- }
473
- }
474
- if (hasConditions && hasArrangeFilters && hasSortBy) {
475
- return true;
476
- }
477
- if (hasConditions && hasArrangeFilters) {
478
- return true;
479
- }
480
- if (hasArrangeFilters && params.arrange.filters.length > 1 && hasSortBy) {
481
- const equalsFilters = params.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
482
- if (equalsFilters.length > 0) {
483
- const propertiesSet = new Set(equalsFilters.map((f) => f.filter.property));
484
- if (propertiesSet.size === 1 &&
485
- equalsFilters.length === params.arrange.filters.length) {
486
- return false;
487
- }
488
- }
489
- return true;
490
- }
491
- return false;
492
- }
493
- async getPaginated(params) {
494
- const idFilterValue = this.getIdFilter(params);
495
- if (idFilterValue) {
496
- const result = await this.searchByIdPartial(params, idFilterValue);
497
- return result;
498
- }
499
- // Detectar preventivamente se deve usar fallback
500
- if (this.shouldUseClientSideFallback(params)) {
501
- await this.trackMissingIndexPreventive(params.collection, params.arrange, params.conditions);
502
- const result = await this.executeClientSideQuery(params);
503
- return result;
504
- }
505
- try {
506
- const result = await this.executeQuery(params);
507
- return result;
508
- }
509
- catch (error) {
510
- if (error &&
511
- typeof error === 'object' &&
512
- 'code' in error &&
513
- error.code === 'failed-precondition') {
514
- await this.trackMissingIndex(error, params.collection, params.arrange, params.conditions);
515
- const result = await this.executeClientSideQuery(params);
516
- return result;
517
- }
518
- else if (error &&
519
- typeof error === 'object' &&
520
- 'code' in error &&
521
- error.code === 'invalid-argument') {
522
- await this.trackMissingIndex(error, params.collection, params.arrange, params.conditions);
523
- const result = await this.executeClientSideQuery(params);
524
- return result;
525
- }
526
- else {
527
- throw error;
528
- }
529
- }
530
- }
531
- async executeClientSideQuery(params) {
532
- // Otimizar usando pelo menos uma cláusula .where() quando possível
533
- let query = this.ngFire.collection(params.collection).ref;
534
- let appliedCondition = null;
535
- let hasAppliedWhereClause = false;
536
- // Primeiro, tenta aplicar condições simples
537
- if (params.conditions && params.conditions.length > 0) {
538
- const simpleCondition = params.conditions.find((cond) => ['==', '>', '<', '>=', '<=', 'in', 'array-contains'].includes(cond.operator));
539
- if (simpleCondition) {
540
- query = query.where(simpleCondition.firestoreProperty, simpleCondition.operator, simpleCondition.dashProperty);
541
- appliedCondition = simpleCondition;
542
- hasAppliedWhereClause = true;
543
- }
544
- }
545
- // Se não há condições disponíveis, tenta aplicar filtros do arrange
546
- let appliedFirestoreFilter = null;
547
- if (!hasAppliedWhereClause && params.arrange?.filters) {
548
- // Agrupar equals filters por propriedade
549
- const equalsFilters = params.arrange.filters.filter((f) => f.arrange === 'equals' && !!f.filter);
550
- const equalsGroupedByProperty = equalsFilters.reduce((acc, current) => {
551
- const prop = current.filter.property;
552
- if (!acc[prop]) {
553
- acc[prop] = [];
554
- }
555
- acc[prop].push(current.filter.filtering);
556
- return acc;
557
- }, {});
558
- // Se há apenas UMA propriedade com múltiplos valores, aplicar no Firestore com 'in'
559
- const properties = Object.keys(equalsGroupedByProperty);
560
- if (properties.length === 1 && equalsFilters.length > 0) {
561
- const prop = properties[0];
562
- const values = equalsGroupedByProperty[prop];
563
- query = query.where(prop, 'in', values);
564
- hasAppliedWhereClause = true;
565
- // Marcar TODOS os equals filters dessa propriedade como aplicados
566
- appliedFirestoreFilter = 'all-equals';
567
- }
568
- else if (properties.length === 0) {
569
- const otherFilter = params.arrange.filters.find((f) => (f.arrange === 'filter' &&
570
- !!f.filter?.filtering &&
571
- !!f.filter?.property) ||
572
- (f.arrange === 'filterByDate' &&
573
- !!f.dateFilter?.initial &&
574
- !!f.dateFilter?.final));
575
- if (otherFilter) {
576
- if (otherFilter.arrange === 'filter' && otherFilter.filter) {
577
- const filterValue = otherFilter.filter.filtering
578
- .trim()
579
- .toUpperCase();
580
- query = query
581
- .where(otherFilter.filter.property, '>=', filterValue)
582
- .where(otherFilter.filter.property, '<=', filterValue + '\uf8ff');
583
- hasAppliedWhereClause = true;
584
- appliedFirestoreFilter = otherFilter;
585
- }
586
- else if (otherFilter.arrange === 'filterByDate' &&
587
- otherFilter.dateFilter &&
588
- params.arrange.sortBy?.field) {
589
- query = query
590
- .where(params.arrange.sortBy.field, '>=', otherFilter.dateFilter.initial)
591
- .where(params.arrange.sortBy.field, '<=', otherFilter.dateFilter.final);
592
- hasAppliedWhereClause = true;
593
- appliedFirestoreFilter = otherFilter;
594
- }
595
- }
596
- }
597
- }
598
- const allDocsSnapshot = await query.get();
599
- let items = allDocsSnapshot.docs.map((doc) => {
600
- const data = doc.data();
601
- return { id: doc.id, ...(data ?? {}) };
602
- });
603
- // Aplicar condições restantes
604
- if (params.conditions) {
605
- const remainingConditions = params.conditions.filter((cond) => cond !== appliedCondition);
606
- if (remainingConditions.length > 0) {
607
- const operators = this.operators;
608
- items = items.filter((item) => {
609
- return remainingConditions.every((cond) => {
610
- const operatorFn = operators[cond.operator];
611
- return operatorFn
612
- ? operatorFn(item[cond.firestoreProperty], cond.dashProperty)
613
- : false;
614
- });
615
- });
616
- }
617
- }
618
- const { filters, sortBy } = params.arrange;
619
- // Track which filter was already applied in Firestore to avoid double filtering
620
- if (hasAppliedWhereClause && !appliedCondition && params.arrange?.filters) {
621
- const equalsFilter = params.arrange.filters.find((f) => f.arrange === 'equals' && !!f.filter?.filtering);
622
- if (equalsFilter) {
623
- appliedFirestoreFilter = equalsFilter;
624
- }
625
- else {
626
- const foundFilter = params.arrange.filters.find((f) => (f.arrange === 'filter' &&
627
- !!f.filter?.filtering &&
628
- !!f.filter?.property) ||
629
- (f.arrange === 'filterByDate' &&
630
- !!f.dateFilter?.initial &&
631
- !!f.dateFilter?.final));
632
- if (foundFilter) {
633
- appliedFirestoreFilter = foundFilter;
634
- }
635
- }
636
- }
637
- const equalsFilters = filters.filter((f) => f.arrange === 'equals');
638
- const otherFilters = filters.filter((f) => f.arrange !== 'equals');
639
- // Aplicar equals filters no client-side apenas se não foram aplicados no Firestore
640
- if (appliedFirestoreFilter !== 'all-equals' && equalsFilters.length > 0) {
641
- // Agrupar por propriedade para aplicar OR dentro de cada propriedade
642
- const groupedByProperty = equalsFilters.reduce((acc, f) => {
643
- if (f.filter) {
644
- const prop = f.filter.property;
645
- if (!acc[prop]) {
646
- acc[prop] = [];
647
- }
648
- acc[prop].push(f.filter.filtering);
649
- }
650
- return acc;
651
- }, {});
652
- // Filtrar: item deve ter pelo menos um valor de CADA propriedade
653
- // (AND entre propriedades, OR dentro de cada propriedade)
654
- items = items.filter((item) => {
655
- return Object.entries(groupedByProperty).every(([prop, values]) => {
656
- const itemValue = item[prop];
657
- return values.includes(itemValue);
658
- });
659
- });
660
- }
661
- otherFilters.forEach((filterItem) => {
662
- if (appliedFirestoreFilter === filterItem) {
663
- return;
664
- }
665
- if (filterItem.arrange === 'filter' &&
666
- filterItem.filter?.filtering &&
667
- filterItem.filter?.property) {
668
- const filterValue = String(filterItem.filter.filtering)
669
- .trim()
670
- .toLowerCase();
671
- items = items.filter((item) => {
672
- const itemValue = String(item[filterItem.filter.property] ?? '').toLowerCase();
673
- return itemValue.includes(filterValue);
674
- });
675
- }
676
- if (filterItem.arrange === 'filterByDate' &&
677
- filterItem.dateFilter?.initial &&
678
- filterItem.dateFilter?.final &&
679
- sortBy.field) {
680
- items = items.filter((item) => {
681
- try {
682
- const fieldValue = item[sortBy.field];
683
- if (!fieldValue) {
684
- return false;
685
- }
686
- let itemDate;
687
- const fieldValueObj = fieldValue;
688
- if (fieldValueObj &&
689
- typeof fieldValueObj === 'object' &&
690
- 'toDate' in fieldValueObj &&
691
- typeof fieldValueObj.toDate === 'function') {
692
- itemDate = fieldValueObj.toDate();
693
- }
694
- else if (fieldValueObj instanceof Date) {
695
- itemDate = fieldValueObj;
696
- }
697
- else if (typeof fieldValueObj === 'string') {
698
- itemDate = new Date(fieldValueObj);
699
- if (isNaN(itemDate.getTime())) {
700
- return false;
701
- }
702
- }
703
- else if (typeof fieldValueObj === 'number') {
704
- itemDate = new Date(fieldValueObj);
705
- }
706
- else {
707
- return false;
708
- }
709
- if (!filterItem.dateFilter) {
710
- return false;
711
- }
712
- return (itemDate >= filterItem.dateFilter.initial &&
713
- itemDate <= filterItem.dateFilter.final);
714
- }
715
- catch (error) {
716
- console.warn('Erro ao processar filtro de data para o item:', item.id, error);
717
- return false;
718
- }
719
- });
720
- }
721
- });
722
- // Aplicar filterFn se existir
723
- if (params.filterFn) {
724
- items = items.filter(params.filterFn);
725
- }
726
- if (sortBy && sortBy.field && sortBy.order) {
727
- items.sort((a, b) => {
728
- const valA = a[sortBy.field];
729
- const valB = b[sortBy.field];
730
- if (valA === null || valA === undefined)
731
- return 1;
732
- if (valB === null || valB === undefined)
733
- return -1;
734
- if (valA < valB) {
735
- return sortBy.order === 'asc' ? -1 : 1;
736
- }
737
- if (valA > valB) {
738
- return sortBy.order === 'asc' ? 1 : -1;
739
- }
740
- return 0;
741
- });
742
- }
743
- // Implementação adequada da paginação
744
- let currentClientPageIndex = 0;
745
- // Determinar a página atual baseada na navegação
746
- if (params.navigation === 'reload') {
747
- currentClientPageIndex = 0;
748
- }
749
- else {
750
- currentClientPageIndex = params.clientPageIndex || 0;
751
- }
752
- const pageSize = params.batchSize;
753
- const startIndex = currentClientPageIndex * pageSize;
754
- const endIndex = startIndex + pageSize;
755
- const paginatedItems = items.slice(startIndex, endIndex);
756
- const totalPages = Math.ceil(items.length / pageSize);
757
- const hasNextPage = currentClientPageIndex < totalPages - 1;
758
- const hasPreviousPage = currentClientPageIndex > 0;
759
- return {
760
- items: paginatedItems,
761
- filterLength: items.length,
762
- lastDoc: null,
763
- firstDoc: null,
764
- hasNextPage: hasNextPage,
765
- hasPreviousPage: hasPreviousPage,
766
- currentClientPageIndex: currentClientPageIndex,
767
- totalPages: totalPages,
768
- };
769
- }
770
- async getItemsData(collection, arrange, conditions = undefined) {
771
- try {
772
- let query = this.ngFire.collection(collection).ref;
773
- query = this.applyFilters(query, arrange, conditions);
774
- const snapshot = await query.get();
775
- return await Promise.all(snapshot.docs.map(async (doc) => {
776
- const data = doc.data();
777
- const id = doc.id;
778
- return {
779
- id,
780
- ...(data ?? {}),
781
- };
782
- }));
783
- }
784
- catch (e) {
785
- throw e;
786
- }
787
- }
788
- async deleteIndex(id, col) {
789
- try {
790
- const batch = this.ngFire.firestore.batch();
791
- const docRef = this.ngFire.collection(col).doc(id);
792
- const docSnapshot = (await firstValueFrom(docRef.get()));
793
- const doc = docSnapshot.data();
794
- batch.delete(docRef.ref);
795
- if (doc && typeof doc['index'] === 'number') {
796
- await this.reindex(doc['index'], col, batch);
797
- }
798
- await batch.commit();
799
- this.toastr.success('Item excluído com sucesso!');
800
- return true;
801
- }
802
- catch (e) {
803
- const error = e;
804
- console.error('Erro ao deletar item:', error);
805
- this.toastr.error('Erro ao deletar item.');
806
- return false;
807
- }
808
- }
809
- async reindex(index, col, batch) {
810
- try {
811
- const snapshot = (await firstValueFrom(this.ngFire.collection(col).get()));
812
- const docs = snapshot.docs;
813
- for (let doc of docs) {
814
- const data = doc.data();
815
- if (data &&
816
- typeof data['index'] === 'number' &&
817
- data['index'] > index) {
818
- data['index'] = (data['index'] - 1);
819
- const docRef = this.ngFire.collection(col).doc(doc.id).ref;
820
- batch.update(docRef, data);
821
- }
822
- }
823
- }
824
- catch (error) {
825
- console.error('Erro ao reindexar:', error);
826
- }
827
- return;
828
- }
829
- dateFormatValidator() {
830
- return (control) => {
831
- if (!control.value) {
832
- return null;
833
- }
834
- const dateStr = control.value.trim();
835
- const datePattern = /^(\d{1,2})\/(\d{1,2})\/(\d{4})$/;
836
- if (!datePattern.test(dateStr)) {
837
- return { invalidFormat: true };
838
- }
839
- const parts = dateStr.split('/');
840
- const day = parts[0].padStart(2, '0');
841
- const month = parts[1].padStart(2, '0');
842
- const year = parts[2];
843
- const normalizedDate = `${day}/${month}/${year}`;
844
- const date = moment(normalizedDate, 'DD/MM/YYYY', true);
845
- if (!date.isValid()) {
846
- return { invalidDate: true };
847
- }
848
- return null;
849
- };
850
- }
851
- async updateIndex(index, id, col) {
852
- await this.ngFire.collection(col).doc(id).update({ index });
853
- }
854
- /**
855
- * Extrai o link de criação de índice da mensagem de erro do Firestore
856
- */
857
- extractIndexLink(error) {
858
- if (!error ||
859
- typeof error !== 'object' ||
860
- !('message' in error) ||
861
- typeof error.message !== 'string')
862
- return null;
863
- const linkMatch = error.message.match(/(https:\/\/console\.firebase\.google\.com\/[^\s]+)/);
864
- return linkMatch ? linkMatch[1] : null;
865
- }
866
- /**
867
- * Rastreia índices ausentes ao usar fallback preventivo
868
- */
869
- async trackMissingIndexPreventive(collection, arrange, conditions = undefined) {
870
- try {
871
- const querySignature = this.generateQuerySignature(collection, arrange, conditions);
872
- const docId = `${collection}_${querySignature}`;
873
- const indexLink = this.generateIndexLink(collection, arrange, conditions);
874
- const indexInstructions = this.generateIndexInstructions(collection, arrange, conditions);
875
- const trackingData = {
876
- collection,
877
- indexLink,
878
- indexInstructions,
879
- arrange: {
880
- sortBy: arrange.sortBy,
881
- filters: arrange.filters?.map((f) => ({
882
- arrange: f.arrange,
883
- property: f.filter?.property || null,
884
- dateField: f.arrange === 'filterByDate' ? arrange.sortBy?.field : null,
885
- })) || [],
886
- },
887
- conditions: conditions?.map((c) => ({
888
- property: c.firestoreProperty,
889
- operator: c.operator,
890
- })) || [],
891
- errorMessage: `Fallback preventivo usado para a collection ${collection}. A query exigiria índice composto.`,
892
- updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
893
- };
894
- const docRef = this.ngFire.collection('missingIndexes').doc(docId);
895
- const doc = await docRef.get().toPromise();
896
- if (doc && doc.exists) {
897
- await docRef.update({
898
- count: firebase.firestore.FieldValue.increment(1),
899
- updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
900
- lastError: trackingData['errorMessage'],
901
- });
902
- }
903
- else {
904
- await docRef.set({
905
- ...trackingData,
906
- count: 1,
907
- createdAt: firebase.firestore.FieldValue.serverTimestamp(),
908
- });
909
- }
910
- }
911
- catch (trackingError) {
912
- console.warn('Falha ao rastrear fallback preventivo:', trackingError);
913
- }
914
- }
915
- /**
916
- * Gera uma assinatura única para uma query
917
- */
918
- generateQuerySignature(collection, arrange, conditions = undefined) {
919
- const signature = {
920
- collection,
921
- sortBy: arrange.sortBy,
922
- filters: arrange.filters?.map((f) => ({
923
- arrange: f.arrange,
924
- property: f.filter?.property || null,
925
- })) || [],
926
- conditions: conditions?.map((c) => ({
927
- property: c.firestoreProperty,
928
- operator: c.operator,
929
- })) || [],
930
- };
931
- return btoa(JSON.stringify(signature))
932
- .replace(/[^a-zA-Z0-9]/g, '')
933
- .substring(0, 20);
934
- }
935
- /**
936
- * Gera instruções claras para criar o índice manualmente
937
- */
938
- generateIndexInstructions(collection, arrange, conditions = undefined) {
939
- const instructions = {
940
- summary: '',
941
- collection: collection,
942
- fields: [],
943
- queryExample: '',
944
- stepByStep: [],
945
- notes: [],
946
- };
947
- const fields = [];
948
- if (conditions && conditions.length > 0) {
949
- conditions.forEach((condition) => {
950
- if (condition.firestoreProperty) {
951
- fields.push({
952
- field: condition.firestoreProperty,
953
- order: 'Ascending',
954
- type: 'WHERE clause',
955
- operator: condition.operator,
956
- description: `Filtrar por ${condition.firestoreProperty} usando operador ${condition.operator}`,
957
- });
958
- }
959
- });
960
- }
961
- if (arrange.filters && arrange.filters.length > 0) {
962
- arrange.filters.forEach((filter) => {
963
- if (filter.filter?.property) {
964
- fields.push({
965
- field: filter.filter.property,
966
- order: 'Ascending',
967
- type: 'WHERE clause (filter)',
968
- operator: filter.arrange === 'filter' ? 'CONTAINS' : 'RANGE',
969
- description: `Filtrar por ${filter.filter.property} usando filtro ${filter.arrange}`,
970
- });
971
- }
972
- });
973
- }
974
- if (arrange.sortBy?.field) {
975
- fields.push({
976
- field: arrange.sortBy.field,
977
- order: arrange.sortBy.order === 'desc' ? 'Descending' : 'Ascending',
978
- type: 'ORDER BY clause',
979
- operator: 'N/A',
980
- description: `Ordenar resultados por ${arrange.sortBy.field} em ordem ${arrange.sortBy.order}`,
981
- });
982
- }
983
- instructions.fields = fields;
984
- const fieldNames = fields.map((f) => f.field).join(' + ');
985
- instructions.summary = `Criar índice composto para ${collection}: ${fieldNames}`;
986
- let queryExample = `db.collection('${collection}')`;
987
- fields.forEach((field, index) => {
988
- if (field.type.includes('WHERE')) {
989
- if (field.operator === '==') {
990
- queryExample += `\n .where('${field.field}', '==', 'value')`;
991
- }
992
- else if (field.operator === 'CONTAINS') {
993
- queryExample += `\n .where('${field.field}', '>=', 'searchText')`;
994
- }
995
- else {
996
- queryExample += `\n .where('${field.field}', '${field.operator}', 'value')`;
997
- }
998
- }
999
- });
1000
- const orderByField = fields.find((f) => f.type.includes('ORDER BY'));
1001
- if (orderByField) {
1002
- queryExample += `\n .orderBy('${orderByField.field}', '${orderByField.order.toLowerCase()}')`;
1003
- }
1004
- instructions.queryExample = queryExample;
1005
- instructions.stepByStep = [
1006
- '1. Ir para Firebase Console → Firestore → Indexes',
1007
- '2. Clicar em "Create Index"',
1008
- `3. Definir Collection ID: ${collection}`,
1009
- '4. Configurar campos nesta ORDEM EXATA:',
1010
- ...fields.map((field, index) => ` ${index + 1}. Campo: ${field.field}, Order: ${field.order}, Array: No`),
1011
- '5. Definir Query scopes: Collection',
1012
- '6. Clicar em "Create" e aguardar conclusão',
1013
- ];
1014
- instructions.notes = [
1015
- '⚠️ A ordem dos campos é CRÍTICA - deve corresponder exatamente à ordem da query',
1016
- '⚠️ As cláusulas WHERE devem vir ANTES do campo ORDER BY',
1017
- '⚠️ Este índice só funcionará para queries com esta combinação EXATA de campos',
1018
- '⚠️ A criação do índice pode levar vários minutos',
1019
- ];
1020
- return instructions;
1021
- }
1022
- /**
1023
- * Gera um link de índice baseado na estrutura da query
1024
- */
1025
- generateIndexLink(collection, arrange, conditions = undefined) {
1026
- try {
1027
- const indexFields = [];
1028
- if (conditions && conditions.length > 0) {
1029
- conditions.forEach((condition) => {
1030
- if (condition.firestoreProperty) {
1031
- indexFields.push(condition.firestoreProperty);
1032
- }
1033
- });
1034
- }
1035
- if (arrange.filters && arrange.filters.length > 0) {
1036
- arrange.filters.forEach((filter) => {
1037
- if (filter.filter?.property) {
1038
- indexFields.push(filter.filter.property);
1039
- }
1040
- });
1041
- }
1042
- if (arrange.sortBy?.field) {
1043
- indexFields.push(arrange.sortBy.field);
1044
- }
1045
- if (indexFields.length > 1) {
1046
- // Obter o projectId dinamicamente do Firebase
1047
- const options = this.ngFire?.firestore?.app?.options;
1048
- const projectId = options?.['projectId'];
1049
- if (!projectId) {
1050
- console.warn('Não foi possível obter o projectId do Firebase');
1051
- return null;
1052
- }
1053
- const baseUrl = `https://console.firebase.google.com/project/${projectId}/firestore/indexes`;
1054
- const queryParams = new URLSearchParams({
1055
- create_composite: `collection=${collection}&fields=${indexFields.join(',')}`,
1056
- });
1057
- const finalLink = `${baseUrl}?${queryParams.toString()}`;
1058
- return finalLink;
1059
- }
1060
- return null;
1061
- }
1062
- catch (error) {
1063
- console.warn('Falha ao gerar link de índice:', error);
1064
- return null;
1065
- }
1066
- }
1067
- async trackMissingIndex(error, collection, arrange, conditions = undefined) {
1068
- try {
1069
- const indexLink = this.extractIndexLink(error);
1070
- if (!indexLink)
1071
- return;
1072
- const linkHash = btoa(indexLink)
1073
- .replace(/[^a-zA-Z0-9]/g, '')
1074
- .substring(0, 20);
1075
- const docId = `${collection}_${linkHash}`;
1076
- const indexInstructions = this.generateIndexInstructions(collection, arrange, conditions);
1077
- const trackingData = {
1078
- collection,
1079
- indexLink,
1080
- indexInstructions,
1081
- arrange: {
1082
- sortBy: arrange.sortBy,
1083
- filters: arrange.filters?.map((f) => ({
1084
- arrange: f.arrange,
1085
- property: f.filter?.property || null,
1086
- dateField: f.arrange === 'filterByDate' ? arrange.sortBy?.field : null,
1087
- })) || [],
1088
- },
1089
- conditions: conditions?.map((c) => ({
1090
- property: c.firestoreProperty,
1091
- operator: c.operator,
1092
- })) || [],
1093
- errorMessage: error &&
1094
- typeof error === 'object' &&
1095
- 'message' in error &&
1096
- typeof error.message === 'string'
1097
- ? error.message
1098
- : String(error),
1099
- updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
1100
- };
1101
- const docRef = this.ngFire.collection('missingIndexes').doc(docId);
1102
- const doc = await docRef.get().toPromise();
1103
- const errorMessage = error &&
1104
- typeof error === 'object' &&
1105
- 'message' in error &&
1106
- typeof error.message === 'string'
1107
- ? error.message
1108
- : String(error);
1109
- if (doc && doc.exists) {
1110
- await docRef.update({
1111
- count: firebase.firestore.FieldValue.increment(1),
1112
- updatedAt: firebase.firestore.FieldValue.serverTimestamp(),
1113
- lastError: errorMessage,
1114
- });
1115
- }
1116
- else {
1117
- await docRef.set({
1118
- ...trackingData,
1119
- count: 1,
1120
- createdAt: firebase.firestore.FieldValue.serverTimestamp(),
1121
- });
1122
- }
1123
- }
1124
- catch (trackingError) {
1125
- console.warn('Falha ao rastrear índice ausente:', trackingError);
1126
- }
1127
- }
1128
- }
1129
- TableService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableService, deps: [{ token: i1.AngularFirestore, optional: true }, { token: i2.MatDialog, optional: true }, { token: i3.ToastrService, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
1130
- TableService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableService });
1131
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableService, decorators: [{
1132
- type: Injectable
1133
- }], ctorParameters: function () { return [{ type: i1.AngularFirestore, decorators: [{
1134
- type: Optional
1135
- }] }, { type: i2.MatDialog, decorators: [{
1136
- type: Optional
1137
- }] }, { type: i3.ToastrService, decorators: [{
1138
- type: Optional
1139
- }] }]; } });
1140
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.service.js","sourceRoot":"","sources":["../../../../../projects/ng-firebase-table-kxp/src/lib/services/table.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Optional } from '@angular/core';\r\nimport {\r\n  AngularFirestore,\r\n  CollectionReference,\r\n  QueryDocumentSnapshot,\r\n} from '@angular/fire/compat/firestore';\r\nimport firebase from 'firebase/compat/app';\r\nimport { Arrange, Condition, Pagination } from '../types/Table';\r\nimport { firstValueFrom } from 'rxjs';\r\nimport { MatDialog } from '@angular/material/dialog';\r\nimport { ToastrService } from 'ngx-toastr';\r\nimport { AbstractControl, ValidatorFn } from '@angular/forms';\r\nimport * as moment from 'moment';\r\n\r\ninterface ErrorWithCode extends Error {\r\n  code: string;\r\n}\r\n\r\ninterface ErrorWithMessage {\r\n  message: string;\r\n}\r\n\r\ninterface PaginationResult {\r\n  items: Array<Record<string, unknown> & { id: string }>;\r\n  filterLength: number | null;\r\n  firstDoc: firebase.firestore.QueryDocumentSnapshot<unknown> | null;\r\n  lastDoc: firebase.firestore.QueryDocumentSnapshot<unknown> | null;\r\n  hasNextPage: boolean;\r\n  hasPreviousPage?: boolean;\r\n  currentClientPageIndex?: number;\r\n  totalPages?: number;\r\n}\r\n\r\n@Injectable()\r\nexport class TableService {\r\n  constructor(\r\n    @Optional() private ngFire: AngularFirestore,\r\n    @Optional() private dialog: MatDialog,\r\n    @Optional() private toastr: ToastrService\r\n  ) {}\r\n\r\n  async getItems(\r\n    collection: CollectionReference<unknown>\r\n  ): Promise<Array<Record<string, unknown> & { id: string }>> {\r\n    try {\r\n      const querySnapshot = await collection.get();\r\n      return querySnapshot.docs.map(\r\n        (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) => {\r\n          const data = doc.data() as Record<string, unknown> | undefined;\r\n          return { ...(data ?? {}), id: doc.id } as Record<string, unknown> & {\r\n            id: string;\r\n          };\r\n        }\r\n      );\r\n    } catch (error) {\r\n      console.warn('Collection não encontrada:', error);\r\n      return [];\r\n    }\r\n  }\r\n\r\n  private async executeQuery(params: Pagination): Promise<PaginationResult> {\r\n    if (params.filterFn) {\r\n      // Lógica com filtro no cliente (filterFn)\r\n      const BATCH_FETCH_SIZE = params.batchSize;\r\n      const GOAL_SIZE = params.batchSize + 1;\r\n\r\n      if (params.navigation === 'forward' || params.navigation === 'reload') {\r\n        if (params.navigation === 'reload' && params.doc) {\r\n          params.doc.lastDoc = null;\r\n        }\r\n\r\n        let lastDocCursor: firebase.firestore.QueryDocumentSnapshot<unknown> | null =\r\n          params.doc ? params.doc.lastDoc : null;\r\n        let pageResults: Array<Record<string, unknown> & { id: string }> = [];\r\n        let allFetchedDocs: firebase.firestore.QueryDocumentSnapshot<unknown>[] =\r\n          [];\r\n        let hasMoreDocsInDb = true;\r\n\r\n        while (pageResults.length < GOAL_SIZE && hasMoreDocsInDb) {\r\n          let query: firebase.firestore.Query<unknown> = this.ngFire.collection(\r\n            params.collection\r\n          ).ref;\r\n          query = this.applyFilters(query, params.arrange, params.conditions);\r\n\r\n          if (lastDocCursor) {\r\n            query = query.startAfter(lastDocCursor);\r\n          }\r\n          query = query.limit(BATCH_FETCH_SIZE);\r\n\r\n          const snapshot = await query.get();\r\n\r\n          if (snapshot.empty) {\r\n            hasMoreDocsInDb = false;\r\n            break;\r\n          }\r\n\r\n          lastDocCursor = snapshot.docs[snapshot.docs.length - 1];\r\n          allFetchedDocs.push(...snapshot.docs);\r\n\r\n          const batchUsers = snapshot.docs\r\n            .map((doc: firebase.firestore.QueryDocumentSnapshot<unknown>) => {\r\n              const data = doc.data() as Record<string, unknown> | undefined;\r\n              return { id: doc.id, ...(data ?? {}) } as Record<\r\n                string,\r\n                unknown\r\n              > & { id: string };\r\n            })\r\n            .filter(params.filterFn);\r\n\r\n          pageResults.push(...batchUsers);\r\n\r\n          if (snapshot.size < BATCH_FETCH_SIZE) {\r\n            hasMoreDocsInDb = false;\r\n          }\r\n        }\r\n\r\n        const hasNextPage = pageResults.length > params.batchSize;\r\n        const finalItems = pageResults.slice(0, params.batchSize);\r\n\r\n        const firstDocOfPage =\r\n          allFetchedDocs.find(\r\n            (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) =>\r\n              doc.id === finalItems[0]?.id\r\n          ) || null;\r\n        const lastDocOfPage =\r\n          allFetchedDocs.find(\r\n            (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) =>\r\n              doc.id === finalItems[finalItems.length - 1]?.id\r\n          ) || null;\r\n\r\n        return {\r\n          items: finalItems,\r\n          filterLength: null,\r\n          firstDoc: firstDocOfPage,\r\n          lastDoc: lastDocOfPage,\r\n          hasNextPage: hasNextPage,\r\n          hasPreviousPage:\r\n            !!(params.doc && params.doc.lastDoc) &&\r\n            params.navigation !== 'reload',\r\n          currentClientPageIndex: undefined,\r\n        };\r\n      }\r\n      // Lógica para trás (backward)\r\n      else if (params.navigation === 'backward') {\r\n        if (!params.doc || !params.doc.firstDoc) {\r\n          return {\r\n            items: [],\r\n            filterLength: null,\r\n            firstDoc: null,\r\n            lastDoc: null,\r\n            hasNextPage: true,\r\n            hasPreviousPage: false,\r\n            currentClientPageIndex: undefined,\r\n          };\r\n        }\r\n\r\n        let pageResults: Array<Record<string, unknown> & { id: string }> = [];\r\n        let allFetchedDocs: firebase.firestore.QueryDocumentSnapshot<unknown>[] =\r\n          [];\r\n        let hasMoreDocsInDb = true;\r\n\r\n        let boundaryDoc: firebase.firestore.QueryDocumentSnapshot<unknown> | null =\r\n          params.doc.firstDoc;\r\n\r\n        while (pageResults.length < GOAL_SIZE && hasMoreDocsInDb) {\r\n          if (!boundaryDoc) break;\r\n\r\n          let query: firebase.firestore.Query<unknown> = this.ngFire.collection(\r\n            params.collection\r\n          ).ref;\r\n\r\n          query = this.applyFilters(query, params.arrange, params.conditions);\r\n\r\n          query = query.endBefore(boundaryDoc);\r\n          query = query.limitToLast(BATCH_FETCH_SIZE);\r\n\r\n          const snapshot = await query.get();\r\n\r\n          if (snapshot.empty) {\r\n            hasMoreDocsInDb = false;\r\n            break;\r\n          }\r\n\r\n          boundaryDoc = snapshot\r\n            .docs[0] as firebase.firestore.QueryDocumentSnapshot<unknown>;\r\n\r\n          allFetchedDocs = [...snapshot.docs, ...allFetchedDocs];\r\n\r\n          const batchUsers = snapshot.docs\r\n            .map((doc: firebase.firestore.QueryDocumentSnapshot<unknown>) => {\r\n              const data = doc.data() as Record<string, unknown> | undefined;\r\n              return { id: doc.id, ...(data ?? {}) } as Record<\r\n                string,\r\n                unknown\r\n              > & { id: string };\r\n            })\r\n            .filter(params.filterFn);\r\n\r\n          pageResults = [...batchUsers, ...pageResults];\r\n\r\n          if (snapshot.size < BATCH_FETCH_SIZE) {\r\n            hasMoreDocsInDb = false;\r\n          }\r\n        }\r\n\r\n        const finalItems = pageResults.slice(0, params.batchSize);\r\n\r\n        const firstDocOfPage =\r\n          allFetchedDocs.find(\r\n            (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) =>\r\n              doc.id === finalItems[0]?.id\r\n          ) || null;\r\n        const lastDocOfPage =\r\n          allFetchedDocs.find(\r\n            (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) =>\r\n              doc.id === finalItems[finalItems.length - 1]?.id\r\n          ) || null;\r\n\r\n        return {\r\n          items: finalItems,\r\n          filterLength: null,\r\n          firstDoc: firstDocOfPage,\r\n          lastDoc: lastDocOfPage,\r\n          hasNextPage: true,\r\n          currentClientPageIndex: undefined,\r\n        };\r\n      }\r\n    } else {\r\n      let items: Array<Record<string, unknown> & { id: string }> = [];\r\n      let docs: firebase.firestore.QueryDocumentSnapshot<unknown>[] = [];\r\n      let hasNextPage = false;\r\n      let filterLength: null | number = null;\r\n\r\n      let query: firebase.firestore.Query<unknown> = this.ngFire.collection(\r\n        params.collection\r\n      ).ref;\r\n\r\n      if (params.conditions) {\r\n        params.conditions.forEach((c: Condition) => {\r\n          if (c.operator === '!=') {\r\n            query = query.orderBy(c.firestoreProperty);\r\n          }\r\n        });\r\n      }\r\n\r\n      query = this.applyFilters(query, params.arrange, params.conditions);\r\n\r\n      if (params.navigation === 'reload') {\r\n        query = query.limit(params.batchSize + 1);\r\n        if (params.doc && params.doc.firstDoc) {\r\n          query = query.startAt(params.doc.firstDoc);\r\n        }\r\n      } else if (params.navigation === 'forward') {\r\n        query = query.limit(params.batchSize + 1);\r\n\r\n        if (params.doc && params.doc.lastDoc) {\r\n          query = query.startAfter(params.doc.lastDoc);\r\n        }\r\n      } else {\r\n        // backward\r\n        query = query.limitToLast(params.batchSize + 1);\r\n        if (params.doc && params.doc.firstDoc) {\r\n          query = query.endBefore(params.doc.firstDoc);\r\n        }\r\n      }\r\n\r\n      const itemCol = await query.get();\r\n      itemCol.docs.forEach(\r\n        (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) =>\r\n          docs.push(doc)\r\n      );\r\n      const itemPromises = docs.map(\r\n        async (item: firebase.firestore.QueryDocumentSnapshot<unknown>) => {\r\n          const itemData = item.data() as Record<string, unknown> | undefined;\r\n          items.push({ id: item.id, ...(itemData ?? {}) } as Record<\r\n            string,\r\n            unknown\r\n          > & { id: string });\r\n        }\r\n      );\r\n\r\n      let lastDoc: firebase.firestore.QueryDocumentSnapshot<unknown> | null =\r\n        docs[docs.length - 1] || null;\r\n      let firstDoc: firebase.firestore.QueryDocumentSnapshot<unknown> | null =\r\n        docs[0];\r\n\r\n      if (\r\n        (items.length > params.batchSize && params.navigation === 'forward') ||\r\n        (params.navigation === 'reload' && items.length > params.batchSize)\r\n      ) {\r\n        lastDoc = docs[docs.length - 2] || null;\r\n        items.pop();\r\n        hasNextPage = true;\r\n      }\r\n      if (items.length > params.batchSize && params.navigation === 'backward') {\r\n        firstDoc = docs[1];\r\n        items.shift();\r\n        hasNextPage = true;\r\n      }\r\n\r\n      await Promise.all(itemPromises);\r\n      return {\r\n        items,\r\n        filterLength,\r\n        lastDoc,\r\n        firstDoc,\r\n        hasNextPage,\r\n        currentClientPageIndex: undefined,\r\n      };\r\n    }\r\n\r\n    // Fallback para garantir que sempre retornamos algo\r\n    return {\r\n      items: [],\r\n      filterLength: null,\r\n      firstDoc: null,\r\n      lastDoc: null,\r\n      hasNextPage: false,\r\n      currentClientPageIndex: undefined,\r\n    };\r\n  }\r\n\r\n  applyFilters(\r\n    query: firebase.firestore.Query<unknown>,\r\n    arrange: Arrange,\r\n    conditions: Condition[] | undefined\r\n  ): firebase.firestore.Query<unknown> {\r\n    if (conditions) {\r\n      conditions.map((cond: Condition) => {\r\n        query = query.where(\r\n          cond.firestoreProperty,\r\n          cond.operator,\r\n          cond.dashProperty\r\n        );\r\n      });\r\n    }\r\n\r\n    let hasFilterSpecificOrderBy = false;\r\n    let appliedOrderByField: string | null = null;\r\n\r\n    const equalsFilters = arrange.filters.filter(\r\n      (\r\n        f\r\n      ): f is Arrange['filters'][0] & {\r\n        arrange: 'equals';\r\n        filter: { property: string; filtering: string };\r\n      } => f.arrange === 'equals' && !!f.filter\r\n    );\r\n    const otherFilters = arrange.filters.filter(\r\n      (f): f is Arrange['filters'][0] => f.arrange !== 'equals'\r\n    );\r\n\r\n    const equalsGroupedByProperty = equalsFilters.reduce(\r\n      (acc: Record<string, string[]>, current) => {\r\n        const prop = current.filter!.property;\r\n        if (!acc[prop]) {\r\n          acc[prop] = [];\r\n        }\r\n        acc[prop].push(current.filter!.filtering);\r\n        return acc;\r\n      },\r\n      {} as Record<string, string[]>\r\n    );\r\n\r\n    for (const prop in equalsGroupedByProperty) {\r\n      const values = equalsGroupedByProperty[prop];\r\n      if (values.length > 0) {\r\n        query = query.where(prop, 'in', values);\r\n      }\r\n    }\r\n\r\n    otherFilters.forEach((filterItem: Arrange['filters'][0]) => {\r\n      // Aplicar filtragem por busca\r\n      if (\r\n        filterItem.filter?.filtering &&\r\n        filterItem.filter?.property !== '' &&\r\n        filterItem.arrange === 'filter'\r\n      ) {\r\n        query = query\r\n          .where(\r\n            filterItem.filter.property,\r\n            '>=',\r\n            filterItem.filter.filtering.trim().toUpperCase()\r\n          )\r\n          .where(\r\n            filterItem.filter.property,\r\n            '<=',\r\n            filterItem.filter.filtering.trim().toUpperCase() + '\\uf8ff'\r\n          );\r\n        if (!hasFilterSpecificOrderBy) {\r\n          query = query.orderBy(filterItem.filter.property);\r\n          hasFilterSpecificOrderBy = true;\r\n          appliedOrderByField = filterItem.filter.property;\r\n        }\r\n      }\r\n\r\n      // Aplicar filtro do tipo \"filterByDate\"\r\n      if (filterItem.dateFilter && filterItem.arrange === 'filterByDate') {\r\n        query = query\r\n          .where(arrange.sortBy.field, '>=', filterItem.dateFilter.initial)\r\n          .where(arrange.sortBy.field, '<=', filterItem.dateFilter.final);\r\n        if (!hasFilterSpecificOrderBy) {\r\n          query = query.orderBy(arrange.sortBy.field);\r\n          hasFilterSpecificOrderBy = true;\r\n          appliedOrderByField = arrange.sortBy.field;\r\n        }\r\n      }\r\n    });\r\n\r\n    // Aplicar sortBy\r\n    if (arrange.sortBy && arrange.sortBy.field && arrange.sortBy.order) {\r\n      if (appliedOrderByField !== arrange.sortBy.field) {\r\n        query = query.orderBy(arrange.sortBy.field, arrange.sortBy.order);\r\n      }\r\n    }\r\n    return query;\r\n  }\r\n\r\n  private getIdFilter(params: Pagination): string | null {\r\n    if (!params.arrange?.filters) return null;\r\n\r\n    const idFilter = params.arrange.filters.find(\r\n      (\r\n        f\r\n      ): f is Arrange['filters'][0] & {\r\n        arrange: 'filter';\r\n        filter: { property: string; filtering: string };\r\n      } =>\r\n        f.arrange === 'filter' &&\r\n        !!f.filter &&\r\n        f.filter.property === 'id' &&\r\n        !!f.filter.filtering\r\n    );\r\n\r\n    return idFilter?.filter?.filtering?.trim() || null;\r\n  }\r\n\r\n  private async getDocumentById(\r\n    collection: string,\r\n    docId: string\r\n  ): Promise<(Record<string, unknown> & { id: string }) | null> {\r\n    try {\r\n      const docRef = this.ngFire.collection(collection).doc(docId);\r\n      const docSnapshot = await docRef.get().toPromise();\r\n\r\n      if (docSnapshot && docSnapshot.exists) {\r\n        const data = docSnapshot.data() as Record<string, unknown> | undefined;\r\n        return {\r\n          id: docSnapshot.id,\r\n          ...(data ?? {}),\r\n        } as Record<string, unknown> & { id: string };\r\n      }\r\n      return null;\r\n    } catch (error) {\r\n      console.warn('Erro ao buscar documento por ID:', error);\r\n      return null;\r\n    }\r\n  }\r\n\r\n  private async searchByIdPartial(\r\n    params: Pagination,\r\n    searchTerm: string\r\n  ): Promise<PaginationResult> {\r\n    const exactMatch = await this.getDocumentById(\r\n      params.collection,\r\n      searchTerm\r\n    );\r\n\r\n    if (exactMatch) {\r\n      if (params.conditions) {\r\n        const operators = this.operators;\r\n        const passesConditions = params.conditions.every((cond) => {\r\n          const operatorFn = operators[cond.operator as keyof typeof operators];\r\n          return operatorFn\r\n            ? operatorFn(exactMatch[cond.firestoreProperty], cond.dashProperty)\r\n            : false;\r\n        });\r\n\r\n        if (!passesConditions) {\r\n          return {\r\n            items: [],\r\n            filterLength: 0,\r\n            firstDoc: null,\r\n            lastDoc: null,\r\n            hasNextPage: false,\r\n            hasPreviousPage: false,\r\n            currentClientPageIndex: 0,\r\n            totalPages: 0,\r\n          };\r\n        }\r\n      }\r\n\r\n      if (params.filterFn && !params.filterFn(exactMatch)) {\r\n        return {\r\n          items: [],\r\n          filterLength: 0,\r\n          firstDoc: null,\r\n          lastDoc: null,\r\n          hasNextPage: false,\r\n          hasPreviousPage: false,\r\n          currentClientPageIndex: 0,\r\n          totalPages: 0,\r\n        };\r\n      }\r\n\r\n      return {\r\n        items: [exactMatch],\r\n        filterLength: 1,\r\n        firstDoc: null,\r\n        lastDoc: null,\r\n        hasNextPage: false,\r\n        hasPreviousPage: false,\r\n        currentClientPageIndex: 0,\r\n        totalPages: 1,\r\n      };\r\n    }\r\n\r\n    const searchTermLower = searchTerm.toLowerCase();\r\n\r\n    const paramsWithoutIdFilter = {\r\n      ...params,\r\n      arrange: {\r\n        ...params.arrange,\r\n        filters: params.arrange.filters.filter(\r\n          (f) => !(f.arrange === 'filter' && f.filter?.property === 'id')\r\n        ),\r\n      },\r\n    };\r\n\r\n    let query: firebase.firestore.Query<unknown> = this.ngFire.collection(\r\n      params.collection\r\n    ).ref;\r\n\r\n    // Aplicar conditions\r\n    if (params.conditions) {\r\n      params.conditions.forEach((cond) => {\r\n        query = query.where(\r\n          cond.firestoreProperty,\r\n          cond.operator,\r\n          cond.dashProperty\r\n        );\r\n      });\r\n    }\r\n\r\n    // Aplicar sortBy\r\n    if (params.arrange?.sortBy?.field && params.arrange?.sortBy?.order) {\r\n      query = query.orderBy(\r\n        params.arrange.sortBy.field,\r\n        params.arrange.sortBy.order\r\n      );\r\n    }\r\n\r\n    const snapshot = await query.get();\r\n    let items: Array<Record<string, unknown> & { id: string }> = snapshot.docs\r\n      .map((doc) => {\r\n        const data = doc.data() as Record<string, unknown> | undefined;\r\n        return { id: doc.id, ...(data ?? {}) } as Record<string, unknown> & {\r\n          id: string;\r\n        };\r\n      })\r\n      .filter((item) => item.id.toLowerCase().includes(searchTermLower));\r\n\r\n    // Separar equals filters e outros filtros\r\n    const equalsFilters = paramsWithoutIdFilter.arrange.filters.filter(\r\n      (\r\n        f\r\n      ): f is Arrange['filters'][0] & {\r\n        arrange: 'equals';\r\n        filter: { property: string; filtering: string };\r\n      } => f.arrange === 'equals' && !!f.filter\r\n    );\r\n    const otherFilters = paramsWithoutIdFilter.arrange.filters.filter(\r\n      (f): f is Arrange['filters'][0] =>\r\n        f.arrange !== 'equals' &&\r\n        (f.arrange !== 'filter' || f.filter?.property !== 'id')\r\n    );\r\n\r\n    // Aplicar equals filters com lógica OR dentro de cada propriedade\r\n    if (equalsFilters.length > 0) {\r\n      // Agrupar por propriedade para aplicar OR\r\n      const groupedByProperty = equalsFilters.reduce(\r\n        (acc: Record<string, (string | boolean)[]>, f) => {\r\n          const prop = f.filter.property;\r\n          if (!acc[prop]) {\r\n            acc[prop] = [];\r\n          }\r\n          acc[prop].push(f.filter.filtering);\r\n          return acc;\r\n        },\r\n        {} as Record<string, (string | boolean)[]>\r\n      );\r\n\r\n      // Filtrar: item deve ter pelo menos um valor de CADA propriedade\r\n      items = items.filter((item) => {\r\n        return Object.entries(groupedByProperty).every(([prop, values]) => {\r\n          const itemValue = item[prop];\r\n          return values.includes(itemValue as string | boolean);\r\n        });\r\n      });\r\n    }\r\n\r\n    // Aplicar outros filtros\r\n    otherFilters.forEach((filterItem) => {\r\n      if (\r\n        filterItem.arrange === 'filter' &&\r\n        filterItem.filter?.filtering &&\r\n        filterItem.filter?.property\r\n      ) {\r\n        const filterValue = String(filterItem.filter.filtering)\r\n          .trim()\r\n          .toLowerCase();\r\n        items = items.filter((item) => {\r\n          const itemValue = String(\r\n            item[filterItem.filter!.property] ?? ''\r\n          ).toLowerCase();\r\n          return itemValue.includes(filterValue);\r\n        });\r\n      }\r\n    });\r\n\r\n    // Aplicar filterFn se existir\r\n    if (params.filterFn) {\r\n      items = items.filter(params.filterFn);\r\n    }\r\n\r\n    // Paginação\r\n    const pageSize = params.batchSize;\r\n    let currentClientPageIndex = 0;\r\n\r\n    if (params.navigation === 'reload') {\r\n      currentClientPageIndex = 0;\r\n    } else {\r\n      // Usar o índice passado pelo componente sem incrementar/decrementar\r\n      currentClientPageIndex = params.clientPageIndex || 0;\r\n    }\r\n\r\n    const startIndex = currentClientPageIndex * pageSize;\r\n    const endIndex = startIndex + pageSize;\r\n    const paginatedItems = items.slice(startIndex, endIndex);\r\n\r\n    const totalPages = Math.ceil(items.length / pageSize);\r\n    const hasNextPage = currentClientPageIndex < totalPages - 1;\r\n    const hasPreviousPage = currentClientPageIndex > 0;\r\n\r\n    return {\r\n      items: paginatedItems,\r\n      filterLength: items.length,\r\n      firstDoc: null,\r\n      lastDoc: null,\r\n      hasNextPage,\r\n      hasPreviousPage,\r\n      currentClientPageIndex,\r\n      totalPages,\r\n    };\r\n  }\r\n\r\n  private shouldUseClientSideFallback(params: Pagination): boolean {\r\n    const hasConditions = params.conditions && params.conditions.length > 0;\r\n    const hasArrangeFilters =\r\n      params.arrange?.filters && params.arrange.filters.length > 0;\r\n    const hasSortBy = params.arrange?.sortBy?.field;\r\n\r\n    if (params.filterFn) {\r\n      return false;\r\n    }\r\n\r\n    if (hasArrangeFilters) {\r\n      const equalsFilters = params.arrange.filters.filter(\r\n        (\r\n          f\r\n        ): f is Arrange['filters'][0] & {\r\n          arrange: 'equals';\r\n          filter: { property: string; filtering: string };\r\n        } => f.arrange === 'equals' && !!f.filter\r\n      );\r\n\r\n      if (equalsFilters.length > 0) {\r\n        const propertiesSet = new Set(\r\n          equalsFilters.map((f) => f.filter.property)\r\n        );\r\n\r\n        if (propertiesSet.size > 1) {\r\n          return true;\r\n        }\r\n      }\r\n    }\r\n\r\n    if (hasConditions && hasArrangeFilters && hasSortBy) {\r\n      return true;\r\n    }\r\n\r\n    if (hasConditions && hasArrangeFilters) {\r\n      return true;\r\n    }\r\n\r\n    if (hasArrangeFilters && params.arrange.filters.length > 1 && hasSortBy) {\r\n      const equalsFilters = params.arrange.filters.filter(\r\n        (\r\n          f\r\n        ): f is Arrange['filters'][0] & {\r\n          arrange: 'equals';\r\n          filter: { property: string; filtering: string };\r\n        } => f.arrange === 'equals' && !!f.filter\r\n      );\r\n\r\n      if (equalsFilters.length > 0) {\r\n        const propertiesSet = new Set(\r\n          equalsFilters.map((f) => f.filter.property)\r\n        );\r\n        if (\r\n          propertiesSet.size === 1 &&\r\n          equalsFilters.length === params.arrange.filters.length\r\n        ) {\r\n          return false;\r\n        }\r\n      }\r\n\r\n      return true;\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  async getPaginated(params: Pagination): Promise<PaginationResult> {\r\n    const idFilterValue = this.getIdFilter(params);\r\n    if (idFilterValue) {\r\n      const result = await this.searchByIdPartial(params, idFilterValue);\r\n      return result;\r\n    }\r\n\r\n    // Detectar preventivamente se deve usar fallback\r\n    if (this.shouldUseClientSideFallback(params)) {\r\n      await this.trackMissingIndexPreventive(\r\n        params.collection,\r\n        params.arrange,\r\n        params.conditions\r\n      );\r\n\r\n      const result = await this.executeClientSideQuery(params);\r\n      return result;\r\n    }\r\n\r\n    try {\r\n      const result = await this.executeQuery(params);\r\n      return result;\r\n    } catch (error: unknown) {\r\n      if (\r\n        error &&\r\n        typeof error === 'object' &&\r\n        'code' in error &&\r\n        (error as ErrorWithCode).code === 'failed-precondition'\r\n      ) {\r\n        await this.trackMissingIndex(\r\n          error,\r\n          params.collection,\r\n          params.arrange,\r\n          params.conditions\r\n        );\r\n\r\n        const result = await this.executeClientSideQuery(params);\r\n        return result;\r\n      } else if (\r\n        error &&\r\n        typeof error === 'object' &&\r\n        'code' in error &&\r\n        (error as ErrorWithCode).code === 'invalid-argument'\r\n      ) {\r\n        await this.trackMissingIndex(\r\n          error,\r\n          params.collection,\r\n          params.arrange,\r\n          params.conditions\r\n        );\r\n\r\n        const result = await this.executeClientSideQuery(params);\r\n        return result;\r\n      } else {\r\n        throw error;\r\n      }\r\n    }\r\n  }\r\n\r\n  async executeClientSideQuery(params: Pagination): Promise<PaginationResult> {\r\n    // Otimizar usando pelo menos uma cláusula .where() quando possível\r\n    let query: firebase.firestore.Query<unknown> = this.ngFire.collection(\r\n      params.collection\r\n    ).ref;\r\n\r\n    let appliedCondition: Condition | null = null;\r\n    let hasAppliedWhereClause = false;\r\n\r\n    // Primeiro, tenta aplicar condições simples\r\n    if (params.conditions && params.conditions.length > 0) {\r\n      const simpleCondition = params.conditions.find((cond: Condition) =>\r\n        ['==', '>', '<', '>=', '<=', 'in', 'array-contains'].includes(\r\n          cond.operator\r\n        )\r\n      );\r\n\r\n      if (simpleCondition) {\r\n        query = query.where(\r\n          simpleCondition.firestoreProperty,\r\n          simpleCondition.operator,\r\n          simpleCondition.dashProperty\r\n        );\r\n        appliedCondition = simpleCondition;\r\n        hasAppliedWhereClause = true;\r\n      }\r\n    }\r\n\r\n    // Se não há condições disponíveis, tenta aplicar filtros do arrange\r\n    let appliedFirestoreFilter: Arrange['filters'][0] | 'all-equals' | null =\r\n      null;\r\n    if (!hasAppliedWhereClause && params.arrange?.filters) {\r\n      // Agrupar equals filters por propriedade\r\n      const equalsFilters = params.arrange.filters.filter(\r\n        (\r\n          f\r\n        ): f is Arrange['filters'][0] & {\r\n          arrange: 'equals';\r\n          filter: { property: string; filtering: string };\r\n        } => f.arrange === 'equals' && !!f.filter\r\n      );\r\n\r\n      const equalsGroupedByProperty = equalsFilters.reduce(\r\n        (acc: Record<string, (string | boolean)[]>, current) => {\r\n          const prop = current.filter.property;\r\n          if (!acc[prop]) {\r\n            acc[prop] = [];\r\n          }\r\n          acc[prop].push(current.filter.filtering);\r\n          return acc;\r\n        },\r\n        {} as Record<string, (string | boolean)[]>\r\n      );\r\n\r\n      // Se há apenas UMA propriedade com múltiplos valores, aplicar no Firestore com 'in'\r\n      const properties = Object.keys(equalsGroupedByProperty);\r\n      if (properties.length === 1 && equalsFilters.length > 0) {\r\n        const prop = properties[0];\r\n        const values = equalsGroupedByProperty[prop];\r\n        query = query.where(prop, 'in', values);\r\n        hasAppliedWhereClause = true;\r\n        // Marcar TODOS os equals filters dessa propriedade como aplicados\r\n        appliedFirestoreFilter = 'all-equals';\r\n      } else if (properties.length === 0) {\r\n        const otherFilter = params.arrange.filters.find(\r\n          (\r\n            f\r\n          ): f is Arrange['filters'][0] &\r\n            (\r\n              | {\r\n                  arrange: 'filter';\r\n                  filter: { property: string; filtering: string };\r\n                }\r\n              | {\r\n                  arrange: 'filterByDate';\r\n                  dateFilter: { initial: Date; final: Date };\r\n                }\r\n            ) =>\r\n            (f.arrange === 'filter' &&\r\n              !!f.filter?.filtering &&\r\n              !!f.filter?.property) ||\r\n            (f.arrange === 'filterByDate' &&\r\n              !!f.dateFilter?.initial &&\r\n              !!f.dateFilter?.final)\r\n        );\r\n\r\n        if (otherFilter) {\r\n          if (otherFilter.arrange === 'filter' && otherFilter.filter) {\r\n            const filterValue = otherFilter.filter.filtering\r\n              .trim()\r\n              .toUpperCase();\r\n            query = query\r\n              .where(otherFilter.filter.property, '>=', filterValue)\r\n              .where(otherFilter.filter.property, '<=', filterValue + '\\uf8ff');\r\n            hasAppliedWhereClause = true;\r\n            appliedFirestoreFilter = otherFilter;\r\n          } else if (\r\n            otherFilter.arrange === 'filterByDate' &&\r\n            otherFilter.dateFilter &&\r\n            params.arrange.sortBy?.field\r\n          ) {\r\n            query = query\r\n              .where(\r\n                params.arrange.sortBy.field,\r\n                '>=',\r\n                otherFilter.dateFilter.initial\r\n              )\r\n              .where(\r\n                params.arrange.sortBy.field,\r\n                '<=',\r\n                otherFilter.dateFilter.final\r\n              );\r\n            hasAppliedWhereClause = true;\r\n            appliedFirestoreFilter = otherFilter;\r\n          }\r\n        }\r\n      }\r\n    }\r\n\r\n    const allDocsSnapshot = await query.get();\r\n    let items: Array<Record<string, unknown> & { id: string }> =\r\n      allDocsSnapshot.docs.map(\r\n        (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) => {\r\n          const data = doc.data() as Record<string, unknown> | undefined;\r\n          return { id: doc.id, ...(data ?? {}) } as Record<string, unknown> & {\r\n            id: string;\r\n          };\r\n        }\r\n      );\r\n\r\n    // Aplicar condições restantes\r\n    if (params.conditions) {\r\n      const remainingConditions = params.conditions.filter(\r\n        (cond: Condition) => cond !== appliedCondition\r\n      );\r\n      if (remainingConditions.length > 0) {\r\n        const operators = this.operators;\r\n        items = items.filter((item) => {\r\n          return remainingConditions.every((cond: Condition) => {\r\n            const operatorFn =\r\n              operators[cond.operator as keyof typeof operators];\r\n            return operatorFn\r\n              ? operatorFn(item[cond.firestoreProperty], cond.dashProperty)\r\n              : false;\r\n          });\r\n        });\r\n      }\r\n    }\r\n\r\n    const { filters, sortBy } = params.arrange;\r\n\r\n    // Track which filter was already applied in Firestore to avoid double filtering\r\n    if (hasAppliedWhereClause && !appliedCondition && params.arrange?.filters) {\r\n      const equalsFilter = params.arrange.filters.find(\r\n        (\r\n          f\r\n        ): f is Arrange['filters'][0] & {\r\n          arrange: 'equals';\r\n          filter: { property: string; filtering: string };\r\n        } => f.arrange === 'equals' && !!f.filter?.filtering\r\n      );\r\n      if (equalsFilter) {\r\n        appliedFirestoreFilter = equalsFilter;\r\n      } else {\r\n        const foundFilter = params.arrange.filters.find(\r\n          (\r\n            f\r\n          ): f is Arrange['filters'][0] &\r\n            (\r\n              | {\r\n                  arrange: 'filter';\r\n                  filter: { property: string; filtering: string };\r\n                }\r\n              | {\r\n                  arrange: 'filterByDate';\r\n                  dateFilter: { initial: Date; final: Date };\r\n                }\r\n            ) =>\r\n            (f.arrange === 'filter' &&\r\n              !!f.filter?.filtering &&\r\n              !!f.filter?.property) ||\r\n            (f.arrange === 'filterByDate' &&\r\n              !!f.dateFilter?.initial &&\r\n              !!f.dateFilter?.final)\r\n        );\r\n        if (foundFilter) {\r\n          appliedFirestoreFilter = foundFilter;\r\n        }\r\n      }\r\n    }\r\n\r\n    const equalsFilters = filters.filter(\r\n      (f): f is Arrange['filters'][0] & { arrange: 'equals' } =>\r\n        f.arrange === 'equals'\r\n    );\r\n    const otherFilters = filters.filter(\r\n      (f): f is Arrange['filters'][0] => f.arrange !== 'equals'\r\n    );\r\n\r\n    // Aplicar equals filters no client-side apenas se não foram aplicados no Firestore\r\n    if (appliedFirestoreFilter !== 'all-equals' && equalsFilters.length > 0) {\r\n      // Agrupar por propriedade para aplicar OR dentro de cada propriedade\r\n      const groupedByProperty = equalsFilters.reduce(\r\n        (acc: Record<string, (string | boolean)[]>, f) => {\r\n          if (f.filter) {\r\n            const prop = f.filter.property;\r\n            if (!acc[prop]) {\r\n              acc[prop] = [];\r\n            }\r\n            acc[prop].push(f.filter.filtering);\r\n          }\r\n          return acc;\r\n        },\r\n        {} as Record<string, (string | boolean)[]>\r\n      );\r\n\r\n      // Filtrar: item deve ter pelo menos um valor de CADA propriedade\r\n      // (AND entre propriedades, OR dentro de cada propriedade)\r\n      items = items.filter((item) => {\r\n        return Object.entries(groupedByProperty).every(([prop, values]) => {\r\n          const itemValue = item[prop];\r\n          return values.includes(itemValue as string | boolean);\r\n        });\r\n      });\r\n    }\r\n\r\n    otherFilters.forEach((filterItem) => {\r\n      if (appliedFirestoreFilter === filterItem) {\r\n        return;\r\n      }\r\n\r\n      if (\r\n        filterItem.arrange === 'filter' &&\r\n        filterItem.filter?.filtering &&\r\n        filterItem.filter?.property\r\n      ) {\r\n        const filterValue = String(filterItem.filter.filtering)\r\n          .trim()\r\n          .toLowerCase();\r\n        items = items.filter((item) => {\r\n          const itemValue = String(\r\n            item[filterItem.filter!.property] ?? ''\r\n          ).toLowerCase();\r\n          return itemValue.includes(filterValue);\r\n        });\r\n      }\r\n\r\n      if (\r\n        filterItem.arrange === 'filterByDate' &&\r\n        filterItem.dateFilter?.initial &&\r\n        filterItem.dateFilter?.final &&\r\n        sortBy.field\r\n      ) {\r\n        items = items.filter((item) => {\r\n          try {\r\n            const fieldValue = item[sortBy.field];\r\n\r\n            if (!fieldValue) {\r\n              return false;\r\n            }\r\n\r\n            let itemDate: Date;\r\n            const fieldValueObj = fieldValue as\r\n              | { toDate?: () => Date }\r\n              | Date\r\n              | string\r\n              | number\r\n              | null\r\n              | undefined;\r\n            if (\r\n              fieldValueObj &&\r\n              typeof fieldValueObj === 'object' &&\r\n              'toDate' in fieldValueObj &&\r\n              typeof fieldValueObj.toDate === 'function'\r\n            ) {\r\n              itemDate = fieldValueObj.toDate();\r\n            } else if (fieldValueObj instanceof Date) {\r\n              itemDate = fieldValueObj;\r\n            } else if (typeof fieldValueObj === 'string') {\r\n              itemDate = new Date(fieldValueObj);\r\n              if (isNaN(itemDate.getTime())) {\r\n                return false;\r\n              }\r\n            } else if (typeof fieldValueObj === 'number') {\r\n              itemDate = new Date(fieldValueObj);\r\n            } else {\r\n              return false;\r\n            }\r\n\r\n            if (!filterItem.dateFilter) {\r\n              return false;\r\n            }\r\n\r\n            return (\r\n              itemDate >= filterItem.dateFilter.initial &&\r\n              itemDate <= filterItem.dateFilter.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\r\n    // Aplicar filterFn se existir\r\n    if (params.filterFn) {\r\n      items = items.filter(params.filterFn);\r\n    }\r\n\r\n    if (sortBy && sortBy.field && sortBy.order) {\r\n      items.sort((a, b) => {\r\n        const valA = a[sortBy.field] as unknown;\r\n        const valB = b[sortBy.field] as unknown;\r\n\r\n        if (valA === null || valA === undefined) return 1;\r\n        if (valB === null || valB === undefined) return -1;\r\n\r\n        if (valA < valB) {\r\n          return sortBy.order === 'asc' ? -1 : 1;\r\n        }\r\n        if (valA > valB) {\r\n          return sortBy.order === 'asc' ? 1 : -1;\r\n        }\r\n        return 0;\r\n      });\r\n    }\r\n\r\n    // Implementação adequada da paginação\r\n    let currentClientPageIndex = 0;\r\n\r\n    // Determinar a página atual baseada na navegação\r\n    if (params.navigation === 'reload') {\r\n      currentClientPageIndex = 0;\r\n    } else {\r\n      currentClientPageIndex = params.clientPageIndex || 0;\r\n    }\r\n\r\n    const pageSize = params.batchSize;\r\n    const startIndex = currentClientPageIndex * pageSize;\r\n    const endIndex = startIndex + pageSize;\r\n    const paginatedItems = items.slice(startIndex, endIndex);\r\n\r\n    const totalPages = Math.ceil(items.length / pageSize);\r\n    const hasNextPage = currentClientPageIndex < totalPages - 1;\r\n    const hasPreviousPage = currentClientPageIndex > 0;\r\n\r\n    return {\r\n      items: paginatedItems,\r\n      filterLength: items.length,\r\n      lastDoc: null,\r\n      firstDoc: null,\r\n      hasNextPage: hasNextPage,\r\n      hasPreviousPage: hasPreviousPage,\r\n      currentClientPageIndex: currentClientPageIndex,\r\n      totalPages: totalPages,\r\n    };\r\n  }\r\n\r\n  async getItemsData(\r\n    collection: string,\r\n    arrange: Arrange,\r\n    conditions: Condition[] | undefined = undefined\r\n  ): Promise<Array<Record<string, unknown> & { id: string }>> {\r\n    try {\r\n      let query: firebase.firestore.Query<unknown> =\r\n        this.ngFire.collection(collection).ref;\r\n\r\n      query = this.applyFilters(query, arrange, conditions);\r\n      const snapshot = await query.get();\r\n      return await Promise.all(\r\n        snapshot.docs.map(\r\n          async (doc: firebase.firestore.QueryDocumentSnapshot<unknown>) => {\r\n            const data = doc.data() as Record<string, unknown> | undefined;\r\n            const id = doc.id;\r\n            return {\r\n              id,\r\n              ...(data ?? {}),\r\n            } as Record<string, unknown> & { id: string };\r\n          }\r\n        )\r\n      );\r\n    } catch (e) {\r\n      throw e;\r\n    }\r\n  }\r\n\r\n  public operators = {\r\n    '==': (a: unknown, b: unknown): boolean => a === b,\r\n    '!=': (a: unknown, b: unknown): boolean => a !== b,\r\n    '>': (a: unknown, b: unknown): boolean => (a as number) > (b as number),\r\n    '<': (a: unknown, b: unknown): boolean => (a as number) < (b as number),\r\n    '>=': (a: unknown, b: unknown): boolean => (a as number) >= (b as number),\r\n    '<=': (a: unknown, b: unknown): boolean => (a as number) <= (b as number),\r\n    in: (a: unknown, b: unknown): boolean => Array.isArray(b) && b.includes(a),\r\n    'not-in': (a: unknown, b: unknown): boolean =>\r\n      Array.isArray(b) && !b.includes(a),\r\n    'array-contains': (a: unknown, b: unknown): boolean =>\r\n      Array.isArray(a) && a.includes(b),\r\n    'array-contains-any': (a: unknown, b: unknown): boolean =>\r\n      Array.isArray(a) &&\r\n      Array.isArray(b) &&\r\n      b.some((item: unknown) => a.includes(item)),\r\n    includes: (a: unknown, b: unknown): boolean => {\r\n      if (typeof a === 'string' && typeof b === 'string') {\r\n        return a.includes(b);\r\n      }\r\n      if (Array.isArray(a)) {\r\n        return a.includes(b);\r\n      }\r\n      return false;\r\n    },\r\n  };\r\n\r\n  async deleteIndex(id: string, col: string): Promise<boolean> {\r\n    try {\r\n      const batch = this.ngFire.firestore.batch();\r\n\r\n      const docRef = this.ngFire.collection(col).doc(id);\r\n      const docSnapshot = (await firstValueFrom(\r\n        docRef.get()\r\n      )) as firebase.firestore.DocumentSnapshot<unknown>;\r\n      const doc = docSnapshot.data() as Record<string, unknown> | undefined;\r\n      batch.delete(docRef.ref);\r\n      if (doc && typeof doc['index'] === 'number') {\r\n        await this.reindex(doc['index'] as number, col, batch);\r\n      }\r\n      await batch.commit();\r\n\r\n      this.toastr.success('Item excluído com sucesso!');\r\n      return true;\r\n    } catch (e) {\r\n      const error = e as unknown;\r\n      console.error('Erro ao deletar item:', error);\r\n      this.toastr.error('Erro ao deletar item.');\r\n      return false;\r\n    }\r\n  }\r\n\r\n  async reindex(\r\n    index: number,\r\n    col: string,\r\n    batch: firebase.firestore.WriteBatch\r\n  ): Promise<void> {\r\n    try {\r\n      const snapshot = (await firstValueFrom(\r\n        this.ngFire.collection(col).get()\r\n      )) as firebase.firestore.QuerySnapshot<unknown>;\r\n      const docs = snapshot.docs;\r\n      for (let doc of docs) {\r\n        const data = doc.data() as Record<string, unknown> | undefined;\r\n        if (\r\n          data &&\r\n          typeof data['index'] === 'number' &&\r\n          (data['index'] as number) > index\r\n        ) {\r\n          data['index'] = ((data['index'] as number) - 1) as unknown;\r\n          const docRef = this.ngFire.collection(col).doc(doc.id).ref;\r\n          batch.update(docRef, data);\r\n        }\r\n      }\r\n    } catch (error) {\r\n      console.error('Erro ao reindexar:', error);\r\n    }\r\n    return;\r\n  }\r\n\r\n  dateFormatValidator(): ValidatorFn {\r\n    return (control: AbstractControl): { [key: string]: unknown } | null => {\r\n      if (!control.value) {\r\n        return null;\r\n      }\r\n\r\n      const dateStr = control.value.trim();\r\n      const datePattern = /^(\\d{1,2})\\/(\\d{1,2})\\/(\\d{4})$/;\r\n\r\n      if (!datePattern.test(dateStr)) {\r\n        return { invalidFormat: true };\r\n      }\r\n\r\n      const parts = dateStr.split('/');\r\n      const day = parts[0].padStart(2, '0');\r\n      const month = parts[1].padStart(2, '0');\r\n      const year = parts[2];\r\n      const normalizedDate = `${day}/${month}/${year}`;\r\n\r\n      const date = moment(normalizedDate, 'DD/MM/YYYY', true);\r\n\r\n      if (!date.isValid()) {\r\n        return { invalidDate: true };\r\n      }\r\n\r\n      return null;\r\n    };\r\n  }\r\n\r\n  async updateIndex(index: number, id: string, col: string): Promise<void> {\r\n    await this.ngFire.collection(col).doc(id).update({ index });\r\n  }\r\n\r\n  /**\r\n   * Extrai o link de criação de índice da mensagem de erro do Firestore\r\n   */\r\n  private extractIndexLink(error: unknown): string | null {\r\n    if (\r\n      !error ||\r\n      typeof error !== 'object' ||\r\n      !('message' in error) ||\r\n      typeof (error as ErrorWithMessage).message !== 'string'\r\n    )\r\n      return null;\r\n\r\n    const linkMatch = (error as ErrorWithMessage).message.match(\r\n      /(https:\\/\\/console\\.firebase\\.google\\.com\\/[^\\s]+)/\r\n    );\r\n    return linkMatch ? linkMatch[1] : null;\r\n  }\r\n\r\n  /**\r\n   * Rastreia índices ausentes ao usar fallback preventivo\r\n   */\r\n  private async trackMissingIndexPreventive(\r\n    collection: string,\r\n    arrange: Arrange,\r\n    conditions: Condition[] | undefined = undefined\r\n  ): Promise<void> {\r\n    try {\r\n      const querySignature = this.generateQuerySignature(\r\n        collection,\r\n        arrange,\r\n        conditions\r\n      );\r\n      const docId = `${collection}_${querySignature}`;\r\n\r\n      const indexLink = this.generateIndexLink(collection, arrange, conditions);\r\n\r\n      const indexInstructions = this.generateIndexInstructions(\r\n        collection,\r\n        arrange,\r\n        conditions\r\n      );\r\n\r\n      const trackingData: Record<string, unknown> = {\r\n        collection,\r\n        indexLink,\r\n        indexInstructions,\r\n        arrange: {\r\n          sortBy: arrange.sortBy,\r\n          filters:\r\n            arrange.filters?.map((f) => ({\r\n              arrange: f.arrange,\r\n              property: f.filter?.property || null,\r\n              dateField:\r\n                f.arrange === 'filterByDate' ? arrange.sortBy?.field : null,\r\n            })) || [],\r\n        },\r\n        conditions:\r\n          conditions?.map((c: Condition) => ({\r\n            property: c.firestoreProperty,\r\n            operator: c.operator,\r\n          })) || [],\r\n        errorMessage: `Fallback preventivo usado para a collection ${collection}. A query exigiria índice composto.`,\r\n        updatedAt: firebase.firestore.FieldValue.serverTimestamp(),\r\n      };\r\n\r\n      const docRef = this.ngFire.collection('missingIndexes').doc(docId);\r\n      const doc = await docRef.get().toPromise();\r\n\r\n      if (doc && doc.exists) {\r\n        await docRef.update({\r\n          count: firebase.firestore.FieldValue.increment(1),\r\n          updatedAt: firebase.firestore.FieldValue.serverTimestamp(),\r\n          lastError: trackingData['errorMessage'],\r\n        });\r\n      } else {\r\n        await docRef.set({\r\n          ...trackingData,\r\n          count: 1,\r\n          createdAt: firebase.firestore.FieldValue.serverTimestamp(),\r\n        });\r\n      }\r\n    } catch (trackingError) {\r\n      console.warn('Falha ao rastrear fallback preventivo:', trackingError);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Gera uma assinatura única para uma query\r\n   */\r\n  private generateQuerySignature(\r\n    collection: string,\r\n    arrange: Arrange,\r\n    conditions: Condition[] | undefined = undefined\r\n  ): string {\r\n    const signature = {\r\n      collection,\r\n      sortBy: arrange.sortBy,\r\n      filters:\r\n        arrange.filters?.map((f: Arrange['filters'][0]) => ({\r\n          arrange: f.arrange,\r\n          property: f.filter?.property || null,\r\n        })) || [],\r\n      conditions:\r\n        conditions?.map((c: Condition) => ({\r\n          property: c.firestoreProperty,\r\n          operator: c.operator,\r\n        })) || [],\r\n    };\r\n\r\n    return btoa(JSON.stringify(signature))\r\n      .replace(/[^a-zA-Z0-9]/g, '')\r\n      .substring(0, 20);\r\n  }\r\n\r\n  /**\r\n   * Gera instruções claras para criar o índice manualmente\r\n   */\r\n  private generateIndexInstructions(\r\n    collection: string,\r\n    arrange: Arrange,\r\n    conditions: Condition[] | undefined = undefined\r\n  ): {\r\n    summary: string;\r\n    collection: string;\r\n    fields: Array<{\r\n      field: string;\r\n      order: string;\r\n      type: string;\r\n      operator: string;\r\n      description: string;\r\n    }>;\r\n    queryExample: string;\r\n    stepByStep: string[];\r\n    notes: string[];\r\n  } {\r\n    const instructions: {\r\n      summary: string;\r\n      collection: string;\r\n      fields: Array<{\r\n        field: string;\r\n        order: string;\r\n        type: string;\r\n        operator: string;\r\n        description: string;\r\n      }>;\r\n      queryExample: string;\r\n      stepByStep: string[];\r\n      notes: string[];\r\n    } = {\r\n      summary: '',\r\n      collection: collection,\r\n      fields: [],\r\n      queryExample: '',\r\n      stepByStep: [],\r\n      notes: [],\r\n    };\r\n\r\n    const fields: Array<{\r\n      field: string;\r\n      order: string;\r\n      type: string;\r\n      operator: string;\r\n      description: string;\r\n    }> = [];\r\n\r\n    if (conditions && conditions.length > 0) {\r\n      conditions.forEach((condition: Condition) => {\r\n        if (condition.firestoreProperty) {\r\n          fields.push({\r\n            field: condition.firestoreProperty,\r\n            order: 'Ascending',\r\n            type: 'WHERE clause',\r\n            operator: condition.operator,\r\n            description: `Filtrar por ${condition.firestoreProperty} usando operador ${condition.operator}`,\r\n          });\r\n        }\r\n      });\r\n    }\r\n\r\n    if (arrange.filters && arrange.filters.length > 0) {\r\n      arrange.filters.forEach((filter) => {\r\n        if (filter.filter?.property) {\r\n          fields.push({\r\n            field: filter.filter.property,\r\n            order: 'Ascending',\r\n            type: 'WHERE clause (filter)',\r\n            operator: filter.arrange === 'filter' ? 'CONTAINS' : 'RANGE',\r\n            description: `Filtrar por ${filter.filter.property} usando filtro ${filter.arrange}`,\r\n          });\r\n        }\r\n      });\r\n    }\r\n\r\n    if (arrange.sortBy?.field) {\r\n      fields.push({\r\n        field: arrange.sortBy.field,\r\n        order: arrange.sortBy.order === 'desc' ? 'Descending' : 'Ascending',\r\n        type: 'ORDER BY clause',\r\n        operator: 'N/A',\r\n        description: `Ordenar resultados por ${arrange.sortBy.field} em ordem ${arrange.sortBy.order}`,\r\n      });\r\n    }\r\n\r\n    instructions.fields = fields;\r\n\r\n    const fieldNames = fields.map((f) => f.field).join(' + ');\r\n    instructions.summary = `Criar índice composto para ${collection}: ${fieldNames}`;\r\n\r\n    let queryExample = `db.collection('${collection}')`;\r\n\r\n    fields.forEach((field, index: number) => {\r\n      if (field.type.includes('WHERE')) {\r\n        if (field.operator === '==') {\r\n          queryExample += `\\n  .where('${field.field}', '==', 'value')`;\r\n        } else if (field.operator === 'CONTAINS') {\r\n          queryExample += `\\n  .where('${field.field}', '>=', 'searchText')`;\r\n        } else {\r\n          queryExample += `\\n  .where('${field.field}', '${field.operator}', 'value')`;\r\n        }\r\n      }\r\n    });\r\n\r\n    const orderByField = fields.find((f) => f.type.includes('ORDER BY'));\r\n    if (orderByField) {\r\n      queryExample += `\\n  .orderBy('${\r\n        orderByField.field\r\n      }', '${orderByField.order.toLowerCase()}')`;\r\n    }\r\n\r\n    instructions.queryExample = queryExample;\r\n\r\n    instructions.stepByStep = [\r\n      '1. Ir para Firebase Console → Firestore → Indexes',\r\n      '2. Clicar em \"Create Index\"',\r\n      `3. Definir Collection ID: ${collection}`,\r\n      '4. Configurar campos nesta ORDEM EXATA:',\r\n      ...fields.map(\r\n        (field, index: number) =>\r\n          `   ${index + 1}. Campo: ${field.field}, Order: ${\r\n            field.order\r\n          }, Array: No`\r\n      ),\r\n      '5. Definir Query scopes: Collection',\r\n      '6. Clicar em \"Create\" e aguardar conclusão',\r\n    ];\r\n\r\n    instructions.notes = [\r\n      '⚠️ A ordem dos campos é CRÍTICA - deve corresponder exatamente à ordem da query',\r\n      '⚠️ As cláusulas WHERE devem vir ANTES do campo ORDER BY',\r\n      '⚠️ Este índice só funcionará para queries com esta combinação EXATA de campos',\r\n      '⚠️ A criação do índice pode levar vários minutos',\r\n    ];\r\n\r\n    return instructions;\r\n  }\r\n\r\n  /**\r\n   * Gera um link de índice baseado na estrutura da query\r\n   */\r\n  private generateIndexLink(\r\n    collection: string,\r\n    arrange: Arrange,\r\n    conditions: Condition[] | undefined = undefined\r\n  ): string | null {\r\n    try {\r\n      const indexFields: string[] = [];\r\n\r\n      if (conditions && conditions.length > 0) {\r\n        conditions.forEach((condition: Condition) => {\r\n          if (condition.firestoreProperty) {\r\n            indexFields.push(condition.firestoreProperty);\r\n          }\r\n        });\r\n      }\r\n\r\n      if (arrange.filters && arrange.filters.length > 0) {\r\n        arrange.filters.forEach((filter: Arrange['filters'][0]) => {\r\n          if (filter.filter?.property) {\r\n            indexFields.push(filter.filter.property);\r\n          }\r\n        });\r\n      }\r\n\r\n      if (arrange.sortBy?.field) {\r\n        indexFields.push(arrange.sortBy.field);\r\n      }\r\n\r\n      if (indexFields.length > 1) {\r\n        // Obter o projectId dinamicamente do Firebase\r\n        const options = this.ngFire?.firestore?.app?.options as\r\n          | Record<string, unknown>\r\n          | undefined;\r\n        const projectId = options?.['projectId'] as string | undefined;\r\n\r\n        if (!projectId) {\r\n          console.warn('Não foi possível obter o projectId do Firebase');\r\n          return null;\r\n        }\r\n\r\n        const baseUrl = `https://console.firebase.google.com/project/${projectId}/firestore/indexes`;\r\n        const queryParams = new URLSearchParams({\r\n          create_composite: `collection=${collection}&fields=${indexFields.join(\r\n            ','\r\n          )}`,\r\n        });\r\n\r\n        const finalLink = `${baseUrl}?${queryParams.toString()}`;\r\n        return finalLink;\r\n      }\r\n\r\n      return null;\r\n    } catch (error) {\r\n      console.warn('Falha ao gerar link de índice:', error);\r\n      return null;\r\n    }\r\n  }\r\n\r\n  private async trackMissingIndex(\r\n    error: unknown,\r\n    collection: string,\r\n    arrange: Arrange,\r\n    conditions: Condition[] | undefined = undefined\r\n  ): Promise<void> {\r\n    try {\r\n      const indexLink = this.extractIndexLink(error);\r\n      if (!indexLink) return;\r\n\r\n      const linkHash = btoa(indexLink)\r\n        .replace(/[^a-zA-Z0-9]/g, '')\r\n        .substring(0, 20);\r\n      const docId = `${collection}_${linkHash}`;\r\n\r\n      const indexInstructions = this.generateIndexInstructions(\r\n        collection,\r\n        arrange,\r\n        conditions\r\n      );\r\n\r\n      const trackingData: Record<string, unknown> = {\r\n        collection,\r\n        indexLink,\r\n        indexInstructions,\r\n        arrange: {\r\n          sortBy: arrange.sortBy,\r\n          filters:\r\n            arrange.filters?.map((f) => ({\r\n              arrange: f.arrange,\r\n              property: f.filter?.property || null,\r\n              dateField:\r\n                f.arrange === 'filterByDate' ? arrange.sortBy?.field : null,\r\n            })) || [],\r\n        },\r\n        conditions:\r\n          conditions?.map((c: Condition) => ({\r\n            property: c.firestoreProperty,\r\n            operator: c.operator,\r\n          })) || [],\r\n        errorMessage:\r\n          error &&\r\n          typeof error === 'object' &&\r\n          'message' in error &&\r\n          typeof (error as ErrorWithMessage).message === 'string'\r\n            ? (error as ErrorWithMessage).message\r\n            : String(error),\r\n        updatedAt: firebase.firestore.FieldValue.serverTimestamp(),\r\n      };\r\n\r\n      const docRef = this.ngFire.collection('missingIndexes').doc(docId);\r\n      const doc = await docRef.get().toPromise();\r\n\r\n      const errorMessage =\r\n        error &&\r\n        typeof error === 'object' &&\r\n        'message' in error &&\r\n        typeof (error as ErrorWithMessage).message === 'string'\r\n          ? (error as ErrorWithMessage).message\r\n          : String(error);\r\n\r\n      if (doc && doc.exists) {\r\n        await docRef.update({\r\n          count: firebase.firestore.FieldValue.increment(1),\r\n          updatedAt: firebase.firestore.FieldValue.serverTimestamp(),\r\n          lastError: errorMessage,\r\n        });\r\n      } else {\r\n        await docRef.set({\r\n          ...trackingData,\r\n          count: 1,\r\n          createdAt: firebase.firestore.FieldValue.serverTimestamp(),\r\n        });\r\n      }\r\n    } catch (trackingError) {\r\n      console.warn('Falha ao rastrear índice ausente:', trackingError);\r\n    }\r\n  }\r\n}\r\n"]}