valtech-components 2.0.498 → 2.0.501

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 (49) hide show
  1. package/esm2022/lib/components/atoms/button/button.component.mjs +87 -48
  2. package/esm2022/lib/components/molecules/action-header/action-header.component.mjs +1 -1
  3. package/esm2022/lib/components/molecules/ad-slot/ad-slot.component.mjs +249 -0
  4. package/esm2022/lib/components/molecules/button-group/button-group.component.mjs +1 -1
  5. package/esm2022/lib/components/molecules/card/card.component.mjs +2 -2
  6. package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +1 -1
  7. package/esm2022/lib/components/molecules/raffle-status-card/raffle-status-card.component.mjs +2 -2
  8. package/esm2022/lib/components/organisms/article/article.component.mjs +2 -2
  9. package/esm2022/lib/components/organisms/menu/menu.component.mjs +1 -1
  10. package/esm2022/lib/components/templates/page-template/page-template.component.mjs +1 -1
  11. package/esm2022/lib/services/ads/ads-consent.service.mjs +152 -0
  12. package/esm2022/lib/services/ads/ads-loader.service.mjs +160 -0
  13. package/esm2022/lib/services/ads/ads.service.mjs +449 -0
  14. package/esm2022/lib/services/ads/config.mjs +118 -0
  15. package/esm2022/lib/services/ads/index.mjs +14 -0
  16. package/esm2022/lib/services/ads/types.mjs +23 -0
  17. package/esm2022/lib/services/auth/auth.service.mjs +103 -6
  18. package/esm2022/lib/services/auth/index.mjs +4 -1
  19. package/esm2022/lib/services/auth/oauth-callback.component.mjs +141 -0
  20. package/esm2022/lib/services/auth/oauth.service.mjs +250 -0
  21. package/esm2022/lib/services/auth/types.mjs +1 -1
  22. package/esm2022/lib/services/firebase/analytics-error-handler.mjs +141 -0
  23. package/esm2022/lib/services/firebase/analytics-router-tracker.mjs +99 -0
  24. package/esm2022/lib/services/firebase/analytics.service.mjs +597 -0
  25. package/esm2022/lib/services/firebase/config.mjs +21 -2
  26. package/esm2022/lib/services/firebase/index.mjs +6 -1
  27. package/esm2022/public-api.mjs +6 -1
  28. package/fesm2022/valtech-components.mjs +2739 -239
  29. package/fesm2022/valtech-components.mjs.map +1 -1
  30. package/lib/components/atoms/button/button.component.d.ts +30 -6
  31. package/lib/components/molecules/ad-slot/ad-slot.component.d.ts +78 -0
  32. package/lib/components/organisms/article/article.component.d.ts +3 -3
  33. package/lib/services/ads/ads-consent.service.d.ts +59 -0
  34. package/lib/services/ads/ads-loader.service.d.ts +46 -0
  35. package/lib/services/ads/ads.service.d.ts +123 -0
  36. package/lib/services/ads/config.d.ts +69 -0
  37. package/lib/services/ads/index.d.ts +10 -0
  38. package/lib/services/ads/types.d.ts +163 -0
  39. package/lib/services/auth/auth.service.d.ts +56 -3
  40. package/lib/services/auth/index.d.ts +2 -0
  41. package/lib/services/auth/oauth-callback.component.d.ts +34 -0
  42. package/lib/services/auth/oauth.service.d.ts +90 -0
  43. package/lib/services/auth/types.d.ts +69 -0
  44. package/lib/services/firebase/analytics-error-handler.d.ts +54 -0
  45. package/lib/services/firebase/analytics-router-tracker.d.ts +51 -0
  46. package/lib/services/firebase/analytics.service.d.ts +256 -0
  47. package/lib/services/firebase/index.d.ts +4 -0
  48. package/package.json +1 -1
  49. package/public-api.d.ts +2 -0
@@ -0,0 +1,597 @@
1
+ /**
2
+ * Analytics Service (Firebase GA4)
3
+ *
4
+ * Servicio para tracking de eventos, page views y errores con Firebase Analytics.
5
+ * Integra con el sistema de auth para user properties y respeta consent mode GDPR.
6
+ */
7
+ import { Inject, Injectable, PLATFORM_ID, signal, computed } from '@angular/core';
8
+ import { isPlatformBrowser } from '@angular/common';
9
+ import { Analytics, logEvent, setUserId, setUserProperties } from '@angular/fire/analytics';
10
+ import { VALTECH_FIREBASE_CONFIG } from './config';
11
+ import * as i0 from "@angular/core";
12
+ /** Key por defecto para persistir consent en localStorage */
13
+ const DEFAULT_CONSENT_STORAGE_KEY = 'analytics_consent';
14
+ /** Máximo de eventos en historial de debug */
15
+ const MAX_DEBUG_HISTORY = 100;
16
+ /**
17
+ * Servicio de Firebase Analytics (GA4).
18
+ *
19
+ * Proporciona tracking de eventos, page views, errores y métricas de performance.
20
+ * Soporta GDPR Consent Mode y modo debug para desarrollo.
21
+ *
22
+ * @example
23
+ * ```typescript
24
+ * @Component({...})
25
+ * export class ProductComponent {
26
+ * private analytics = inject(AnalyticsService);
27
+ *
28
+ * onPurchase(product: Product) {
29
+ * this.analytics.logEvent('purchase', {
30
+ * transaction_id: order.id,
31
+ * value: order.total,
32
+ * currency: 'EUR'
33
+ * });
34
+ * }
35
+ * }
36
+ * ```
37
+ */
38
+ export class AnalyticsService {
39
+ constructor(injector, config, platformId) {
40
+ this.injector = injector;
41
+ this.config = config;
42
+ this.platformId = platformId;
43
+ // ===========================================================================
44
+ // ESTADO (Signals)
45
+ // ===========================================================================
46
+ this._consentState = signal({
47
+ settings: {},
48
+ updatedAt: null,
49
+ hasDecided: false,
50
+ });
51
+ this._isDebugMode = signal(false);
52
+ this._debugHistory = signal([]);
53
+ /** Estado de consentimiento actual (readonly) */
54
+ this.consentState = this._consentState.asReadonly();
55
+ /** Indica si está en modo debug */
56
+ this.isDebugMode = this._isDebugMode.asReadonly();
57
+ /** Indica si analytics está habilitado y funcionando */
58
+ this.isEnabled = computed(() => {
59
+ return this.isAnalyticsSupported() && this._consentState().settings.analytics === true;
60
+ });
61
+ this.analyticsConfig = config.analyticsConfig ?? {};
62
+ this.consentStorageKey =
63
+ this.analyticsConfig.consentStorageKey ?? DEFAULT_CONSENT_STORAGE_KEY;
64
+ this.eventPrefix = this.analyticsConfig.eventPrefix ?? '';
65
+ this.samplingRate = this.analyticsConfig.samplingRate ?? 1.0;
66
+ this.initializeAnalytics();
67
+ }
68
+ // ===========================================================================
69
+ // INICIALIZACIÓN
70
+ // ===========================================================================
71
+ /**
72
+ * Inicializa el servicio de analytics
73
+ */
74
+ initializeAnalytics() {
75
+ if (!isPlatformBrowser(this.platformId)) {
76
+ return;
77
+ }
78
+ // Cargar consent desde localStorage
79
+ this.loadConsentFromStorage();
80
+ // Configurar debug mode
81
+ const debugMode = this.analyticsConfig.debugMode ?? false;
82
+ this._isDebugMode.set(debugMode);
83
+ // Aplicar consent inicial a gtag
84
+ this.applyConsentToGtag(this._consentState().settings);
85
+ // Setear user properties por defecto
86
+ if (this.analyticsConfig.defaultUserProperties) {
87
+ this.setUserProperties(this.analyticsConfig.defaultUserProperties);
88
+ }
89
+ if (debugMode) {
90
+ console.log('[Analytics] Inicializado en modo debug - eventos NO se envían a Firebase');
91
+ }
92
+ }
93
+ /**
94
+ * Obtiene la instancia de Analytics de forma perezosa.
95
+ * Esto evita el error de APP_INITIALIZER de AngularFire.
96
+ */
97
+ getAnalyticsInstance() {
98
+ if (!this.config.enableAnalytics) {
99
+ return null;
100
+ }
101
+ try {
102
+ return this.injector.get(Analytics, null);
103
+ }
104
+ catch {
105
+ return null;
106
+ }
107
+ }
108
+ /**
109
+ * Verifica si Analytics está soportado
110
+ */
111
+ isAnalyticsSupported() {
112
+ if (!isPlatformBrowser(this.platformId)) {
113
+ return false;
114
+ }
115
+ if (!this.config.enableAnalytics) {
116
+ return false;
117
+ }
118
+ if (!this.config.firebase.measurementId) {
119
+ console.warn('[Analytics] measurementId no configurado en firebase config');
120
+ return false;
121
+ }
122
+ return true;
123
+ }
124
+ /**
125
+ * Verifica si debe enviar el evento (sampling)
126
+ */
127
+ shouldSample() {
128
+ if (this.samplingRate >= 1.0) {
129
+ return true;
130
+ }
131
+ return Math.random() < this.samplingRate;
132
+ }
133
+ // ===========================================================================
134
+ // PAGE VIEWS
135
+ // ===========================================================================
136
+ /**
137
+ * Registra un page view.
138
+ * Normalmente se usa automáticamente via AnalyticsRouterTracker.
139
+ *
140
+ * @param pagePath - Ruta de la página (ej: '/products/123')
141
+ * @param pageTitle - Título de la página (opcional)
142
+ */
143
+ logPageView(pagePath, pageTitle) {
144
+ this.logEvent('page_view', {
145
+ page_path: pagePath,
146
+ page_title: pageTitle ?? document?.title,
147
+ page_location: window?.location?.href,
148
+ });
149
+ }
150
+ /**
151
+ * Registra un screen view (para apps tipo SPA).
152
+ *
153
+ * @param screenName - Nombre del screen
154
+ * @param screenClass - Clase del screen (opcional)
155
+ */
156
+ logScreenView(screenName, screenClass) {
157
+ this.logEvent('screen_view', {
158
+ screen_name: screenName,
159
+ screen_class: screenClass,
160
+ });
161
+ }
162
+ // ===========================================================================
163
+ // EVENTOS
164
+ // ===========================================================================
165
+ /**
166
+ * Registra un evento tipado GA4.
167
+ *
168
+ * @param eventName - Nombre del evento (tipado)
169
+ * @param params - Parámetros del evento (tipados según el nombre)
170
+ *
171
+ * @example
172
+ * ```typescript
173
+ * // Evento tipado con autocompletado
174
+ * analytics.logEvent('add_to_cart', {
175
+ * item_id: '123',
176
+ * item_name: 'Producto',
177
+ * value: 99.99,
178
+ * currency: 'EUR'
179
+ * });
180
+ * ```
181
+ */
182
+ logEvent(eventName, params) {
183
+ this.trackEvent(eventName, params);
184
+ }
185
+ /**
186
+ * Registra un evento custom con parámetros libres.
187
+ * Usar cuando el evento no está en el catálogo tipado.
188
+ *
189
+ * @param eventName - Nombre del evento custom
190
+ * @param params - Parámetros libres
191
+ */
192
+ logCustomEvent(eventName, params) {
193
+ const prefixedName = this.eventPrefix + eventName;
194
+ this.trackEvent(prefixedName, params);
195
+ }
196
+ /**
197
+ * Lógica común para enviar eventos
198
+ */
199
+ trackEvent(eventName, params) {
200
+ // Verificar consent
201
+ if (!this._consentState().settings.analytics) {
202
+ this.addToDebugHistory('event', eventName, params, false);
203
+ return;
204
+ }
205
+ // Aplicar sampling
206
+ if (!this.shouldSample()) {
207
+ return;
208
+ }
209
+ // Debug mode: solo log, no enviar
210
+ if (this._isDebugMode()) {
211
+ this.addToDebugHistory('event', eventName, params, false);
212
+ console.log(`[Analytics] Event: ${eventName}`, params);
213
+ return;
214
+ }
215
+ // Enviar a Firebase
216
+ const analytics = this.getAnalyticsInstance();
217
+ if (analytics) {
218
+ try {
219
+ logEvent(analytics, eventName, params);
220
+ this.addToDebugHistory('event', eventName, params, true);
221
+ }
222
+ catch (error) {
223
+ console.error('[Analytics] Error enviando evento:', error);
224
+ }
225
+ }
226
+ }
227
+ // ===========================================================================
228
+ // ECOMMERCE
229
+ // ===========================================================================
230
+ /**
231
+ * Registra vista de item
232
+ */
233
+ logViewItem(item) {
234
+ this.logEvent('view_item', {
235
+ item_id: item.item_id,
236
+ item_name: item.item_name,
237
+ value: item.price,
238
+ currency: item.currency,
239
+ });
240
+ }
241
+ /**
242
+ * Registra agregar al carrito
243
+ */
244
+ logAddToCart(item, quantity = 1) {
245
+ this.logEvent('add_to_cart', {
246
+ item_id: item.item_id,
247
+ item_name: item.item_name,
248
+ value: (item.price ?? 0) * quantity,
249
+ currency: item.currency,
250
+ quantity,
251
+ });
252
+ }
253
+ /**
254
+ * Registra inicio de checkout
255
+ */
256
+ logBeginCheckout(items, value, currency = 'EUR') {
257
+ this.logEvent('begin_checkout', {
258
+ value,
259
+ currency,
260
+ items,
261
+ });
262
+ }
263
+ /**
264
+ * Registra compra completada
265
+ */
266
+ logPurchase(transactionId, items, value, currency = 'EUR') {
267
+ this.logEvent('purchase', {
268
+ transaction_id: transactionId,
269
+ value,
270
+ currency,
271
+ items,
272
+ });
273
+ }
274
+ // ===========================================================================
275
+ // USER PROPERTIES
276
+ // ===========================================================================
277
+ /**
278
+ * Setea el userId para asociar eventos con el usuario.
279
+ * Llamado automáticamente si enableAuthIntegration=true.
280
+ *
281
+ * @param userId - ID del usuario o null para limpiar
282
+ */
283
+ setUserId(userId) {
284
+ if (!this.isAnalyticsSupported()) {
285
+ return;
286
+ }
287
+ if (this._isDebugMode()) {
288
+ console.log(`[Analytics] Set userId: ${userId}`);
289
+ this.addToDebugHistory('user_property', 'user_id', { userId }, false);
290
+ return;
291
+ }
292
+ const analytics = this.getAnalyticsInstance();
293
+ if (analytics) {
294
+ try {
295
+ setUserId(analytics, userId);
296
+ }
297
+ catch (error) {
298
+ console.error('[Analytics] Error seteando userId:', error);
299
+ }
300
+ }
301
+ }
302
+ /**
303
+ * Setea propiedades del usuario para segmentación.
304
+ *
305
+ * @param properties - Propiedades key-value
306
+ *
307
+ * @example
308
+ * ```typescript
309
+ * analytics.setUserProperties({
310
+ * subscription_tier: 'premium',
311
+ * preferred_language: 'es'
312
+ * });
313
+ * ```
314
+ */
315
+ setUserProperties(properties) {
316
+ if (!this.isAnalyticsSupported()) {
317
+ return;
318
+ }
319
+ if (this._isDebugMode()) {
320
+ console.log('[Analytics] Set user properties:', properties);
321
+ this.addToDebugHistory('user_property', 'properties', properties, false);
322
+ return;
323
+ }
324
+ const analytics = this.getAnalyticsInstance();
325
+ if (analytics) {
326
+ try {
327
+ // Convertir a Record<string, string> para Firebase
328
+ const stringProps = {};
329
+ for (const [key, value] of Object.entries(properties)) {
330
+ if (value !== undefined) {
331
+ stringProps[key] = String(value);
332
+ }
333
+ }
334
+ setUserProperties(analytics, stringProps);
335
+ }
336
+ catch (error) {
337
+ console.error('[Analytics] Error seteando user properties:', error);
338
+ }
339
+ }
340
+ }
341
+ /**
342
+ * Setea la organización activa (multi-tenant).
343
+ * Llamado automáticamente si enableAuthIntegration=true.
344
+ *
345
+ * @param orgId - ID de la organización o null
346
+ */
347
+ setActiveOrganization(orgId) {
348
+ if (orgId) {
349
+ this.setUserProperties({ active_organization: orgId });
350
+ }
351
+ }
352
+ // ===========================================================================
353
+ // ERROR TRACKING
354
+ // ===========================================================================
355
+ /**
356
+ * Registra un error para tracking.
357
+ * Integra automáticamente con Angular ErrorHandler si enableErrorTracking=true.
358
+ *
359
+ * @param error - Error o mensaje de error
360
+ * @param context - Contexto adicional
361
+ *
362
+ * @example
363
+ * ```typescript
364
+ * try {
365
+ * await riskyOperation();
366
+ * } catch (error) {
367
+ * analytics.logError(error, { context: 'checkout_flow' });
368
+ * }
369
+ * ```
370
+ */
371
+ logError(error, context) {
372
+ const errorMessage = error instanceof Error ? error.message : error;
373
+ const errorStack = error instanceof Error ? error.stack : undefined;
374
+ const errorType = error instanceof Error ? error.name : 'Error';
375
+ this.logEvent('error_occurred', {
376
+ error_type: errorType,
377
+ error_message: errorMessage.substring(0, 100), // Limitar longitud
378
+ error_stack: errorStack?.substring(0, 500),
379
+ context: context ? JSON.stringify(context) : undefined,
380
+ });
381
+ }
382
+ /**
383
+ * Registra un error fatal (crash-level).
384
+ */
385
+ logFatalError(error, context) {
386
+ this.logError(error, { ...context, severity: 'fatal' });
387
+ }
388
+ // ===========================================================================
389
+ // CONSENT MODE (GDPR)
390
+ // ===========================================================================
391
+ /**
392
+ * Actualiza el estado de consentimiento del usuario.
393
+ * Afecta qué datos se recolectan y envían.
394
+ *
395
+ * @param consent - Settings de consentimiento
396
+ *
397
+ * @example
398
+ * ```typescript
399
+ * // Usuario acepta todo
400
+ * analytics.updateConsent({ analytics: true, advertising: true });
401
+ *
402
+ * // Usuario rechaza publicidad
403
+ * analytics.updateConsent({ analytics: true, advertising: false });
404
+ * ```
405
+ */
406
+ updateConsent(consent) {
407
+ const newState = {
408
+ settings: { ...this._consentState().settings, ...consent },
409
+ updatedAt: new Date(),
410
+ hasDecided: true,
411
+ };
412
+ this._consentState.set(newState);
413
+ this.saveConsentToStorage(newState);
414
+ this.applyConsentToGtag(newState.settings);
415
+ this.addToDebugHistory('consent', 'update', consent, false);
416
+ }
417
+ /**
418
+ * Deniega todo consentimiento.
419
+ */
420
+ denyAllConsent() {
421
+ this.updateConsent({
422
+ analytics: false,
423
+ advertising: false,
424
+ functionality: false,
425
+ security: true, // Siempre permitido
426
+ });
427
+ }
428
+ /**
429
+ * Acepta todo consentimiento.
430
+ */
431
+ grantAllConsent() {
432
+ this.updateConsent({
433
+ analytics: true,
434
+ advertising: true,
435
+ functionality: true,
436
+ security: true,
437
+ });
438
+ }
439
+ /**
440
+ * Obtiene el estado actual de consentimiento.
441
+ */
442
+ getConsentState() {
443
+ return this._consentState();
444
+ }
445
+ /**
446
+ * Aplica consent settings a gtag (GA4 Consent Mode v2)
447
+ */
448
+ applyConsentToGtag(settings) {
449
+ if (!isPlatformBrowser(this.platformId)) {
450
+ return;
451
+ }
452
+ const gtag = window.gtag;
453
+ if (typeof gtag !== 'function') {
454
+ return;
455
+ }
456
+ try {
457
+ gtag('consent', 'update', {
458
+ analytics_storage: settings.analytics ? 'granted' : 'denied',
459
+ ad_storage: settings.advertising ? 'granted' : 'denied',
460
+ ad_user_data: settings.advertising ? 'granted' : 'denied',
461
+ ad_personalization: settings.advertising ? 'granted' : 'denied',
462
+ functionality_storage: settings.functionality ? 'granted' : 'denied',
463
+ security_storage: settings.security !== false ? 'granted' : 'denied',
464
+ });
465
+ }
466
+ catch (error) {
467
+ console.warn('[Analytics] Error aplicando consent a gtag:', error);
468
+ }
469
+ }
470
+ /**
471
+ * Carga consent desde localStorage
472
+ */
473
+ loadConsentFromStorage() {
474
+ if (!isPlatformBrowser(this.platformId)) {
475
+ return;
476
+ }
477
+ try {
478
+ const stored = localStorage.getItem(this.consentStorageKey);
479
+ if (stored) {
480
+ const parsed = JSON.parse(stored);
481
+ this._consentState.set({
482
+ ...parsed,
483
+ updatedAt: parsed.updatedAt ? new Date(parsed.updatedAt) : null,
484
+ });
485
+ }
486
+ else if (this.analyticsConfig.defaultConsent) {
487
+ // Aplicar consent por defecto si no hay guardado
488
+ this._consentState.set({
489
+ settings: this.analyticsConfig.defaultConsent,
490
+ updatedAt: null,
491
+ hasDecided: false,
492
+ });
493
+ }
494
+ }
495
+ catch (error) {
496
+ console.warn('[Analytics] Error cargando consent:', error);
497
+ }
498
+ }
499
+ /**
500
+ * Guarda consent en localStorage
501
+ */
502
+ saveConsentToStorage(state) {
503
+ if (!isPlatformBrowser(this.platformId)) {
504
+ return;
505
+ }
506
+ try {
507
+ localStorage.setItem(this.consentStorageKey, JSON.stringify(state));
508
+ }
509
+ catch (error) {
510
+ console.warn('[Analytics] Error guardando consent:', error);
511
+ }
512
+ }
513
+ // ===========================================================================
514
+ // TIMING / PERFORMANCE
515
+ // ===========================================================================
516
+ /**
517
+ * Registra una métrica de timing/performance.
518
+ *
519
+ * @param name - Nombre de la métrica
520
+ * @param valueMs - Valor en milisegundos
521
+ * @param params - Parámetros adicionales
522
+ *
523
+ * @example
524
+ * ```typescript
525
+ * const start = performance.now();
526
+ * await loadData();
527
+ * analytics.logTiming('data_load', performance.now() - start, {
528
+ * category: 'api',
529
+ * endpoint: '/products'
530
+ * });
531
+ * ```
532
+ */
533
+ logTiming(name, valueMs, params) {
534
+ this.logEvent('performance_metric', {
535
+ metric_name: name,
536
+ value: Math.round(valueMs),
537
+ unit: 'ms',
538
+ ...params,
539
+ });
540
+ }
541
+ // ===========================================================================
542
+ // DEBUG
543
+ // ===========================================================================
544
+ /**
545
+ * Habilita/deshabilita modo debug.
546
+ * En debug: logea a consola, no envía a Firebase.
547
+ */
548
+ setDebugMode(enabled) {
549
+ this._isDebugMode.set(enabled);
550
+ console.log(`[Analytics] Debug mode: ${enabled ? 'ON' : 'OFF'}`);
551
+ }
552
+ /**
553
+ * Obtiene historial de eventos (solo en debug mode).
554
+ * Útil para testing y desarrollo.
555
+ */
556
+ getDebugHistory() {
557
+ return this._debugHistory();
558
+ }
559
+ /**
560
+ * Limpia historial de debug.
561
+ */
562
+ clearDebugHistory() {
563
+ this._debugHistory.set([]);
564
+ }
565
+ /**
566
+ * Agrega evento al historial de debug
567
+ */
568
+ addToDebugHistory(type, name, params, sent = false) {
569
+ if (!this._isDebugMode()) {
570
+ return;
571
+ }
572
+ const event = {
573
+ timestamp: new Date(),
574
+ type,
575
+ name,
576
+ params,
577
+ sent,
578
+ };
579
+ this._debugHistory.update((history) => {
580
+ const newHistory = [event, ...history];
581
+ return newHistory.slice(0, MAX_DEBUG_HISTORY);
582
+ });
583
+ }
584
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AnalyticsService, deps: [{ token: i0.Injector }, { token: VALTECH_FIREBASE_CONFIG }, { token: PLATFORM_ID }], target: i0.ɵɵFactoryTarget.Injectable }); }
585
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AnalyticsService, providedIn: 'root' }); }
586
+ }
587
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AnalyticsService, decorators: [{
588
+ type: Injectable,
589
+ args: [{ providedIn: 'root' }]
590
+ }], ctorParameters: () => [{ type: i0.Injector }, { type: undefined, decorators: [{
591
+ type: Inject,
592
+ args: [VALTECH_FIREBASE_CONFIG]
593
+ }] }, { type: Object, decorators: [{
594
+ type: Inject,
595
+ args: [PLATFORM_ID]
596
+ }] }] });
597
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"analytics.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/analytics.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Analytics Service (Firebase GA4)\n *\n * Servicio para tracking de eventos, page views y errores con Firebase Analytics.\n * Integra con el sistema de auth para user properties y respeta consent mode GDPR.\n */\n\nimport { Inject, Injectable, Injector, PLATFORM_ID, signal, computed } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { Analytics, logEvent, setUserId, setUserProperties } from '@angular/fire/analytics';\n\nimport { VALTECH_FIREBASE_CONFIG } from './config';\nimport { ValtechFirebaseConfig } from './types';\nimport {\n  AnalyticsConfig,\n  AnalyticsDebugEvent,\n  AnalyticsEventName,\n  AnalyticsEventParams,\n  AnalyticsItem,\n  ConsentSettings,\n  ConsentState,\n  DebugEventType,\n  TimingMetric,\n  UserProperties,\n} from './analytics-types';\n\n/** Key por defecto para persistir consent en localStorage */\nconst DEFAULT_CONSENT_STORAGE_KEY = 'analytics_consent';\n\n/** Máximo de eventos en historial de debug */\nconst MAX_DEBUG_HISTORY = 100;\n\n/**\n * Servicio de Firebase Analytics (GA4).\n *\n * Proporciona tracking de eventos, page views, errores y métricas de performance.\n * Soporta GDPR Consent Mode y modo debug para desarrollo.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class ProductComponent {\n *   private analytics = inject(AnalyticsService);\n *\n *   onPurchase(product: Product) {\n *     this.analytics.logEvent('purchase', {\n *       transaction_id: order.id,\n *       value: order.total,\n *       currency: 'EUR'\n *     });\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AnalyticsService {\n  // ===========================================================================\n  // ESTADO (Signals)\n  // ===========================================================================\n\n  private readonly _consentState = signal<ConsentState>({\n    settings: {},\n    updatedAt: null,\n    hasDecided: false,\n  });\n\n  private readonly _isDebugMode = signal<boolean>(false);\n  private readonly _debugHistory = signal<AnalyticsDebugEvent[]>([]);\n\n  /** Estado de consentimiento actual (readonly) */\n  readonly consentState = this._consentState.asReadonly();\n\n  /** Indica si está en modo debug */\n  readonly isDebugMode = this._isDebugMode.asReadonly();\n\n  /** Indica si analytics está habilitado y funcionando */\n  readonly isEnabled = computed(() => {\n    return this.isAnalyticsSupported() && this._consentState().settings.analytics === true;\n  });\n\n  // ===========================================================================\n  // CONFIGURACIÓN\n  // ===========================================================================\n\n  private readonly analyticsConfig: AnalyticsConfig;\n  private readonly consentStorageKey: string;\n  private readonly eventPrefix: string;\n  private readonly samplingRate: number;\n\n  constructor(\n    private injector: Injector,\n    @Inject(VALTECH_FIREBASE_CONFIG) private config: ValtechFirebaseConfig,\n    @Inject(PLATFORM_ID) private platformId: Object\n  ) {\n    this.analyticsConfig = config.analyticsConfig ?? {};\n    this.consentStorageKey =\n      this.analyticsConfig.consentStorageKey ?? DEFAULT_CONSENT_STORAGE_KEY;\n    this.eventPrefix = this.analyticsConfig.eventPrefix ?? '';\n    this.samplingRate = this.analyticsConfig.samplingRate ?? 1.0;\n\n    this.initializeAnalytics();\n  }\n\n  // ===========================================================================\n  // INICIALIZACIÓN\n  // ===========================================================================\n\n  /**\n   * Inicializa el servicio de analytics\n   */\n  private initializeAnalytics(): void {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    // Cargar consent desde localStorage\n    this.loadConsentFromStorage();\n\n    // Configurar debug mode\n    const debugMode = this.analyticsConfig.debugMode ?? false;\n    this._isDebugMode.set(debugMode);\n\n    // Aplicar consent inicial a gtag\n    this.applyConsentToGtag(this._consentState().settings);\n\n    // Setear user properties por defecto\n    if (this.analyticsConfig.defaultUserProperties) {\n      this.setUserProperties(this.analyticsConfig.defaultUserProperties);\n    }\n\n    if (debugMode) {\n      console.log('[Analytics] Inicializado en modo debug - eventos NO se envían a Firebase');\n    }\n  }\n\n  /**\n   * Obtiene la instancia de Analytics de forma perezosa.\n   * Esto evita el error de APP_INITIALIZER de AngularFire.\n   */\n  private getAnalyticsInstance(): Analytics | null {\n    if (!this.config.enableAnalytics) {\n      return null;\n    }\n\n    try {\n      return this.injector.get(Analytics, null);\n    } catch {\n      return null;\n    }\n  }\n\n  /**\n   * Verifica si Analytics está soportado\n   */\n  private isAnalyticsSupported(): boolean {\n    if (!isPlatformBrowser(this.platformId)) {\n      return false;\n    }\n\n    if (!this.config.enableAnalytics) {\n      return false;\n    }\n\n    if (!this.config.firebase.measurementId) {\n      console.warn('[Analytics] measurementId no configurado en firebase config');\n      return false;\n    }\n\n    return true;\n  }\n\n  /**\n   * Verifica si debe enviar el evento (sampling)\n   */\n  private shouldSample(): boolean {\n    if (this.samplingRate >= 1.0) {\n      return true;\n    }\n    return Math.random() < this.samplingRate;\n  }\n\n  // ===========================================================================\n  // PAGE VIEWS\n  // ===========================================================================\n\n  /**\n   * Registra un page view.\n   * Normalmente se usa automáticamente via AnalyticsRouterTracker.\n   *\n   * @param pagePath - Ruta de la página (ej: '/products/123')\n   * @param pageTitle - Título de la página (opcional)\n   */\n  logPageView(pagePath: string, pageTitle?: string): void {\n    this.logEvent('page_view', {\n      page_path: pagePath,\n      page_title: pageTitle ?? document?.title,\n      page_location: window?.location?.href,\n    });\n  }\n\n  /**\n   * Registra un screen view (para apps tipo SPA).\n   *\n   * @param screenName - Nombre del screen\n   * @param screenClass - Clase del screen (opcional)\n   */\n  logScreenView(screenName: string, screenClass?: string): void {\n    this.logEvent('screen_view', {\n      screen_name: screenName,\n      screen_class: screenClass,\n    });\n  }\n\n  // ===========================================================================\n  // EVENTOS\n  // ===========================================================================\n\n  /**\n   * Registra un evento tipado GA4.\n   *\n   * @param eventName - Nombre del evento (tipado)\n   * @param params - Parámetros del evento (tipados según el nombre)\n   *\n   * @example\n   * ```typescript\n   * // Evento tipado con autocompletado\n   * analytics.logEvent('add_to_cart', {\n   *   item_id: '123',\n   *   item_name: 'Producto',\n   *   value: 99.99,\n   *   currency: 'EUR'\n   * });\n   * ```\n   */\n  logEvent<T extends AnalyticsEventName>(\n    eventName: T,\n    params?: AnalyticsEventParams[T]\n  ): void {\n    this.trackEvent(eventName, params as Record<string, unknown>);\n  }\n\n  /**\n   * Registra un evento custom con parámetros libres.\n   * Usar cuando el evento no está en el catálogo tipado.\n   *\n   * @param eventName - Nombre del evento custom\n   * @param params - Parámetros libres\n   */\n  logCustomEvent(eventName: string, params?: Record<string, unknown>): void {\n    const prefixedName = this.eventPrefix + eventName;\n    this.trackEvent(prefixedName, params);\n  }\n\n  /**\n   * Lógica común para enviar eventos\n   */\n  private trackEvent(eventName: string, params?: Record<string, unknown>): void {\n    // Verificar consent\n    if (!this._consentState().settings.analytics) {\n      this.addToDebugHistory('event', eventName, params, false);\n      return;\n    }\n\n    // Aplicar sampling\n    if (!this.shouldSample()) {\n      return;\n    }\n\n    // Debug mode: solo log, no enviar\n    if (this._isDebugMode()) {\n      this.addToDebugHistory('event', eventName, params, false);\n      console.log(`[Analytics] Event: ${eventName}`, params);\n      return;\n    }\n\n    // Enviar a Firebase\n    const analytics = this.getAnalyticsInstance();\n    if (analytics) {\n      try {\n        logEvent(analytics, eventName as string, params);\n        this.addToDebugHistory('event', eventName, params, true);\n      } catch (error) {\n        console.error('[Analytics] Error enviando evento:', error);\n      }\n    }\n  }\n\n  // ===========================================================================\n  // ECOMMERCE\n  // ===========================================================================\n\n  /**\n   * Registra vista de item\n   */\n  logViewItem(item: AnalyticsItem): void {\n    this.logEvent('view_item', {\n      item_id: item.item_id,\n      item_name: item.item_name,\n      value: item.price,\n      currency: item.currency,\n    });\n  }\n\n  /**\n   * Registra agregar al carrito\n   */\n  logAddToCart(item: AnalyticsItem, quantity = 1): void {\n    this.logEvent('add_to_cart', {\n      item_id: item.item_id,\n      item_name: item.item_name,\n      value: (item.price ?? 0) * quantity,\n      currency: item.currency,\n      quantity,\n    });\n  }\n\n  /**\n   * Registra inicio de checkout\n   */\n  logBeginCheckout(items: AnalyticsItem[], value: number, currency = 'EUR'): void {\n    this.logEvent('begin_checkout', {\n      value,\n      currency,\n      items,\n    });\n  }\n\n  /**\n   * Registra compra completada\n   */\n  logPurchase(\n    transactionId: string,\n    items: AnalyticsItem[],\n    value: number,\n    currency = 'EUR'\n  ): void {\n    this.logEvent('purchase', {\n      transaction_id: transactionId,\n      value,\n      currency,\n      items,\n    });\n  }\n\n  // ===========================================================================\n  // USER PROPERTIES\n  // ===========================================================================\n\n  /**\n   * Setea el userId para asociar eventos con el usuario.\n   * Llamado automáticamente si enableAuthIntegration=true.\n   *\n   * @param userId - ID del usuario o null para limpiar\n   */\n  setUserId(userId: string | null): void {\n    if (!this.isAnalyticsSupported()) {\n      return;\n    }\n\n    if (this._isDebugMode()) {\n      console.log(`[Analytics] Set userId: ${userId}`);\n      this.addToDebugHistory('user_property', 'user_id', { userId }, false);\n      return;\n    }\n\n    const analytics = this.getAnalyticsInstance();\n    if (analytics) {\n      try {\n        setUserId(analytics, userId);\n      } catch (error) {\n        console.error('[Analytics] Error seteando userId:', error);\n      }\n    }\n  }\n\n  /**\n   * Setea propiedades del usuario para segmentación.\n   *\n   * @param properties - Propiedades key-value\n   *\n   * @example\n   * ```typescript\n   * analytics.setUserProperties({\n   *   subscription_tier: 'premium',\n   *   preferred_language: 'es'\n   * });\n   * ```\n   */\n  setUserProperties(properties: UserProperties | Record<string, string | number | boolean>): void {\n    if (!this.isAnalyticsSupported()) {\n      return;\n    }\n\n    if (this._isDebugMode()) {\n      console.log('[Analytics] Set user properties:', properties);\n      this.addToDebugHistory('user_property', 'properties', properties, false);\n      return;\n    }\n\n    const analytics = this.getAnalyticsInstance();\n    if (analytics) {\n      try {\n        // Convertir a Record<string, string> para Firebase\n        const stringProps: Record<string, string> = {};\n        for (const [key, value] of Object.entries(properties)) {\n          if (value !== undefined) {\n            stringProps[key] = String(value);\n          }\n        }\n        setUserProperties(analytics, stringProps);\n      } catch (error) {\n        console.error('[Analytics] Error seteando user properties:', error);\n      }\n    }\n  }\n\n  /**\n   * Setea la organización activa (multi-tenant).\n   * Llamado automáticamente si enableAuthIntegration=true.\n   *\n   * @param orgId - ID de la organización o null\n   */\n  setActiveOrganization(orgId: string | null): void {\n    if (orgId) {\n      this.setUserProperties({ active_organization: orgId });\n    }\n  }\n\n  // ===========================================================================\n  // ERROR TRACKING\n  // ===========================================================================\n\n  /**\n   * Registra un error para tracking.\n   * Integra automáticamente con Angular ErrorHandler si enableErrorTracking=true.\n   *\n   * @param error - Error o mensaje de error\n   * @param context - Contexto adicional\n   *\n   * @example\n   * ```typescript\n   * try {\n   *   await riskyOperation();\n   * } catch (error) {\n   *   analytics.logError(error, { context: 'checkout_flow' });\n   * }\n   * ```\n   */\n  logError(error: Error | string, context?: Record<string, string>): void {\n    const errorMessage = error instanceof Error ? error.message : error;\n    const errorStack = error instanceof Error ? error.stack : undefined;\n    const errorType = error instanceof Error ? error.name : 'Error';\n\n    this.logEvent('error_occurred', {\n      error_type: errorType,\n      error_message: errorMessage.substring(0, 100), // Limitar longitud\n      error_stack: errorStack?.substring(0, 500),\n      context: context ? JSON.stringify(context) : undefined,\n    });\n  }\n\n  /**\n   * Registra un error fatal (crash-level).\n   */\n  logFatalError(error: Error, context?: Record<string, string>): void {\n    this.logError(error, { ...context, severity: 'fatal' });\n  }\n\n  // ===========================================================================\n  // CONSENT MODE (GDPR)\n  // ===========================================================================\n\n  /**\n   * Actualiza el estado de consentimiento del usuario.\n   * Afecta qué datos se recolectan y envían.\n   *\n   * @param consent - Settings de consentimiento\n   *\n   * @example\n   * ```typescript\n   * // Usuario acepta todo\n   * analytics.updateConsent({ analytics: true, advertising: true });\n   *\n   * // Usuario rechaza publicidad\n   * analytics.updateConsent({ analytics: true, advertising: false });\n   * ```\n   */\n  updateConsent(consent: ConsentSettings): void {\n    const newState: ConsentState = {\n      settings: { ...this._consentState().settings, ...consent },\n      updatedAt: new Date(),\n      hasDecided: true,\n    };\n\n    this._consentState.set(newState);\n    this.saveConsentToStorage(newState);\n    this.applyConsentToGtag(newState.settings);\n\n    this.addToDebugHistory('consent', 'update', consent as unknown as Record<string, unknown>, false);\n  }\n\n  /**\n   * Deniega todo consentimiento.\n   */\n  denyAllConsent(): void {\n    this.updateConsent({\n      analytics: false,\n      advertising: false,\n      functionality: false,\n      security: true, // Siempre permitido\n    });\n  }\n\n  /**\n   * Acepta todo consentimiento.\n   */\n  grantAllConsent(): void {\n    this.updateConsent({\n      analytics: true,\n      advertising: true,\n      functionality: true,\n      security: true,\n    });\n  }\n\n  /**\n   * Obtiene el estado actual de consentimiento.\n   */\n  getConsentState(): ConsentState {\n    return this._consentState();\n  }\n\n  /**\n   * Aplica consent settings a gtag (GA4 Consent Mode v2)\n   */\n  private applyConsentToGtag(settings: ConsentSettings): void {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    const gtag = (window as any).gtag;\n    if (typeof gtag !== 'function') {\n      return;\n    }\n\n    try {\n      gtag('consent', 'update', {\n        analytics_storage: settings.analytics ? 'granted' : 'denied',\n        ad_storage: settings.advertising ? 'granted' : 'denied',\n        ad_user_data: settings.advertising ? 'granted' : 'denied',\n        ad_personalization: settings.advertising ? 'granted' : 'denied',\n        functionality_storage: settings.functionality ? 'granted' : 'denied',\n        security_storage: settings.security !== false ? 'granted' : 'denied',\n      });\n    } catch (error) {\n      console.warn('[Analytics] Error aplicando consent a gtag:', error);\n    }\n  }\n\n  /**\n   * Carga consent desde localStorage\n   */\n  private loadConsentFromStorage(): void {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    try {\n      const stored = localStorage.getItem(this.consentStorageKey);\n      if (stored) {\n        const parsed = JSON.parse(stored) as ConsentState;\n        this._consentState.set({\n          ...parsed,\n          updatedAt: parsed.updatedAt ? new Date(parsed.updatedAt) : null,\n        });\n      } else if (this.analyticsConfig.defaultConsent) {\n        // Aplicar consent por defecto si no hay guardado\n        this._consentState.set({\n          settings: this.analyticsConfig.defaultConsent,\n          updatedAt: null,\n          hasDecided: false,\n        });\n      }\n    } catch (error) {\n      console.warn('[Analytics] Error cargando consent:', error);\n    }\n  }\n\n  /**\n   * Guarda consent en localStorage\n   */\n  private saveConsentToStorage(state: ConsentState): void {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    try {\n      localStorage.setItem(this.consentStorageKey, JSON.stringify(state));\n    } catch (error) {\n      console.warn('[Analytics] Error guardando consent:', error);\n    }\n  }\n\n  // ===========================================================================\n  // TIMING / PERFORMANCE\n  // ===========================================================================\n\n  /**\n   * Registra una métrica de timing/performance.\n   *\n   * @param name - Nombre de la métrica\n   * @param valueMs - Valor en milisegundos\n   * @param params - Parámetros adicionales\n   *\n   * @example\n   * ```typescript\n   * const start = performance.now();\n   * await loadData();\n   * analytics.logTiming('data_load', performance.now() - start, {\n   *   category: 'api',\n   *   endpoint: '/products'\n   * });\n   * ```\n   */\n  logTiming(name: string, valueMs: number, params?: Record<string, string>): void {\n    this.logEvent('performance_metric', {\n      metric_name: name,\n      value: Math.round(valueMs),\n      unit: 'ms',\n      ...params,\n    });\n  }\n\n  // ===========================================================================\n  // DEBUG\n  // ===========================================================================\n\n  /**\n   * Habilita/deshabilita modo debug.\n   * En debug: logea a consola, no envía a Firebase.\n   */\n  setDebugMode(enabled: boolean): void {\n    this._isDebugMode.set(enabled);\n    console.log(`[Analytics] Debug mode: ${enabled ? 'ON' : 'OFF'}`);\n  }\n\n  /**\n   * Obtiene historial de eventos (solo en debug mode).\n   * Útil para testing y desarrollo.\n   */\n  getDebugHistory(): AnalyticsDebugEvent[] {\n    return this._debugHistory();\n  }\n\n  /**\n   * Limpia historial de debug.\n   */\n  clearDebugHistory(): void {\n    this._debugHistory.set([]);\n  }\n\n  /**\n   * Agrega evento al historial de debug\n   */\n  private addToDebugHistory(\n    type: DebugEventType,\n    name: string,\n    params?: Record<string, unknown>,\n    sent = false\n  ): void {\n    if (!this._isDebugMode()) {\n      return;\n    }\n\n    const event: AnalyticsDebugEvent = {\n      timestamp: new Date(),\n      type,\n      name,\n      params,\n      sent,\n    };\n\n    this._debugHistory.update((history) => {\n      const newHistory = [event, ...history];\n      return newHistory.slice(0, MAX_DEBUG_HISTORY);\n    });\n  }\n}\n"]}