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.
- package/esm2022/lib/components/atoms/button/button.component.mjs +87 -48
- package/esm2022/lib/components/molecules/action-header/action-header.component.mjs +1 -1
- package/esm2022/lib/components/molecules/ad-slot/ad-slot.component.mjs +249 -0
- package/esm2022/lib/components/molecules/button-group/button-group.component.mjs +1 -1
- package/esm2022/lib/components/molecules/card/card.component.mjs +2 -2
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +1 -1
- package/esm2022/lib/components/molecules/raffle-status-card/raffle-status-card.component.mjs +2 -2
- package/esm2022/lib/components/organisms/article/article.component.mjs +2 -2
- package/esm2022/lib/components/organisms/menu/menu.component.mjs +1 -1
- package/esm2022/lib/components/templates/page-template/page-template.component.mjs +1 -1
- package/esm2022/lib/services/ads/ads-consent.service.mjs +152 -0
- package/esm2022/lib/services/ads/ads-loader.service.mjs +160 -0
- package/esm2022/lib/services/ads/ads.service.mjs +449 -0
- package/esm2022/lib/services/ads/config.mjs +118 -0
- package/esm2022/lib/services/ads/index.mjs +14 -0
- package/esm2022/lib/services/ads/types.mjs +23 -0
- package/esm2022/lib/services/auth/auth.service.mjs +103 -6
- package/esm2022/lib/services/auth/index.mjs +4 -1
- package/esm2022/lib/services/auth/oauth-callback.component.mjs +141 -0
- package/esm2022/lib/services/auth/oauth.service.mjs +250 -0
- package/esm2022/lib/services/auth/types.mjs +1 -1
- package/esm2022/lib/services/firebase/analytics-error-handler.mjs +141 -0
- package/esm2022/lib/services/firebase/analytics-router-tracker.mjs +99 -0
- package/esm2022/lib/services/firebase/analytics.service.mjs +597 -0
- package/esm2022/lib/services/firebase/config.mjs +21 -2
- package/esm2022/lib/services/firebase/index.mjs +6 -1
- package/esm2022/public-api.mjs +6 -1
- package/fesm2022/valtech-components.mjs +2739 -239
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/button/button.component.d.ts +30 -6
- package/lib/components/molecules/ad-slot/ad-slot.component.d.ts +78 -0
- package/lib/components/organisms/article/article.component.d.ts +3 -3
- package/lib/services/ads/ads-consent.service.d.ts +59 -0
- package/lib/services/ads/ads-loader.service.d.ts +46 -0
- package/lib/services/ads/ads.service.d.ts +123 -0
- package/lib/services/ads/config.d.ts +69 -0
- package/lib/services/ads/index.d.ts +10 -0
- package/lib/services/ads/types.d.ts +163 -0
- package/lib/services/auth/auth.service.d.ts +56 -3
- package/lib/services/auth/index.d.ts +2 -0
- package/lib/services/auth/oauth-callback.component.d.ts +34 -0
- package/lib/services/auth/oauth.service.d.ts +90 -0
- package/lib/services/auth/types.d.ts +69 -0
- package/lib/services/firebase/analytics-error-handler.d.ts +54 -0
- package/lib/services/firebase/analytics-router-tracker.d.ts +51 -0
- package/lib/services/firebase/analytics.service.d.ts +256 -0
- package/lib/services/firebase/index.d.ts +4 -0
- package/package.json +1 -1
- 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"]}
|