valtech-components 2.0.407 → 2.0.410

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 (110) hide show
  1. package/esm2022/lib/components/atoms/horizontal-scroll/horizontal-scroll.component.mjs +82 -0
  2. package/esm2022/lib/components/atoms/horizontal-scroll/types.mjs +2 -0
  3. package/esm2022/lib/components/atoms/rights-footer/rights-footer.component.mjs +82 -0
  4. package/esm2022/lib/components/atoms/rights-footer/types.mjs +2 -0
  5. package/esm2022/lib/components/molecules/check-input/check-input.component.mjs +55 -11
  6. package/esm2022/lib/components/molecules/email-input/email-input.component.mjs +13 -4
  7. package/esm2022/lib/components/molecules/expandable-text/expandable-text.component.mjs +27 -23
  8. package/esm2022/lib/components/molecules/footer-links/footer-links.component.mjs +277 -0
  9. package/esm2022/lib/components/molecules/footer-links/types.mjs +2 -0
  10. package/esm2022/lib/components/molecules/links-accordion/links-accordion.component.mjs +157 -0
  11. package/esm2022/lib/components/molecules/links-accordion/types.mjs +2 -0
  12. package/esm2022/lib/components/molecules/password-input/password-input.component.mjs +12 -2
  13. package/esm2022/lib/components/molecules/prompter/prompter.component.mjs +21 -9
  14. package/esm2022/lib/components/molecules/prompter/types.mjs +1 -1
  15. package/esm2022/lib/components/molecules/radio-input/radio-input.component.mjs +13 -4
  16. package/esm2022/lib/components/molecules/recap-card/recap-card.component.mjs +78 -0
  17. package/esm2022/lib/components/molecules/recap-card/types.mjs +2 -0
  18. package/esm2022/lib/components/molecules/select-input/select-input.component.mjs +31 -14
  19. package/esm2022/lib/components/molecules/swipe-carousel/swipe-carousel.component.mjs +206 -0
  20. package/esm2022/lib/components/molecules/swipe-carousel/types.mjs +2 -0
  21. package/esm2022/lib/components/molecules/testimonial-card/testimonial-card.component.mjs +138 -0
  22. package/esm2022/lib/components/molecules/testimonial-card/types.mjs +2 -0
  23. package/esm2022/lib/components/molecules/text-input/text-input.component.mjs +14 -4
  24. package/esm2022/lib/components/organisms/cards-carousel/cards-carousel.component.mjs +61 -0
  25. package/esm2022/lib/components/organisms/cards-carousel/types.mjs +2 -0
  26. package/esm2022/lib/components/organisms/company-footer/company-footer.component.mjs +72 -0
  27. package/esm2022/lib/components/organisms/company-footer/types.mjs +2 -0
  28. package/esm2022/lib/components/organisms/data-table/data-table.component.mjs +175 -3
  29. package/esm2022/lib/components/organisms/data-table/types.mjs +1 -1
  30. package/esm2022/lib/components/organisms/form/form.component.mjs +2 -2
  31. package/esm2022/lib/components/organisms/fun-header/fun-header.component.mjs +225 -0
  32. package/esm2022/lib/components/organisms/fun-header/types.mjs +2 -0
  33. package/esm2022/lib/components/organisms/menu/menu.component.mjs +197 -0
  34. package/esm2022/lib/components/organisms/menu/types.mjs +2 -0
  35. package/esm2022/lib/components/organisms/testimonial-carousel/testimonial-carousel.component.mjs +72 -0
  36. package/esm2022/lib/components/organisms/testimonial-carousel/types.mjs +2 -0
  37. package/esm2022/lib/components/templates/page-content/page-content.component.mjs +156 -0
  38. package/esm2022/lib/components/templates/page-content/types.mjs +2 -0
  39. package/esm2022/lib/components/templates/page-template/page-template.component.mjs +181 -0
  40. package/esm2022/lib/components/templates/page-template/types.mjs +2 -0
  41. package/esm2022/lib/components/templates/page-wrapper/page-wrapper.component.mjs +195 -0
  42. package/esm2022/lib/components/templates/page-wrapper/types.mjs +2 -0
  43. package/esm2022/lib/components/types.mjs +1 -1
  44. package/esm2022/lib/services/firebase/config.mjs +103 -0
  45. package/esm2022/lib/services/firebase/firebase.service.mjs +285 -0
  46. package/esm2022/lib/services/firebase/firestore-collection.mjs +266 -0
  47. package/esm2022/lib/services/firebase/firestore.service.mjs +508 -0
  48. package/esm2022/lib/services/firebase/index.mjs +46 -0
  49. package/esm2022/lib/services/firebase/messaging.service.mjs +503 -0
  50. package/esm2022/lib/services/firebase/storage.service.mjs +421 -0
  51. package/esm2022/lib/services/firebase/types.mjs +8 -0
  52. package/esm2022/lib/services/firebase/utils/path-builder.mjs +195 -0
  53. package/esm2022/lib/services/firebase/utils/query-builder.mjs +302 -0
  54. package/esm2022/lib/services/icons.service.mjs +3 -2
  55. package/esm2022/public-api.mjs +33 -1
  56. package/fesm2022/valtech-components.mjs +5823 -869
  57. package/fesm2022/valtech-components.mjs.map +1 -1
  58. package/lib/components/atoms/horizontal-scroll/horizontal-scroll.component.d.ts +41 -0
  59. package/lib/components/atoms/horizontal-scroll/types.d.ts +13 -0
  60. package/lib/components/atoms/rights-footer/rights-footer.component.d.ts +39 -0
  61. package/lib/components/atoms/rights-footer/types.d.ts +13 -0
  62. package/lib/components/molecules/check-input/check-input.component.d.ts +17 -2
  63. package/lib/components/molecules/email-input/email-input.component.d.ts +1 -2
  64. package/lib/components/molecules/footer-links/footer-links.component.d.ts +47 -0
  65. package/lib/components/molecules/footer-links/types.d.ts +37 -0
  66. package/lib/components/molecules/links-accordion/links-accordion.component.d.ts +48 -0
  67. package/lib/components/molecules/links-accordion/types.d.ts +33 -0
  68. package/lib/components/molecules/password-input/password-input.component.d.ts +1 -1
  69. package/lib/components/molecules/prompter/prompter.component.d.ts +8 -1
  70. package/lib/components/molecules/prompter/types.d.ts +7 -1
  71. package/lib/components/molecules/radio-input/radio-input.component.d.ts +1 -2
  72. package/lib/components/molecules/recap-card/recap-card.component.d.ts +36 -0
  73. package/lib/components/molecules/recap-card/types.d.ts +30 -0
  74. package/lib/components/molecules/select-input/select-input.component.d.ts +6 -1
  75. package/lib/components/molecules/swipe-carousel/swipe-carousel.component.d.ts +66 -0
  76. package/lib/components/molecules/swipe-carousel/types.d.ts +35 -0
  77. package/lib/components/molecules/testimonial-card/testimonial-card.component.d.ts +41 -0
  78. package/lib/components/molecules/testimonial-card/types.d.ts +25 -0
  79. package/lib/components/molecules/text-input/text-input.component.d.ts +13 -4
  80. package/lib/components/organisms/cards-carousel/cards-carousel.component.d.ts +30 -0
  81. package/lib/components/organisms/cards-carousel/types.d.ts +11 -0
  82. package/lib/components/organisms/company-footer/company-footer.component.d.ts +32 -0
  83. package/lib/components/organisms/company-footer/types.d.ts +15 -0
  84. package/lib/components/organisms/data-table/data-table.component.d.ts +1 -1
  85. package/lib/components/organisms/data-table/types.d.ts +6 -0
  86. package/lib/components/organisms/fun-header/fun-header.component.d.ts +72 -0
  87. package/lib/components/organisms/fun-header/types.d.ts +28 -0
  88. package/lib/components/organisms/menu/menu.component.d.ts +39 -0
  89. package/lib/components/organisms/menu/types.d.ts +23 -0
  90. package/lib/components/organisms/testimonial-carousel/testimonial-carousel.component.d.ts +33 -0
  91. package/lib/components/organisms/testimonial-carousel/types.d.ts +8 -0
  92. package/lib/components/templates/page-content/page-content.component.d.ts +55 -0
  93. package/lib/components/templates/page-content/types.d.ts +14 -0
  94. package/lib/components/templates/page-template/page-template.component.d.ts +49 -0
  95. package/lib/components/templates/page-template/types.d.ts +17 -0
  96. package/lib/components/templates/page-wrapper/page-wrapper.component.d.ts +61 -0
  97. package/lib/components/templates/page-wrapper/types.d.ts +19 -0
  98. package/lib/components/types.d.ts +14 -0
  99. package/lib/services/firebase/config.d.ts +49 -0
  100. package/lib/services/firebase/firebase.service.d.ts +140 -0
  101. package/lib/services/firebase/firestore-collection.d.ts +195 -0
  102. package/lib/services/firebase/firestore.service.d.ts +303 -0
  103. package/lib/services/firebase/index.d.ts +38 -0
  104. package/lib/services/firebase/messaging.service.d.ts +254 -0
  105. package/lib/services/firebase/storage.service.d.ts +204 -0
  106. package/lib/services/firebase/types.d.ts +279 -0
  107. package/lib/services/firebase/utils/path-builder.d.ts +132 -0
  108. package/lib/services/firebase/utils/query-builder.d.ts +210 -0
  109. package/package.json +3 -1
  110. package/public-api.d.ts +31 -0
@@ -0,0 +1,195 @@
1
+ /**
2
+ * Path Builder
3
+ *
4
+ * Utilidades para construir rutas de Firestore con templates.
5
+ * Soporta rutas multi-nivel y anidadas.
6
+ */
7
+ /**
8
+ * Construye una ruta de Firestore reemplazando placeholders.
9
+ *
10
+ * @param template - Template con placeholders en formato {param}
11
+ * @param params - Objeto con los valores a reemplazar
12
+ * @returns Ruta construida
13
+ * @throws Error si faltan parámetros requeridos
14
+ *
15
+ * @example
16
+ * ```typescript
17
+ * // Ruta simple
18
+ * buildPath('users/{userId}', { userId: 'abc123' });
19
+ * // => 'users/abc123'
20
+ *
21
+ * // Ruta anidada
22
+ * buildPath('users/{userId}/documents/{docId}', {
23
+ * userId: 'abc123',
24
+ * docId: 'doc456'
25
+ * });
26
+ * // => 'users/abc123/documents/doc456'
27
+ *
28
+ * // Múltiples niveles
29
+ * buildPath('orgs/{orgId}/teams/{teamId}/members/{memberId}', {
30
+ * orgId: 'org1',
31
+ * teamId: 'team2',
32
+ * memberId: 'member3'
33
+ * });
34
+ * // => 'orgs/org1/teams/team2/members/member3'
35
+ * ```
36
+ */
37
+ export function buildPath(template, params) {
38
+ let result = template;
39
+ // Encontrar todos los placeholders
40
+ const placeholders = template.match(/\{([^}]+)\}/g);
41
+ if (!placeholders) {
42
+ return template;
43
+ }
44
+ for (const placeholder of placeholders) {
45
+ const key = placeholder.slice(1, -1); // Remover { y }
46
+ const value = params[key];
47
+ if (value === undefined || value === null) {
48
+ throw new Error(`Parámetro requerido '${key}' no proporcionado para la ruta: ${template}`);
49
+ }
50
+ if (typeof value !== 'string' || value.trim() === '') {
51
+ throw new Error(`El parámetro '${key}' debe ser un string no vacío`);
52
+ }
53
+ // Validar que no contenga caracteres inválidos para Firestore
54
+ if (value.includes('/')) {
55
+ throw new Error(`El parámetro '${key}' no puede contener '/'`);
56
+ }
57
+ result = result.replace(placeholder, value);
58
+ }
59
+ return result;
60
+ }
61
+ /**
62
+ * Extrae los nombres de los parámetros de un template de ruta.
63
+ *
64
+ * @param template - Template de ruta
65
+ * @returns Array con los nombres de los parámetros
66
+ *
67
+ * @example
68
+ * ```typescript
69
+ * extractParams('users/{userId}/documents/{docId}');
70
+ * // => ['userId', 'docId']
71
+ * ```
72
+ */
73
+ export function extractPathParams(template) {
74
+ const matches = template.match(/\{([^}]+)\}/g);
75
+ if (!matches)
76
+ return [];
77
+ return matches.map((m) => m.slice(1, -1));
78
+ }
79
+ /**
80
+ * Valida que una ruta de Firestore sea válida.
81
+ *
82
+ * @param path - Ruta a validar
83
+ * @returns true si la ruta es válida
84
+ *
85
+ * @example
86
+ * ```typescript
87
+ * isValidPath('users/abc123'); // true
88
+ * isValidPath('users/abc123/documents'); // true
89
+ * isValidPath('users//documents'); // false (segmento vacío)
90
+ * isValidPath(''); // false (vacío)
91
+ * ```
92
+ */
93
+ export function isValidPath(path) {
94
+ if (!path || path.trim() === '')
95
+ return false;
96
+ const segments = path.split('/');
97
+ // No puede tener segmentos vacíos
98
+ if (segments.some((s) => s.trim() === ''))
99
+ return false;
100
+ // No puede empezar o terminar con /
101
+ if (path.startsWith('/') || path.endsWith('/'))
102
+ return false;
103
+ return true;
104
+ }
105
+ /**
106
+ * Obtiene la ruta de la colección padre de un documento.
107
+ *
108
+ * @param documentPath - Ruta completa del documento
109
+ * @returns Ruta de la colección padre
110
+ *
111
+ * @example
112
+ * ```typescript
113
+ * getCollectionPath('users/abc123');
114
+ * // => 'users'
115
+ *
116
+ * getCollectionPath('users/abc123/documents/doc456');
117
+ * // => 'users/abc123/documents'
118
+ * ```
119
+ */
120
+ export function getCollectionPath(documentPath) {
121
+ const segments = documentPath.split('/');
122
+ if (segments.length < 2) {
123
+ throw new Error(`Ruta de documento inválida: ${documentPath}`);
124
+ }
125
+ return segments.slice(0, -1).join('/');
126
+ }
127
+ /**
128
+ * Obtiene el ID del documento de una ruta.
129
+ *
130
+ * @param documentPath - Ruta completa del documento
131
+ * @returns ID del documento
132
+ *
133
+ * @example
134
+ * ```typescript
135
+ * getDocumentId('users/abc123');
136
+ * // => 'abc123'
137
+ *
138
+ * getDocumentId('users/abc123/documents/doc456');
139
+ * // => 'doc456'
140
+ * ```
141
+ */
142
+ export function getDocumentId(documentPath) {
143
+ const segments = documentPath.split('/');
144
+ if (segments.length < 2 || segments.length % 2 !== 0) {
145
+ throw new Error(`Ruta de documento inválida: ${documentPath}`);
146
+ }
147
+ return segments[segments.length - 1];
148
+ }
149
+ /**
150
+ * Verifica si una ruta apunta a un documento (número par de segmentos).
151
+ *
152
+ * @param path - Ruta a verificar
153
+ * @returns true si es una ruta de documento
154
+ *
155
+ * @example
156
+ * ```typescript
157
+ * isDocumentPath('users/abc123'); // true
158
+ * isDocumentPath('users'); // false (colección)
159
+ * isDocumentPath('users/abc123/documents'); // false (colección)
160
+ * ```
161
+ */
162
+ export function isDocumentPath(path) {
163
+ const segments = path.split('/').filter((s) => s.trim() !== '');
164
+ return segments.length > 0 && segments.length % 2 === 0;
165
+ }
166
+ /**
167
+ * Verifica si una ruta apunta a una colección (número impar de segmentos).
168
+ *
169
+ * @param path - Ruta a verificar
170
+ * @returns true si es una ruta de colección
171
+ */
172
+ export function isCollectionPath(path) {
173
+ const segments = path.split('/').filter((s) => s.trim() !== '');
174
+ return segments.length > 0 && segments.length % 2 !== 0;
175
+ }
176
+ /**
177
+ * Combina una ruta base con segmentos adicionales.
178
+ *
179
+ * @param basePath - Ruta base
180
+ * @param segments - Segmentos adicionales
181
+ * @returns Ruta combinada
182
+ *
183
+ * @example
184
+ * ```typescript
185
+ * joinPath('users', 'abc123', 'documents');
186
+ * // => 'users/abc123/documents'
187
+ * ```
188
+ */
189
+ export function joinPath(...segments) {
190
+ return segments
191
+ .filter((s) => s && s.trim() !== '')
192
+ .map((s) => s.replace(/^\/+|\/+$/g, '')) // Remover / al inicio y final
193
+ .join('/');
194
+ }
195
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"path-builder.js","sourceRoot":"","sources":["../../../../../../../src/lib/services/firebase/utils/path-builder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Path Builder\n *\n * Utilidades para construir rutas de Firestore con templates.\n * Soporta rutas multi-nivel y anidadas.\n */\n\n/**\n * Construye una ruta de Firestore reemplazando placeholders.\n *\n * @param template - Template con placeholders en formato {param}\n * @param params - Objeto con los valores a reemplazar\n * @returns Ruta construida\n * @throws Error si faltan parámetros requeridos\n *\n * @example\n * ```typescript\n * // Ruta simple\n * buildPath('users/{userId}', { userId: 'abc123' });\n * // => 'users/abc123'\n *\n * // Ruta anidada\n * buildPath('users/{userId}/documents/{docId}', {\n *   userId: 'abc123',\n *   docId: 'doc456'\n * });\n * // => 'users/abc123/documents/doc456'\n *\n * // Múltiples niveles\n * buildPath('orgs/{orgId}/teams/{teamId}/members/{memberId}', {\n *   orgId: 'org1',\n *   teamId: 'team2',\n *   memberId: 'member3'\n * });\n * // => 'orgs/org1/teams/team2/members/member3'\n * ```\n */\nexport function buildPath(template: string, params: Record<string, string>): string {\n  let result = template;\n\n  // Encontrar todos los placeholders\n  const placeholders = template.match(/\\{([^}]+)\\}/g);\n\n  if (!placeholders) {\n    return template;\n  }\n\n  for (const placeholder of placeholders) {\n    const key = placeholder.slice(1, -1); // Remover { y }\n    const value = params[key];\n\n    if (value === undefined || value === null) {\n      throw new Error(`Parámetro requerido '${key}' no proporcionado para la ruta: ${template}`);\n    }\n\n    if (typeof value !== 'string' || value.trim() === '') {\n      throw new Error(`El parámetro '${key}' debe ser un string no vacío`);\n    }\n\n    // Validar que no contenga caracteres inválidos para Firestore\n    if (value.includes('/')) {\n      throw new Error(`El parámetro '${key}' no puede contener '/'`);\n    }\n\n    result = result.replace(placeholder, value);\n  }\n\n  return result;\n}\n\n/**\n * Extrae los nombres de los parámetros de un template de ruta.\n *\n * @param template - Template de ruta\n * @returns Array con los nombres de los parámetros\n *\n * @example\n * ```typescript\n * extractParams('users/{userId}/documents/{docId}');\n * // => ['userId', 'docId']\n * ```\n */\nexport function extractPathParams(template: string): string[] {\n  const matches = template.match(/\\{([^}]+)\\}/g);\n  if (!matches) return [];\n  return matches.map((m) => m.slice(1, -1));\n}\n\n/**\n * Valida que una ruta de Firestore sea válida.\n *\n * @param path - Ruta a validar\n * @returns true si la ruta es válida\n *\n * @example\n * ```typescript\n * isValidPath('users/abc123');           // true\n * isValidPath('users/abc123/documents'); // true\n * isValidPath('users//documents');       // false (segmento vacío)\n * isValidPath('');                       // false (vacío)\n * ```\n */\nexport function isValidPath(path: string): boolean {\n  if (!path || path.trim() === '') return false;\n\n  const segments = path.split('/');\n\n  // No puede tener segmentos vacíos\n  if (segments.some((s) => s.trim() === '')) return false;\n\n  // No puede empezar o terminar con /\n  if (path.startsWith('/') || path.endsWith('/')) return false;\n\n  return true;\n}\n\n/**\n * Obtiene la ruta de la colección padre de un documento.\n *\n * @param documentPath - Ruta completa del documento\n * @returns Ruta de la colección padre\n *\n * @example\n * ```typescript\n * getCollectionPath('users/abc123');\n * // => 'users'\n *\n * getCollectionPath('users/abc123/documents/doc456');\n * // => 'users/abc123/documents'\n * ```\n */\nexport function getCollectionPath(documentPath: string): string {\n  const segments = documentPath.split('/');\n  if (segments.length < 2) {\n    throw new Error(`Ruta de documento inválida: ${documentPath}`);\n  }\n  return segments.slice(0, -1).join('/');\n}\n\n/**\n * Obtiene el ID del documento de una ruta.\n *\n * @param documentPath - Ruta completa del documento\n * @returns ID del documento\n *\n * @example\n * ```typescript\n * getDocumentId('users/abc123');\n * // => 'abc123'\n *\n * getDocumentId('users/abc123/documents/doc456');\n * // => 'doc456'\n * ```\n */\nexport function getDocumentId(documentPath: string): string {\n  const segments = documentPath.split('/');\n  if (segments.length < 2 || segments.length % 2 !== 0) {\n    throw new Error(`Ruta de documento inválida: ${documentPath}`);\n  }\n  return segments[segments.length - 1];\n}\n\n/**\n * Verifica si una ruta apunta a un documento (número par de segmentos).\n *\n * @param path - Ruta a verificar\n * @returns true si es una ruta de documento\n *\n * @example\n * ```typescript\n * isDocumentPath('users/abc123');           // true\n * isDocumentPath('users');                  // false (colección)\n * isDocumentPath('users/abc123/documents'); // false (colección)\n * ```\n */\nexport function isDocumentPath(path: string): boolean {\n  const segments = path.split('/').filter((s) => s.trim() !== '');\n  return segments.length > 0 && segments.length % 2 === 0;\n}\n\n/**\n * Verifica si una ruta apunta a una colección (número impar de segmentos).\n *\n * @param path - Ruta a verificar\n * @returns true si es una ruta de colección\n */\nexport function isCollectionPath(path: string): boolean {\n  const segments = path.split('/').filter((s) => s.trim() !== '');\n  return segments.length > 0 && segments.length % 2 !== 0;\n}\n\n/**\n * Combina una ruta base con segmentos adicionales.\n *\n * @param basePath - Ruta base\n * @param segments - Segmentos adicionales\n * @returns Ruta combinada\n *\n * @example\n * ```typescript\n * joinPath('users', 'abc123', 'documents');\n * // => 'users/abc123/documents'\n * ```\n */\nexport function joinPath(...segments: string[]): string {\n  return segments\n    .filter((s) => s && s.trim() !== '')\n    .map((s) => s.replace(/^\\/+|\\/+$/g, '')) // Remover / al inicio y final\n    .join('/');\n}\n"]}
@@ -0,0 +1,302 @@
1
+ /**
2
+ * Query Builder
3
+ *
4
+ * Builder fluido para construir queries de Firestore de manera legible.
5
+ * Alternativa más expresiva a pasar objetos QueryOptions directamente.
6
+ */
7
+ /**
8
+ * Builder fluido para queries de Firestore.
9
+ *
10
+ * @example
11
+ * ```typescript
12
+ * // Construir query con builder
13
+ * const options = new QueryBuilder()
14
+ * .where('status', '==', 'active')
15
+ * .where('age', '>=', 18)
16
+ * .orderBy('createdAt', 'desc')
17
+ * .limit(10)
18
+ * .build();
19
+ *
20
+ * // Usar con FirestoreService
21
+ * const users = await firestoreService.getDocs<User>('users', options);
22
+ *
23
+ * // O con método estático
24
+ * const options2 = QueryBuilder.create()
25
+ * .where('category', '==', 'electronics')
26
+ * .orderBy('price', 'asc')
27
+ * .build();
28
+ * ```
29
+ */
30
+ export class QueryBuilder {
31
+ constructor() {
32
+ this.whereConditions = [];
33
+ this.orderByConditions = [];
34
+ }
35
+ /**
36
+ * Crea una nueva instancia del builder (método estático alternativo).
37
+ */
38
+ static create() {
39
+ return new QueryBuilder();
40
+ }
41
+ /**
42
+ * Agrega una condición where.
43
+ *
44
+ * @param field - Campo a filtrar
45
+ * @param operator - Operador de comparación
46
+ * @param value - Valor a comparar
47
+ *
48
+ * @example
49
+ * ```typescript
50
+ * builder.where('status', '==', 'active')
51
+ * builder.where('price', '>=', 100)
52
+ * builder.where('tags', 'array-contains', 'featured')
53
+ * builder.where('category', 'in', ['electronics', 'books'])
54
+ * ```
55
+ */
56
+ where(field, operator, value) {
57
+ this.whereConditions.push({ field, operator, value });
58
+ return this;
59
+ }
60
+ /**
61
+ * Shortcut para where con operador '=='.
62
+ *
63
+ * @example
64
+ * ```typescript
65
+ * builder.whereEquals('status', 'active')
66
+ * // equivalente a: builder.where('status', '==', 'active')
67
+ * ```
68
+ */
69
+ whereEquals(field, value) {
70
+ return this.where(field, '==', value);
71
+ }
72
+ /**
73
+ * Shortcut para where con operador '!='.
74
+ */
75
+ whereNotEquals(field, value) {
76
+ return this.where(field, '!=', value);
77
+ }
78
+ /**
79
+ * Shortcut para where con operador '>'.
80
+ */
81
+ whereGreaterThan(field, value) {
82
+ return this.where(field, '>', value);
83
+ }
84
+ /**
85
+ * Shortcut para where con operador '>='.
86
+ */
87
+ whereGreaterOrEqual(field, value) {
88
+ return this.where(field, '>=', value);
89
+ }
90
+ /**
91
+ * Shortcut para where con operador '<'.
92
+ */
93
+ whereLessThan(field, value) {
94
+ return this.where(field, '<', value);
95
+ }
96
+ /**
97
+ * Shortcut para where con operador '<='.
98
+ */
99
+ whereLessOrEqual(field, value) {
100
+ return this.where(field, '<=', value);
101
+ }
102
+ /**
103
+ * Shortcut para where con operador 'array-contains'.
104
+ *
105
+ * @example
106
+ * ```typescript
107
+ * builder.whereArrayContains('tags', 'featured')
108
+ * ```
109
+ */
110
+ whereArrayContains(field, value) {
111
+ return this.where(field, 'array-contains', value);
112
+ }
113
+ /**
114
+ * Shortcut para where con operador 'array-contains-any'.
115
+ *
116
+ * @example
117
+ * ```typescript
118
+ * builder.whereArrayContainsAny('tags', ['featured', 'new'])
119
+ * ```
120
+ */
121
+ whereArrayContainsAny(field, values) {
122
+ return this.where(field, 'array-contains-any', values);
123
+ }
124
+ /**
125
+ * Shortcut para where con operador 'in'.
126
+ *
127
+ * @example
128
+ * ```typescript
129
+ * builder.whereIn('status', ['active', 'pending'])
130
+ * ```
131
+ */
132
+ whereIn(field, values) {
133
+ return this.where(field, 'in', values);
134
+ }
135
+ /**
136
+ * Shortcut para where con operador 'not-in'.
137
+ */
138
+ whereNotIn(field, values) {
139
+ return this.where(field, 'not-in', values);
140
+ }
141
+ /**
142
+ * Agrega ordenamiento por un campo.
143
+ *
144
+ * @param field - Campo por el cual ordenar
145
+ * @param direction - Dirección: 'asc' o 'desc' (default: 'asc')
146
+ *
147
+ * @example
148
+ * ```typescript
149
+ * builder.orderBy('createdAt', 'desc')
150
+ * builder.orderBy('name') // asc por defecto
151
+ * ```
152
+ */
153
+ orderBy(field, direction = 'asc') {
154
+ this.orderByConditions.push({ field, direction });
155
+ return this;
156
+ }
157
+ /**
158
+ * Shortcut para orderBy descendente.
159
+ */
160
+ orderByDesc(field) {
161
+ return this.orderBy(field, 'desc');
162
+ }
163
+ /**
164
+ * Shortcut para orderBy ascendente.
165
+ */
166
+ orderByAsc(field) {
167
+ return this.orderBy(field, 'asc');
168
+ }
169
+ /**
170
+ * Limita el número de resultados.
171
+ *
172
+ * @param count - Número máximo de documentos
173
+ *
174
+ * @example
175
+ * ```typescript
176
+ * builder.limit(10)
177
+ * ```
178
+ */
179
+ limit(count) {
180
+ if (count <= 0) {
181
+ throw new Error('El límite debe ser mayor a 0');
182
+ }
183
+ this.limitValue = count;
184
+ return this;
185
+ }
186
+ /**
187
+ * Cursor para paginación: empezar después de un documento.
188
+ *
189
+ * @param cursor - Documento o snapshot desde donde continuar
190
+ *
191
+ * @example
192
+ * ```typescript
193
+ * // Primera página
194
+ * const page1 = await service.getPaginated('users', builder.limit(10).build());
195
+ *
196
+ * // Siguiente página
197
+ * const page2 = await service.getPaginated('users',
198
+ * builder.startAfter(page1.lastDoc).limit(10).build()
199
+ * );
200
+ * ```
201
+ */
202
+ startAfter(cursor) {
203
+ this.startAfterValue = cursor;
204
+ return this;
205
+ }
206
+ /**
207
+ * Cursor para paginación: empezar en un documento.
208
+ */
209
+ startAt(cursor) {
210
+ this.startAtValue = cursor;
211
+ return this;
212
+ }
213
+ /**
214
+ * Cursor para paginación: terminar antes de un documento.
215
+ */
216
+ endBefore(cursor) {
217
+ this.endBeforeValue = cursor;
218
+ return this;
219
+ }
220
+ /**
221
+ * Cursor para paginación: terminar en un documento.
222
+ */
223
+ endAt(cursor) {
224
+ this.endAtValue = cursor;
225
+ return this;
226
+ }
227
+ /**
228
+ * Construye el objeto QueryOptions.
229
+ *
230
+ * @returns QueryOptions para usar con FirestoreService
231
+ */
232
+ build() {
233
+ const options = {};
234
+ if (this.whereConditions.length > 0) {
235
+ options.where = [...this.whereConditions];
236
+ }
237
+ if (this.orderByConditions.length > 0) {
238
+ options.orderBy = [...this.orderByConditions];
239
+ }
240
+ if (this.limitValue !== undefined) {
241
+ options.limit = this.limitValue;
242
+ }
243
+ if (this.startAfterValue !== undefined) {
244
+ options.startAfter = this.startAfterValue;
245
+ }
246
+ if (this.startAtValue !== undefined) {
247
+ options.startAt = this.startAtValue;
248
+ }
249
+ if (this.endBeforeValue !== undefined) {
250
+ options.endBefore = this.endBeforeValue;
251
+ }
252
+ if (this.endAtValue !== undefined) {
253
+ options.endAt = this.endAtValue;
254
+ }
255
+ return options;
256
+ }
257
+ /**
258
+ * Resetea el builder para reutilización.
259
+ */
260
+ reset() {
261
+ this.whereConditions = [];
262
+ this.orderByConditions = [];
263
+ this.limitValue = undefined;
264
+ this.startAfterValue = undefined;
265
+ this.startAtValue = undefined;
266
+ this.endBeforeValue = undefined;
267
+ this.endAtValue = undefined;
268
+ return this;
269
+ }
270
+ /**
271
+ * Clona el builder actual.
272
+ */
273
+ clone() {
274
+ const cloned = new QueryBuilder();
275
+ cloned.whereConditions = [...this.whereConditions];
276
+ cloned.orderByConditions = [...this.orderByConditions];
277
+ cloned.limitValue = this.limitValue;
278
+ cloned.startAfterValue = this.startAfterValue;
279
+ cloned.startAtValue = this.startAtValue;
280
+ cloned.endBeforeValue = this.endBeforeValue;
281
+ cloned.endAtValue = this.endAtValue;
282
+ return cloned;
283
+ }
284
+ }
285
+ /**
286
+ * Función helper para crear un QueryBuilder.
287
+ *
288
+ * @example
289
+ * ```typescript
290
+ * import { query } from 'valtech-components';
291
+ *
292
+ * const options = query()
293
+ * .where('status', '==', 'active')
294
+ * .orderBy('createdAt', 'desc')
295
+ * .limit(10)
296
+ * .build();
297
+ * ```
298
+ */
299
+ export function query() {
300
+ return new QueryBuilder();
301
+ }
302
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-builder.js","sourceRoot":"","sources":["../../../../../../../src/lib/services/firebase/utils/query-builder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Query Builder\n *\n * Builder fluido para construir queries de Firestore de manera legible.\n * Alternativa más expresiva a pasar objetos QueryOptions directamente.\n */\n\nimport { OrderByClause, OrderDirection, QueryOptions, WhereClause, WhereOperator } from '../types';\n\n/**\n * Builder fluido para queries de Firestore.\n *\n * @example\n * ```typescript\n * // Construir query con builder\n * const options = new QueryBuilder()\n *   .where('status', '==', 'active')\n *   .where('age', '>=', 18)\n *   .orderBy('createdAt', 'desc')\n *   .limit(10)\n *   .build();\n *\n * // Usar con FirestoreService\n * const users = await firestoreService.getDocs<User>('users', options);\n *\n * // O con método estático\n * const options2 = QueryBuilder.create()\n *   .where('category', '==', 'electronics')\n *   .orderBy('price', 'asc')\n *   .build();\n * ```\n */\nexport class QueryBuilder {\n  private whereConditions: WhereClause[] = [];\n  private orderByConditions: OrderByClause[] = [];\n  private limitValue?: number;\n  private startAfterValue?: unknown;\n  private startAtValue?: unknown;\n  private endBeforeValue?: unknown;\n  private endAtValue?: unknown;\n\n  /**\n   * Crea una nueva instancia del builder (método estático alternativo).\n   */\n  static create(): QueryBuilder {\n    return new QueryBuilder();\n  }\n\n  /**\n   * Agrega una condición where.\n   *\n   * @param field - Campo a filtrar\n   * @param operator - Operador de comparación\n   * @param value - Valor a comparar\n   *\n   * @example\n   * ```typescript\n   * builder.where('status', '==', 'active')\n   * builder.where('price', '>=', 100)\n   * builder.where('tags', 'array-contains', 'featured')\n   * builder.where('category', 'in', ['electronics', 'books'])\n   * ```\n   */\n  where(field: string, operator: WhereOperator, value: unknown): QueryBuilder {\n    this.whereConditions.push({ field, operator, value });\n    return this;\n  }\n\n  /**\n   * Shortcut para where con operador '=='.\n   *\n   * @example\n   * ```typescript\n   * builder.whereEquals('status', 'active')\n   * // equivalente a: builder.where('status', '==', 'active')\n   * ```\n   */\n  whereEquals(field: string, value: unknown): QueryBuilder {\n    return this.where(field, '==', value);\n  }\n\n  /**\n   * Shortcut para where con operador '!='.\n   */\n  whereNotEquals(field: string, value: unknown): QueryBuilder {\n    return this.where(field, '!=', value);\n  }\n\n  /**\n   * Shortcut para where con operador '>'.\n   */\n  whereGreaterThan(field: string, value: unknown): QueryBuilder {\n    return this.where(field, '>', value);\n  }\n\n  /**\n   * Shortcut para where con operador '>='.\n   */\n  whereGreaterOrEqual(field: string, value: unknown): QueryBuilder {\n    return this.where(field, '>=', value);\n  }\n\n  /**\n   * Shortcut para where con operador '<'.\n   */\n  whereLessThan(field: string, value: unknown): QueryBuilder {\n    return this.where(field, '<', value);\n  }\n\n  /**\n   * Shortcut para where con operador '<='.\n   */\n  whereLessOrEqual(field: string, value: unknown): QueryBuilder {\n    return this.where(field, '<=', value);\n  }\n\n  /**\n   * Shortcut para where con operador 'array-contains'.\n   *\n   * @example\n   * ```typescript\n   * builder.whereArrayContains('tags', 'featured')\n   * ```\n   */\n  whereArrayContains(field: string, value: unknown): QueryBuilder {\n    return this.where(field, 'array-contains', value);\n  }\n\n  /**\n   * Shortcut para where con operador 'array-contains-any'.\n   *\n   * @example\n   * ```typescript\n   * builder.whereArrayContainsAny('tags', ['featured', 'new'])\n   * ```\n   */\n  whereArrayContainsAny(field: string, values: unknown[]): QueryBuilder {\n    return this.where(field, 'array-contains-any', values);\n  }\n\n  /**\n   * Shortcut para where con operador 'in'.\n   *\n   * @example\n   * ```typescript\n   * builder.whereIn('status', ['active', 'pending'])\n   * ```\n   */\n  whereIn(field: string, values: unknown[]): QueryBuilder {\n    return this.where(field, 'in', values);\n  }\n\n  /**\n   * Shortcut para where con operador 'not-in'.\n   */\n  whereNotIn(field: string, values: unknown[]): QueryBuilder {\n    return this.where(field, 'not-in', values);\n  }\n\n  /**\n   * Agrega ordenamiento por un campo.\n   *\n   * @param field - Campo por el cual ordenar\n   * @param direction - Dirección: 'asc' o 'desc' (default: 'asc')\n   *\n   * @example\n   * ```typescript\n   * builder.orderBy('createdAt', 'desc')\n   * builder.orderBy('name') // asc por defecto\n   * ```\n   */\n  orderBy(field: string, direction: OrderDirection = 'asc'): QueryBuilder {\n    this.orderByConditions.push({ field, direction });\n    return this;\n  }\n\n  /**\n   * Shortcut para orderBy descendente.\n   */\n  orderByDesc(field: string): QueryBuilder {\n    return this.orderBy(field, 'desc');\n  }\n\n  /**\n   * Shortcut para orderBy ascendente.\n   */\n  orderByAsc(field: string): QueryBuilder {\n    return this.orderBy(field, 'asc');\n  }\n\n  /**\n   * Limita el número de resultados.\n   *\n   * @param count - Número máximo de documentos\n   *\n   * @example\n   * ```typescript\n   * builder.limit(10)\n   * ```\n   */\n  limit(count: number): QueryBuilder {\n    if (count <= 0) {\n      throw new Error('El límite debe ser mayor a 0');\n    }\n    this.limitValue = count;\n    return this;\n  }\n\n  /**\n   * Cursor para paginación: empezar después de un documento.\n   *\n   * @param cursor - Documento o snapshot desde donde continuar\n   *\n   * @example\n   * ```typescript\n   * // Primera página\n   * const page1 = await service.getPaginated('users', builder.limit(10).build());\n   *\n   * // Siguiente página\n   * const page2 = await service.getPaginated('users',\n   *   builder.startAfter(page1.lastDoc).limit(10).build()\n   * );\n   * ```\n   */\n  startAfter(cursor: unknown): QueryBuilder {\n    this.startAfterValue = cursor;\n    return this;\n  }\n\n  /**\n   * Cursor para paginación: empezar en un documento.\n   */\n  startAt(cursor: unknown): QueryBuilder {\n    this.startAtValue = cursor;\n    return this;\n  }\n\n  /**\n   * Cursor para paginación: terminar antes de un documento.\n   */\n  endBefore(cursor: unknown): QueryBuilder {\n    this.endBeforeValue = cursor;\n    return this;\n  }\n\n  /**\n   * Cursor para paginación: terminar en un documento.\n   */\n  endAt(cursor: unknown): QueryBuilder {\n    this.endAtValue = cursor;\n    return this;\n  }\n\n  /**\n   * Construye el objeto QueryOptions.\n   *\n   * @returns QueryOptions para usar con FirestoreService\n   */\n  build(): QueryOptions {\n    const options: QueryOptions = {};\n\n    if (this.whereConditions.length > 0) {\n      options.where = [...this.whereConditions];\n    }\n\n    if (this.orderByConditions.length > 0) {\n      options.orderBy = [...this.orderByConditions];\n    }\n\n    if (this.limitValue !== undefined) {\n      options.limit = this.limitValue;\n    }\n\n    if (this.startAfterValue !== undefined) {\n      options.startAfter = this.startAfterValue;\n    }\n\n    if (this.startAtValue !== undefined) {\n      options.startAt = this.startAtValue;\n    }\n\n    if (this.endBeforeValue !== undefined) {\n      options.endBefore = this.endBeforeValue;\n    }\n\n    if (this.endAtValue !== undefined) {\n      options.endAt = this.endAtValue;\n    }\n\n    return options;\n  }\n\n  /**\n   * Resetea el builder para reutilización.\n   */\n  reset(): QueryBuilder {\n    this.whereConditions = [];\n    this.orderByConditions = [];\n    this.limitValue = undefined;\n    this.startAfterValue = undefined;\n    this.startAtValue = undefined;\n    this.endBeforeValue = undefined;\n    this.endAtValue = undefined;\n    return this;\n  }\n\n  /**\n   * Clona el builder actual.\n   */\n  clone(): QueryBuilder {\n    const cloned = new QueryBuilder();\n    cloned.whereConditions = [...this.whereConditions];\n    cloned.orderByConditions = [...this.orderByConditions];\n    cloned.limitValue = this.limitValue;\n    cloned.startAfterValue = this.startAfterValue;\n    cloned.startAtValue = this.startAtValue;\n    cloned.endBeforeValue = this.endBeforeValue;\n    cloned.endAtValue = this.endAtValue;\n    return cloned;\n  }\n}\n\n/**\n * Función helper para crear un QueryBuilder.\n *\n * @example\n * ```typescript\n * import { query } from 'valtech-components';\n *\n * const options = query()\n *   .where('status', '==', 'active')\n *   .orderBy('createdAt', 'desc')\n *   .limit(10)\n *   .build();\n * ```\n */\nexport function query(): QueryBuilder {\n  return new QueryBuilder();\n}\n"]}
@@ -5,7 +5,7 @@
5
5
  */
6
6
  import { Injectable } from '@angular/core';
7
7
  import { addIcons } from 'ionicons';
8
- import { addCircleOutline, addOutline, alertCircleOutline, alertOutline, arrowBackOutline, arrowDownOutline, arrowForwardOutline, businessOutline, calendarOutline, chatbubblesOutline, checkmarkCircleOutline, checkmarkOutline, chevronDownOutline, chevronForwardOutline, clipboardOutline, closeOutline, copyOutline, createOutline, documentTextOutline, ellipsisHorizontalOutline, eyeOffOutline, eyeOutline, filterOutline, heart, heartOutline, homeOutline, informationCircleOutline, locationOutline, lockClosedOutline, logoFacebook, logoInstagram, logoLinkedin, logoNpm, logoTiktok, logoTwitter, logoYoutube, notificationsOutline, openOutline, playOutline, refreshOutline, removeOutline, scanOutline, shareOutline, trashOutline, } from 'ionicons/icons';
8
+ import { addCircleOutline, addOutline, alertCircleOutline, alertOutline, arrowBackOutline, arrowDownOutline, arrowForwardOutline, businessOutline, calendarOutline, chatbubblesOutline, checkmarkCircleOutline, checkmarkOutline, chevronDownOutline, chevronForwardOutline, clipboardOutline, closeOutline, copyOutline, createOutline, documentTextOutline, ellipsisHorizontalOutline, eyeOffOutline, eyeOutline, filterOutline, heart, heartOutline, homeOutline, informationCircleOutline, locationOutline, lockClosedOutline, logoFacebook, logoGoogle, logoInstagram, logoLinkedin, logoNpm, logoTiktok, logoTwitter, logoYoutube, notificationsOutline, openOutline, playOutline, refreshOutline, removeOutline, scanOutline, shareOutline, trashOutline, } from 'ionicons/icons';
9
9
  import * as i0 from "@angular/core";
10
10
  export class IconService {
11
11
  /**
@@ -48,6 +48,7 @@ export class IconService {
48
48
  logoYoutube,
49
49
  logoTiktok,
50
50
  logoFacebook,
51
+ logoGoogle,
51
52
  createOutline,
52
53
  trashOutline,
53
54
  playOutline,
@@ -68,4 +69,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
68
69
  providedIn: 'root',
69
70
  }]
70
71
  }], ctorParameters: () => [] });
71
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWNvbnMuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvc2VydmljZXMvaWNvbnMuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSx5Q0FBeUM7QUFFekM7OztHQUdHO0FBRUgsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzQyxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQ3BDLE9BQU8sRUFDTCxnQkFBZ0IsRUFDaEIsVUFBVSxFQUNWLGtCQUFrQixFQUNsQixZQUFZLEVBQ1osZ0JBQWdCLEVBQ2hCLGdCQUFnQixFQUNoQixtQkFBbUIsRUFDbkIsZUFBZSxFQUNmLGVBQWUsRUFDZixrQkFBa0IsRUFDbEIsc0JBQXNCLEVBQ3RCLGdCQUFnQixFQUNoQixrQkFBa0IsRUFDbEIscUJBQXFCLEVBQ3JCLGdCQUFnQixFQUNoQixZQUFZLEVBQ1osV0FBVyxFQUNYLGFBQWEsRUFDYixtQkFBbUIsRUFDbkIseUJBQXlCLEVBQ3pCLGFBQWEsRUFDYixVQUFVLEVBQ1YsYUFBYSxFQUNiLEtBQUssRUFDTCxZQUFZLEVBQ1osV0FBVyxFQUNYLHdCQUF3QixFQUN4QixlQUFlLEVBQ2YsaUJBQWlCLEVBQ2pCLFlBQVksRUFDWixhQUFhLEVBQ2IsWUFBWSxFQUNaLE9BQU8sRUFDUCxVQUFVLEVBQ1YsV0FBVyxFQUNYLFdBQVcsRUFDWCxvQkFBb0IsRUFDcEIsV0FBVyxFQUNYLFdBQVcsRUFDWCxjQUFjLEVBQ2QsYUFBYSxFQUNiLFdBQVcsRUFDWCxZQUFZLEVBQ1osWUFBWSxHQUNiLE1BQU0sZ0JBQWdCLENBQUM7O0FBS3hCLE1BQU0sT0FBTyxXQUFXO0lBQ3RCOztPQUVHO0lBQ0g7UUFDRSxRQUFRLENBQUM7WUFDUCxVQUFVO1lBQ1YsZ0JBQWdCO1lBQ2hCLFlBQVk7WUFDWixrQkFBa0I7WUFDbEIsZ0JBQWdCO1lBQ2hCLG1CQUFtQjtZQUNuQixnQkFBZ0I7WUFDaEIsc0JBQXNCO1lBQ3RCLHlCQUF5QjtZQUN6QixvQkFBb0I7WUFDcEIsV0FBVztZQUNYLFlBQVk7WUFDWixrQkFBa0I7WUFDbEIsWUFBWTtZQUNaLEtBQUs7WUFDTCxZQUFZO1lBQ1osV0FBVztZQUNYLGFBQWE7WUFDYixVQUFVO1lBQ1YsV0FBVztZQUNYLGtCQUFrQjtZQUNsQixxQkFBcUI7WUFDckIsZ0JBQWdCO1lBQ2hCLGdCQUFnQjtZQUNoQixXQUFXO1lBQ1gsYUFBYTtZQUNiLGVBQWU7WUFDZixlQUFlO1lBQ2YsZUFBZTtZQUNmLFdBQVc7WUFDWCxhQUFhO1lBQ2IsWUFBWTtZQUNaLFdBQVc7WUFDWCxVQUFVO1lBQ1YsWUFBWTtZQUNaLGFBQWE7WUFDYixZQUFZO1lBQ1osV0FBVztZQUNYLGNBQWM7WUFDZCxtQkFBbUI7WUFDbkIsaUJBQWlCO1lBQ2pCLHdCQUF3QjtZQUN4QixPQUFPO1lBQ1AsYUFBYTtTQUNkLENBQUMsQ0FBQztJQUNMLENBQUM7K0dBbkRVLFdBQVc7bUhBQVgsV0FBVyxjQUZWLE1BQU07OzRGQUVQLFdBQVc7a0JBSHZCLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgbm8tdW5kZXJzY29yZS1kYW5nbGUgKi9cblxuLyoqXG4gKiBTZXJ2aWNlIGZvciByZWdpc3RlcmluZyBhbmQgbWFuYWdpbmcgY3VzdG9tIElvbmljb25zIGljb25zIGluIHRoZSBhcHBsaWNhdGlvbi5cbiAqIEF1dG9tYXRpY2FsbHkgcmVnaXN0ZXJzIGEgc2V0IG9mIGNvbW1vbmx5IHVzZWQgaWNvbnMgb24gaW5pdGlhbGl6YXRpb24uXG4gKi9cblxuaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgYWRkSWNvbnMgfSBmcm9tICdpb25pY29ucyc7XG5pbXBvcnQge1xuICBhZGRDaXJjbGVPdXRsaW5lLFxuICBhZGRPdXRsaW5lLFxuICBhbGVydENpcmNsZU91dGxpbmUsXG4gIGFsZXJ0T3V0bGluZSxcbiAgYXJyb3dCYWNrT3V0bGluZSxcbiAgYXJyb3dEb3duT3V0bGluZSxcbiAgYXJyb3dGb3J3YXJkT3V0bGluZSxcbiAgYnVzaW5lc3NPdXRsaW5lLFxuICBjYWxlbmRhck91dGxpbmUsXG4gIGNoYXRidWJibGVzT3V0bGluZSxcbiAgY2hlY2ttYXJrQ2lyY2xlT3V0bGluZSxcbiAgY2hlY2ttYXJrT3V0bGluZSxcbiAgY2hldnJvbkRvd25PdXRsaW5lLFxuICBjaGV2cm9uRm9yd2FyZE91dGxpbmUsXG4gIGNsaXBib2FyZE91dGxpbmUsXG4gIGNsb3NlT3V0bGluZSxcbiAgY29weU91dGxpbmUsXG4gIGNyZWF0ZU91dGxpbmUsXG4gIGRvY3VtZW50VGV4dE91dGxpbmUsXG4gIGVsbGlwc2lzSG9yaXpvbnRhbE91dGxpbmUsXG4gIGV5ZU9mZk91dGxpbmUsXG4gIGV5ZU91dGxpbmUsXG4gIGZpbHRlck91dGxpbmUsXG4gIGhlYXJ0LFxuICBoZWFydE91dGxpbmUsXG4gIGhvbWVPdXRsaW5lLFxuICBpbmZvcm1hdGlvbkNpcmNsZU91dGxpbmUsXG4gIGxvY2F0aW9uT3V0bGluZSxcbiAgbG9ja0Nsb3NlZE91dGxpbmUsXG4gIGxvZ29GYWNlYm9vayxcbiAgbG9nb0luc3RhZ3JhbSxcbiAgbG9nb0xpbmtlZGluLFxuICBsb2dvTnBtLFxuICBsb2dvVGlrdG9rLFxuICBsb2dvVHdpdHRlcixcbiAgbG9nb1lvdXR1YmUsXG4gIG5vdGlmaWNhdGlvbnNPdXRsaW5lLFxuICBvcGVuT3V0bGluZSxcbiAgcGxheU91dGxpbmUsXG4gIHJlZnJlc2hPdXRsaW5lLFxuICByZW1vdmVPdXRsaW5lLFxuICBzY2FuT3V0bGluZSxcbiAgc2hhcmVPdXRsaW5lLFxuICB0cmFzaE91dGxpbmUsXG59IGZyb20gJ2lvbmljb25zL2ljb25zJztcblxuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCcsXG59KVxuZXhwb3J0IGNsYXNzIEljb25TZXJ2aWNlIHtcbiAgLyoqXG4gICAqIFJlZ2lzdGVycyBhIHNldCBvZiBJb25pY29ucyBpY29ucyBmb3IgdXNlIHRocm91Z2hvdXQgdGhlIGFwcC5cbiAgICovXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIGFkZEljb25zKHtcbiAgICAgIGFkZE91dGxpbmUsXG4gICAgICBhZGRDaXJjbGVPdXRsaW5lLFxuICAgICAgYWxlcnRPdXRsaW5lLFxuICAgICAgYWxlcnRDaXJjbGVPdXRsaW5lLFxuICAgICAgYXJyb3dCYWNrT3V0bGluZSxcbiAgICAgIGFycm93Rm9yd2FyZE91dGxpbmUsXG4gICAgICBhcnJvd0Rvd25PdXRsaW5lLFxuICAgICAgY2hlY2ttYXJrQ2lyY2xlT3V0bGluZSxcbiAgICAgIGVsbGlwc2lzSG9yaXpvbnRhbE91dGxpbmUsXG4gICAgICBub3RpZmljYXRpb25zT3V0bGluZSxcbiAgICAgIG9wZW5PdXRsaW5lLFxuICAgICAgY2xvc2VPdXRsaW5lLFxuICAgICAgY2hhdGJ1YmJsZXNPdXRsaW5lLFxuICAgICAgc2hhcmVPdXRsaW5lLFxuICAgICAgaGVhcnQsXG4gICAgICBoZWFydE91dGxpbmUsXG4gICAgICBob21lT3V0bGluZSxcbiAgICAgIGV5ZU9mZk91dGxpbmUsXG4gICAgICBleWVPdXRsaW5lLFxuICAgICAgc2Nhbk91dGxpbmUsXG4gICAgICBjaGV2cm9uRG93bk91dGxpbmUsXG4gICAgICBjaGV2cm9uRm9yd2FyZE91dGxpbmUsXG4gICAgICBjaGVja21hcmtPdXRsaW5lLFxuICAgICAgY2xpcGJvYXJkT3V0bGluZSxcbiAgICAgIGNvcHlPdXRsaW5lLFxuICAgICAgZmlsdGVyT3V0bGluZSxcbiAgICAgIGxvY2F0aW9uT3V0bGluZSxcbiAgICAgIGNhbGVuZGFyT3V0bGluZSxcbiAgICAgIGJ1c2luZXNzT3V0bGluZSxcbiAgICAgIGxvZ29Ud2l0dGVyLFxuICAgICAgbG9nb0luc3RhZ3JhbSxcbiAgICAgIGxvZ29MaW5rZWRpbixcbiAgICAgIGxvZ29Zb3V0dWJlLFxuICAgICAgbG9nb1Rpa3RvayxcbiAgICAgIGxvZ29GYWNlYm9vayxcbiAgICAgIGNyZWF0ZU91dGxpbmUsXG4gICAgICB0cmFzaE91dGxpbmUsXG4gICAgICBwbGF5T3V0bGluZSxcbiAgICAgIHJlZnJlc2hPdXRsaW5lLFxuICAgICAgZG9jdW1lbnRUZXh0T3V0bGluZSxcbiAgICAgIGxvY2tDbG9zZWRPdXRsaW5lLFxuICAgICAgaW5mb3JtYXRpb25DaXJjbGVPdXRsaW5lLFxuICAgICAgbG9nb05wbSxcbiAgICAgIHJlbW92ZU91dGxpbmUsXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
72
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWNvbnMuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvc2VydmljZXMvaWNvbnMuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSx5Q0FBeUM7QUFFekM7OztHQUdHO0FBRUgsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzQyxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQ3BDLE9BQU8sRUFDTCxnQkFBZ0IsRUFDaEIsVUFBVSxFQUNWLGtCQUFrQixFQUNsQixZQUFZLEVBQ1osZ0JBQWdCLEVBQ2hCLGdCQUFnQixFQUNoQixtQkFBbUIsRUFDbkIsZUFBZSxFQUNmLGVBQWUsRUFDZixrQkFBa0IsRUFDbEIsc0JBQXNCLEVBQ3RCLGdCQUFnQixFQUNoQixrQkFBa0IsRUFDbEIscUJBQXFCLEVBQ3JCLGdCQUFnQixFQUNoQixZQUFZLEVBQ1osV0FBVyxFQUNYLGFBQWEsRUFDYixtQkFBbUIsRUFDbkIseUJBQXlCLEVBQ3pCLGFBQWEsRUFDYixVQUFVLEVBQ1YsYUFBYSxFQUNiLEtBQUssRUFDTCxZQUFZLEVBQ1osV0FBVyxFQUNYLHdCQUF3QixFQUN4QixlQUFlLEVBQ2YsaUJBQWlCLEVBQ2pCLFlBQVksRUFDWixVQUFVLEVBQ1YsYUFBYSxFQUNiLFlBQVksRUFDWixPQUFPLEVBQ1AsVUFBVSxFQUNWLFdBQVcsRUFDWCxXQUFXLEVBQ1gsb0JBQW9CLEVBQ3BCLFdBQVcsRUFDWCxXQUFXLEVBQ1gsY0FBYyxFQUNkLGFBQWEsRUFDYixXQUFXLEVBQ1gsWUFBWSxFQUNaLFlBQVksR0FDYixNQUFNLGdCQUFnQixDQUFDOztBQUt4QixNQUFNLE9BQU8sV0FBVztJQUN0Qjs7T0FFRztJQUNIO1FBQ0UsUUFBUSxDQUFDO1lBQ1AsVUFBVTtZQUNWLGdCQUFnQjtZQUNoQixZQUFZO1lBQ1osa0JBQWtCO1lBQ2xCLGdCQUFnQjtZQUNoQixtQkFBbUI7WUFDbkIsZ0JBQWdCO1lBQ2hCLHNCQUFzQjtZQUN0Qix5QkFBeUI7WUFDekIsb0JBQW9CO1lBQ3BCLFdBQVc7WUFDWCxZQUFZO1lBQ1osa0JBQWtCO1lBQ2xCLFlBQVk7WUFDWixLQUFLO1lBQ0wsWUFBWTtZQUNaLFdBQVc7WUFDWCxhQUFhO1lBQ2IsVUFBVTtZQUNWLFdBQVc7WUFDWCxrQkFBa0I7WUFDbEIscUJBQXFCO1lBQ3JCLGdCQUFnQjtZQUNoQixnQkFBZ0I7WUFDaEIsV0FBVztZQUNYLGFBQWE7WUFDYixlQUFlO1lBQ2YsZUFBZTtZQUNmLGVBQWU7WUFDZixXQUFXO1lBQ1gsYUFBYTtZQUNiLFlBQVk7WUFDWixXQUFXO1lBQ1gsVUFBVTtZQUNWLFlBQVk7WUFDWixVQUFVO1lBQ1YsYUFBYTtZQUNiLFlBQVk7WUFDWixXQUFXO1lBQ1gsY0FBYztZQUNkLG1CQUFtQjtZQUNuQixpQkFBaUI7WUFDakIsd0JBQXdCO1lBQ3hCLE9BQU87WUFDUCxhQUFhO1NBQ2QsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzsrR0FwRFUsV0FBVzttSEFBWCxXQUFXLGNBRlYsTUFBTTs7NEZBRVAsV0FBVztrQkFIdkIsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby11bmRlcnNjb3JlLWRhbmdsZSAqL1xuXG4vKipcbiAqIFNlcnZpY2UgZm9yIHJlZ2lzdGVyaW5nIGFuZCBtYW5hZ2luZyBjdXN0b20gSW9uaWNvbnMgaWNvbnMgaW4gdGhlIGFwcGxpY2F0aW9uLlxuICogQXV0b21hdGljYWxseSByZWdpc3RlcnMgYSBzZXQgb2YgY29tbW9ubHkgdXNlZCBpY29ucyBvbiBpbml0aWFsaXphdGlvbi5cbiAqL1xuXG5pbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBhZGRJY29ucyB9IGZyb20gJ2lvbmljb25zJztcbmltcG9ydCB7XG4gIGFkZENpcmNsZU91dGxpbmUsXG4gIGFkZE91dGxpbmUsXG4gIGFsZXJ0Q2lyY2xlT3V0bGluZSxcbiAgYWxlcnRPdXRsaW5lLFxuICBhcnJvd0JhY2tPdXRsaW5lLFxuICBhcnJvd0Rvd25PdXRsaW5lLFxuICBhcnJvd0ZvcndhcmRPdXRsaW5lLFxuICBidXNpbmVzc091dGxpbmUsXG4gIGNhbGVuZGFyT3V0bGluZSxcbiAgY2hhdGJ1YmJsZXNPdXRsaW5lLFxuICBjaGVja21hcmtDaXJjbGVPdXRsaW5lLFxuICBjaGVja21hcmtPdXRsaW5lLFxuICBjaGV2cm9uRG93bk91dGxpbmUsXG4gIGNoZXZyb25Gb3J3YXJkT3V0bGluZSxcbiAgY2xpcGJvYXJkT3V0bGluZSxcbiAgY2xvc2VPdXRsaW5lLFxuICBjb3B5T3V0bGluZSxcbiAgY3JlYXRlT3V0bGluZSxcbiAgZG9jdW1lbnRUZXh0T3V0bGluZSxcbiAgZWxsaXBzaXNIb3Jpem9udGFsT3V0bGluZSxcbiAgZXllT2ZmT3V0bGluZSxcbiAgZXllT3V0bGluZSxcbiAgZmlsdGVyT3V0bGluZSxcbiAgaGVhcnQsXG4gIGhlYXJ0T3V0bGluZSxcbiAgaG9tZU91dGxpbmUsXG4gIGluZm9ybWF0aW9uQ2lyY2xlT3V0bGluZSxcbiAgbG9jYXRpb25PdXRsaW5lLFxuICBsb2NrQ2xvc2VkT3V0bGluZSxcbiAgbG9nb0ZhY2Vib29rLFxuICBsb2dvR29vZ2xlLFxuICBsb2dvSW5zdGFncmFtLFxuICBsb2dvTGlua2VkaW4sXG4gIGxvZ29OcG0sXG4gIGxvZ29UaWt0b2ssXG4gIGxvZ29Ud2l0dGVyLFxuICBsb2dvWW91dHViZSxcbiAgbm90aWZpY2F0aW9uc091dGxpbmUsXG4gIG9wZW5PdXRsaW5lLFxuICBwbGF5T3V0bGluZSxcbiAgcmVmcmVzaE91dGxpbmUsXG4gIHJlbW92ZU91dGxpbmUsXG4gIHNjYW5PdXRsaW5lLFxuICBzaGFyZU91dGxpbmUsXG4gIHRyYXNoT3V0bGluZSxcbn0gZnJvbSAnaW9uaWNvbnMvaWNvbnMnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgSWNvblNlcnZpY2Uge1xuICAvKipcbiAgICogUmVnaXN0ZXJzIGEgc2V0IG9mIElvbmljb25zIGljb25zIGZvciB1c2UgdGhyb3VnaG91dCB0aGUgYXBwLlxuICAgKi9cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgYWRkSWNvbnMoe1xuICAgICAgYWRkT3V0bGluZSxcbiAgICAgIGFkZENpcmNsZU91dGxpbmUsXG4gICAgICBhbGVydE91dGxpbmUsXG4gICAgICBhbGVydENpcmNsZU91dGxpbmUsXG4gICAgICBhcnJvd0JhY2tPdXRsaW5lLFxuICAgICAgYXJyb3dGb3J3YXJkT3V0bGluZSxcbiAgICAgIGFycm93RG93bk91dGxpbmUsXG4gICAgICBjaGVja21hcmtDaXJjbGVPdXRsaW5lLFxuICAgICAgZWxsaXBzaXNIb3Jpem9udGFsT3V0bGluZSxcbiAgICAgIG5vdGlmaWNhdGlvbnNPdXRsaW5lLFxuICAgICAgb3Blbk91dGxpbmUsXG4gICAgICBjbG9zZU91dGxpbmUsXG4gICAgICBjaGF0YnViYmxlc091dGxpbmUsXG4gICAgICBzaGFyZU91dGxpbmUsXG4gICAgICBoZWFydCxcbiAgICAgIGhlYXJ0T3V0bGluZSxcbiAgICAgIGhvbWVPdXRsaW5lLFxuICAgICAgZXllT2ZmT3V0bGluZSxcbiAgICAgIGV5ZU91dGxpbmUsXG4gICAgICBzY2FuT3V0bGluZSxcbiAgICAgIGNoZXZyb25Eb3duT3V0bGluZSxcbiAgICAgIGNoZXZyb25Gb3J3YXJkT3V0bGluZSxcbiAgICAgIGNoZWNrbWFya091dGxpbmUsXG4gICAgICBjbGlwYm9hcmRPdXRsaW5lLFxuICAgICAgY29weU91dGxpbmUsXG4gICAgICBmaWx0ZXJPdXRsaW5lLFxuICAgICAgbG9jYXRpb25PdXRsaW5lLFxuICAgICAgY2FsZW5kYXJPdXRsaW5lLFxuICAgICAgYnVzaW5lc3NPdXRsaW5lLFxuICAgICAgbG9nb1R3aXR0ZXIsXG4gICAgICBsb2dvSW5zdGFncmFtLFxuICAgICAgbG9nb0xpbmtlZGluLFxuICAgICAgbG9nb1lvdXR1YmUsXG4gICAgICBsb2dvVGlrdG9rLFxuICAgICAgbG9nb0ZhY2Vib29rLFxuICAgICAgbG9nb0dvb2dsZSxcbiAgICAgIGNyZWF0ZU91dGxpbmUsXG4gICAgICB0cmFzaE91dGxpbmUsXG4gICAgICBwbGF5T3V0bGluZSxcbiAgICAgIHJlZnJlc2hPdXRsaW5lLFxuICAgICAgZG9jdW1lbnRUZXh0T3V0bGluZSxcbiAgICAgIGxvY2tDbG9zZWRPdXRsaW5lLFxuICAgICAgaW5mb3JtYXRpb25DaXJjbGVPdXRsaW5lLFxuICAgICAgbG9nb05wbSxcbiAgICAgIHJlbW92ZU91dGxpbmUsXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==