valtech-components 2.0.480 → 2.0.481
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.
|
@@ -326,6 +326,7 @@ export class MessagingService {
|
|
|
326
326
|
}
|
|
327
327
|
/**
|
|
328
328
|
* Verifica si FCM está soportado en el navegador actual.
|
|
329
|
+
* Usa lógica inteligente para evitar falsos negativos por timing del Injector.
|
|
329
330
|
*
|
|
330
331
|
* @returns true si FCM está soportado
|
|
331
332
|
*
|
|
@@ -339,6 +340,23 @@ export class MessagingService {
|
|
|
339
340
|
* ```
|
|
340
341
|
*/
|
|
341
342
|
async isSupported() {
|
|
343
|
+
// Si ya tenemos token en estado, claramente FCM funciona
|
|
344
|
+
if (this.stateSubject.value.token) {
|
|
345
|
+
return true;
|
|
346
|
+
}
|
|
347
|
+
// Si ya calculamos soporte exitosamente, usarlo
|
|
348
|
+
if (this.stateSubject.value.isSupported) {
|
|
349
|
+
return true;
|
|
350
|
+
}
|
|
351
|
+
// Verificaciones básicas que no dependen del Injector
|
|
352
|
+
if (!isPlatformBrowser(this.platformId)) {
|
|
353
|
+
return false;
|
|
354
|
+
}
|
|
355
|
+
if (!('Notification' in window) || !('serviceWorker' in navigator)) {
|
|
356
|
+
return false;
|
|
357
|
+
}
|
|
358
|
+
// Si el navegador soporta las APIs pero el Injector no está listo,
|
|
359
|
+
// podríamos tener un falso negativo. Intentar checkSupport de todas formas.
|
|
342
360
|
return this.checkSupport();
|
|
343
361
|
}
|
|
344
362
|
/**
|
|
@@ -361,6 +379,29 @@ export class MessagingService {
|
|
|
361
379
|
get hasPermission() {
|
|
362
380
|
return this.stateSubject.value.permission === 'granted';
|
|
363
381
|
}
|
|
382
|
+
/**
|
|
383
|
+
* Hidrata el estado del token desde un valor externo (ej: localStorage).
|
|
384
|
+
* Útil cuando el token ya existe pero el MessagingService no lo tiene en memoria.
|
|
385
|
+
*
|
|
386
|
+
* @param token Token FCM a setear
|
|
387
|
+
*
|
|
388
|
+
* @example
|
|
389
|
+
* ```typescript
|
|
390
|
+
* const storedToken = localStorage.getItem('fcm_token');
|
|
391
|
+
* if (storedToken) {
|
|
392
|
+
* messaging.setTokenFromStorage(storedToken);
|
|
393
|
+
* }
|
|
394
|
+
* ```
|
|
395
|
+
*/
|
|
396
|
+
setTokenFromStorage(token) {
|
|
397
|
+
if (!token)
|
|
398
|
+
return;
|
|
399
|
+
this.stateSubject.next({
|
|
400
|
+
...this.stateSubject.value,
|
|
401
|
+
token,
|
|
402
|
+
isSupported: true, // Si hay token, claramente FCM funciona
|
|
403
|
+
});
|
|
404
|
+
}
|
|
364
405
|
// ===========================================================================
|
|
365
406
|
// DEEP LINKING / NAVEGACIÓN
|
|
366
407
|
// ===========================================================================
|
|
@@ -529,4 +570,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
529
570
|
type: Inject,
|
|
530
571
|
args: [PLATFORM_ID]
|
|
531
572
|
}] }, { type: i0.NgZone }] });
|
|
532
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"messaging.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/messaging.service.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAE,MAAM,EAAE,UAAU,EAAoB,WAAW,EAAE,MAAM,eAAe,CAAC;AAClF,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AACpD,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,yBAAyB,CAAC;AACtF,OAAO,EAAc,OAAO,EAAE,eAAe,EAAE,MAAM,MAAM,CAAC;AAE5D,OAAO,EAAE,uBAAuB,EAAE,MAAM,UAAU,CAAC;;AAkBnD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BG;AAEH,MAAM,OAAO,gBAAgB;IAW3B,YACU,QAAkB,EACe,MAA6B,EACzC,UAAkB,EACvC,MAAc;QAHd,aAAQ,GAAR,QAAQ,CAAU;QACe,WAAM,GAAN,MAAM,CAAuB;QACzC,eAAU,GAAV,UAAU,CAAQ;QACvC,WAAM,GAAN,MAAM,CAAQ;QAdhB,mBAAc,GAAG,IAAI,OAAO,EAAuB,CAAC;QACpD,6BAAwB,GAAG,IAAI,OAAO,EAA0B,CAAC;QACjE,iBAAY,GAAG,IAAI,eAAe,CAAiB;YACzD,KAAK,EAAE,IAAI;YACX,UAAU,EAAE,SAAS;YACrB,WAAW,EAAE,KAAK;SACnB,CAAC,CAAC;QAUD,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAED;;;OAGG;IACK,oBAAoB;QAC1B,IAAI,CAAC;YACH,OAAO,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QAC5C,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED,8EAA8E;IAC9E,iBAAiB;IACjB,8EAA8E;IAE9E;;OAEG;IACK,KAAK,CAAC,mBAAmB;QAC/B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACxC,OAAO;QACT,CAAC;QAED,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QAC5C,MAAM,UAAU,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAE7C,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;YACrB,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK;YAC1B,WAAW,EAAE,SAAS;YACtB,UAAU;SACX,CAAC,CAAC;QAEH,4CAA4C;QAC5C,IAAI,SAAS,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;YAC1C,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC9B,CAAC;QAED,6EAA6E;QAC7E,IAAI,CAAC,0BAA0B,EAAE,CAAC;IACpC,CAAC;IAED;;;OAGG;IACK,0BAA0B;QAChC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,eAAe,IAAI,SAAS,CAAC,EAAE,CAAC;YAC3E,OAAO;QACT,CAAC;QAED,SAAS,CAAC,aAAa,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC,KAAK,EAAE,EAAE;YAC5D,oDAAoD;YACpD,IAAI,KAAK,CAAC,IAAI,EAAE,IAAI,KAAK,oBAAoB,EAAE,CAAC;gBAC9C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;oBACnB,MAAM,YAAY,GAAG,KAAK,CAAC,IAAI,CAAC,YAAmC,CAAC;oBACpE,MAAM,MAAM,GAAG,IAAI,CAAC,qBAAqB,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;oBAE7D,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC;wBACjC,YAAY;wBACZ,MAAM;wBACN,SAAS,EAAE,IAAI,IAAI,EAAE;qBACtB,CAAC,CAAC;gBACL,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,YAAY;QACxB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACxC,OAAO,KAAK,CAAC;QACf,CAAC;QAED,4BAA4B;QAC5B,IAAI,CAAC,CAAC,cAAc,IAAI,MAAM,CAAC,EAAE,CAAC;YAChC,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,CAAC,CAAC,eAAe,IAAI,SAAS,CAAC,EAAE,CAAC;YACpC,OAAO,KAAK,CAAC;QACf,CAAC;QAED,0CAA0C;QAC1C,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE,EAAE,CAAC;YACjC,OAAO,KAAK,CAAC;QACf,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED,8EAA8E;IAC9E,mBAAmB;IACnB,8EAA8E;IAE9E;;;;;;;;;;;;;;;OAeG;IACH,KAAK,CAAC,iBAAiB;QACrB,IAAI,CAAC,MAAM,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;YAC9B,OAAO,CAAC,IAAI,CAAC,yCAAyC,CAAC,CAAC;YACxD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,CAAC;YACH,sCAAsC;YACtC,MAAM,UAAU,GAAG,MAAM,YAAY,CAAC,iBAAiB,EAAE,CAAC;YAE1D,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;gBACrB,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK;gBAC1B,UAAU,EAAE,UAAoC;aACjD,CAAC,CAAC;YAEH,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;gBAC7B,OAAO,CAAC,IAAI,CAAC,oCAAoC,CAAC,CAAC;gBACnD,OAAO,IAAI,CAAC;YACd,CAAC;YAED,oBAAoB;YACpB,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,CAAC;YAEpC,IAAI,KAAK,EAAE,CAAC;gBACV,kCAAkC;gBAClC,IAAI,CAAC,oBAAoB,EAAE,CAAC;YAC9B,CAAC;YAED,OAAO,KAAK,CAAC;QACf,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,8CAA8C,EAAE,KAAK,CAAC,CAAC;YACrE,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;;;;;;;OASG;IACH,KAAK,CAAC,QAAQ;QACZ,MAAM,SAAS,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC9C,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC;QAC/C,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO,CAAC,IAAI,CAAC,8CAA8C,CAAC,CAAC;YAC7D,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,CAAC;YACH,oDAAoD;YACpD,0EAA0E;YAC1E,MAAM,YAAY,GAAG,MAAM,SAAS,CAAC,aAAa,CAAC,QAAQ,CAAC,2BAA2B,CAAC,CAAC;YACzF,kCAAkC;YAClC,MAAM,SAAS,CAAC,aAAa,CAAC,KAAK,CAAC;YAEpC,MAAM,KAAK,GAAG,MAAM,QAAQ,CAAC,SAAS,EAAE;gBACtC,QAAQ;gBACR,yBAAyB,EAAE,YAAY;aACxC,CAAC,CAAC;YAEH,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;gBACrB,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK;gBAC1B,KAAK;aACN,CAAC,CAAC;YAEH,OAAO,KAAK,CAAC;QACf,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAC;YACpD,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;;;;;;OAQG;IACH,KAAK,CAAC,WAAW;QACf,MAAM,SAAS,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC9C,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,MAAM,WAAW,CAAC,SAAS,CAAC,CAAC;YAE7B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;gBACrB,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK;gBAC1B,KAAK,EAAE,IAAI;aACZ,CAAC,CAAC;YAEH,+BAA+B;YAC/B,IAAI,IAAI,CAAC,oBAAoB,EAAE,CAAC;gBAC9B,IAAI,CAAC,oBAAoB,EAAE,CAAC;gBAC5B,IAAI,CAAC,oBAAoB,GAAG,SAAS,CAAC;YACxC,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAC;YACpD,MAAM,IAAI,KAAK,CAAC,gDAAgD,CAAC,CAAC;QACpE,CAAC;IACH,CAAC;IAED,8EAA8E;IAC9E,WAAW;IACX,8EAA8E;IAE9E;;;;;;;;;;;;;;OAcG;IACH,SAAS;QACP,OAAO,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,CAAC;IAC5C,CAAC;IAED;;OAEG;IACK,oBAAoB;QAC1B,MAAM,SAAS,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC9C,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,oBAAoB,EAAE,CAAC;YAC5C,OAAO;QACT,CAAC;QAED,IAAI,CAAC,oBAAoB,GAAG,SAAS,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,EAAE;YAC3D,MAAM,YAAY,GAAwB;gBACxC,KAAK,EAAE,OAAO,CAAC,YAAY,EAAE,KAAK;gBAClC,IAAI,EAAE,OAAO,CAAC,YAAY,EAAE,IAAI;gBAChC,KAAK,EAAE,OAAO,CAAC,YAAY,EAAE,KAAK;gBAClC,IAAI,EAAE,OAAO,CAAC,IAA8B;gBAC5C,SAAS,EAAE,OAAO,CAAC,SAAS;aAC7B,CAAC;YAEF,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACzC,CAAC,CAAC,CAAC;IACL,CAAC;IAED,8EAA8E;IAC9E,sBAAsB;IACtB,8EAA8E;IAE9E;;;;;;;;;;;;;;;;OAgBG;IACH,kBAAkB;QAChB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACxC,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,IAAI,CAAC,CAAC,cAAc,IAAI,MAAM,CAAC,EAAE,CAAC;YAChC,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,OAAO,YAAY,CAAC,UAAoC,CAAC;IAC3D,CAAC;IAED;;;;;;;;;;;;;OAaG;IACH,KAAK,CAAC,WAAW;QACf,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC;IAC7B,CAAC;IAED;;;;OAIG;IACH,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC;IACvC,CAAC;IAED;;OAEG;IACH,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,YAAY,CAAC,YAAY,EAAE,CAAC;IAC1C,CAAC;IAED;;OAEG;IACH,IAAI,aAAa;QACf,OAAO,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,UAAU,KAAK,SAAS,CAAC;IAC1D,CAAC;IAED,8EAA8E;IAC9E,4BAA4B;IAC5B,8EAA8E;IAE9E;;;;;;;;;;;;;;;;;;;;;;;;;;OA0BG;IACH,mBAAmB;QACjB,OAAO,IAAI,CAAC,wBAAwB,CAAC,YAAY,EAAE,CAAC;IACtD,CAAC;IAED;;;;;;;;;;;;;;;;;;;;OAoBG;IACH,qBAAqB,CAAC,IAA6B;QACjD,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,EAAE,CAAC;QACZ,CAAC;QAED,MAAM,MAAM,GAAuB,EAAE,CAAC;QAEtC,eAAe;QACf,IAAI,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YAClB,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC;QAC/B,CAAC;QAED,cAAc;QACd,IAAI,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YAChB,MAAM,CAAC,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC;QAC3B,CAAC;QAED,iBAAiB;QACjB,IAAI,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC;YACxB,MAAM,CAAC,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC;QAC1C,CAAC;QAED,8CAA8C;QAC9C,IAAI,IAAI,CAAC,cAAc,CAAC,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,CAAC,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC;YACxD,CAAC;YAAC,MAAM,CAAC;gBACP,wDAAwD;gBACxD,MAAM,CAAC,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC;YACnE,CAAC;QACH,CAAC;QAED,wCAAwC;QACxC,MAAM,UAAU,GAA4B,EAAE,CAAC;QAC/C,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;YAChD,IAAI,GAAG,CAAC,UAAU,CAAC,SAAS,CAAC,IAAI,GAAG,KAAK,aAAa,EAAE,CAAC;gBACvD,MAAM,QAAQ,GAAG,GAAG,CAAC,OAAO,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;gBAC5C,sCAAsC;gBACtC,IAAI,CAAC;oBACH,UAAU,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;gBAC3C,CAAC;gBAAC,MAAM,CAAC;oBACP,UAAU,CAAC,QAAQ,CAAC,GAAG,KAAK,CAAC;gBAC/B,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAI,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvC,MAAM,CAAC,UAAU,GAAG,UAAU,CAAC;QACjC,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;;;;;;;;;;OAkBG;IACH,uBAAuB,CAAC,YAAiC;QACvD,MAAM,MAAM,GAAG,IAAI,CAAC,qBAAqB,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;QAE7D,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC;YACjC,YAAY;YACZ,MAAM;YACN,SAAS,EAAE,IAAI,IAAI,EAAE;SACtB,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,mBAAmB,CAAC,IAA6B;QAC/C,IAAI,CAAC,IAAI;YAAE,OAAO,KAAK,CAAC;QACxB,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1C,CAAC;IAED;;OAEG;IACK,gBAAgB,CAAC,WAAmB;QAC1C,MAAM,MAAM,GAA2B,EAAE,CAAC;QAE1C,IAAI,CAAC,WAAW;YAAE,OAAO,MAAM,CAAC;QAEhC,8BAA8B;QAC9B,MAAM,UAAU,GAAG,WAAW,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC;QAEpF,KAAK,MAAM,IAAI,IAAI,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC;YACzC,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACrC,IAAI,GAAG,EAAE,CAAC;gBACR,MAAM,CAAC,kBAAkB,CAAC,GAAG,CAAC,CAAC,GAAG,kBAAkB,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC;YACpE,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;+GAvhBU,gBAAgB,0CAajB,uBAAuB,aACvB,WAAW;mHAdV,gBAAgB,cADH,MAAM;;4FACnB,gBAAgB;kBAD5B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;0BAc7B,MAAM;2BAAC,uBAAuB;;0BAC9B,MAAM;2BAAC,WAAW","sourcesContent":["/**\n * Messaging Service (FCM)\n *\n * Servicio para Firebase Cloud Messaging (Push Notifications).\n * Permite solicitar permisos, obtener tokens, escuchar mensajes y manejar\n * navegación (deep linking) cuando el usuario toca una notificación.\n */\n\nimport { Inject, Injectable, Injector, NgZone, PLATFORM_ID } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { Messaging, getToken, deleteToken, onMessage } from '@angular/fire/messaging';\nimport { Observable, Subject, BehaviorSubject } from 'rxjs';\n\nimport { VALTECH_FIREBASE_CONFIG } from './config';\nimport {\n  NotificationAction,\n  NotificationClickEvent,\n  NotificationPayload,\n  NotificationPermission,\n  ValtechFirebaseConfig,\n} from './types';\n\n/**\n * Estado interno del servicio de messaging\n */\ninterface MessagingState {\n  token: string | null;\n  permission: NotificationPermission;\n  isSupported: boolean;\n}\n\n/**\n * Servicio para Firebase Cloud Messaging (FCM).\n *\n * Permite recibir notificaciones push en la aplicación web.\n * Requiere VAPID key configurada en ValtechFirebaseConfig.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class NotificationComponent {\n *   private messaging = inject(MessagingService);\n *\n *   token = signal<string | null>(null);\n *\n *   async enableNotifications() {\n *     // Solicitar permiso y obtener token\n *     const token = await this.messaging.requestPermission();\n *\n *     if (token) {\n *       this.token.set(token);\n *       // Enviar token a tu backend para almacenarlo\n *       await this.backend.registerDeviceToken(token);\n *     }\n *   }\n *\n *   // Escuchar mensajes en foreground\n *   messages$ = this.messaging.onMessage();\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class MessagingService {\n  private messageSubject = new Subject<NotificationPayload>();\n  private notificationClickSubject = new Subject<NotificationClickEvent>();\n  private stateSubject = new BehaviorSubject<MessagingState>({\n    token: null,\n    permission: 'default',\n    isSupported: false,\n  });\n\n  private unsubscribeOnMessage?: () => void;\n\n  constructor(\n    private injector: Injector,\n    @Inject(VALTECH_FIREBASE_CONFIG) private config: ValtechFirebaseConfig,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    private ngZone: NgZone\n  ) {\n    this.initializeMessaging();\n  }\n\n  /**\n   * Obtiene la instancia de Messaging de forma perezosa.\n   * Esto evita el error de APP_INITIALIZER de AngularFire.\n   */\n  private getMessagingInstance(): Messaging | null {\n    try {\n      return this.injector.get(Messaging, null);\n    } catch {\n      return null;\n    }\n  }\n\n  // ===========================================================================\n  // INICIALIZACIÓN\n  // ===========================================================================\n\n  /**\n   * Inicializa el servicio de messaging\n   */\n  private async initializeMessaging(): Promise<void> {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    const supported = await this.checkSupport();\n    const permission = this.getPermissionState();\n\n    this.stateSubject.next({\n      ...this.stateSubject.value,\n      isSupported: supported,\n      permission,\n    });\n\n    // Si ya tiene permiso, configurar listeners\n    if (supported && permission === 'granted') {\n      this.setupMessageListener();\n    }\n\n    // Escuchar mensajes del Service Worker (clicks en notificaciones background)\n    this.setupServiceWorkerListener();\n  }\n\n  /**\n   * Configura listener para mensajes del Service Worker.\n   * Recibe eventos cuando el usuario hace click en una notificación background.\n   */\n  private setupServiceWorkerListener(): void {\n    if (!isPlatformBrowser(this.platformId) || !('serviceWorker' in navigator)) {\n      return;\n    }\n\n    navigator.serviceWorker.addEventListener('message', (event) => {\n      // Verificar que es un mensaje de notificación click\n      if (event.data?.type === 'NOTIFICATION_CLICK') {\n        this.ngZone.run(() => {\n          const notification = event.data.notification as NotificationPayload;\n          const action = this.extractActionFromData(notification.data);\n\n          this.notificationClickSubject.next({\n            notification,\n            action,\n            timestamp: new Date(),\n          });\n        });\n      }\n    });\n  }\n\n  /**\n   * Verifica si FCM está soportado en el navegador actual\n   */\n  private async checkSupport(): Promise<boolean> {\n    if (!isPlatformBrowser(this.platformId)) {\n      return false;\n    }\n\n    // Verificar APIs necesarias\n    if (!('Notification' in window)) {\n      return false;\n    }\n\n    if (!('serviceWorker' in navigator)) {\n      return false;\n    }\n\n    // Verificar que messaging esté disponible\n    if (!this.getMessagingInstance()) {\n      return false;\n    }\n\n    return true;\n  }\n\n  // ===========================================================================\n  // PERMISOS Y TOKEN\n  // ===========================================================================\n\n  /**\n   * Solicita permiso de notificaciones y obtiene el token FCM.\n   *\n   * @returns Token FCM si se otorgó permiso, null si se denegó\n   *\n   * @example\n   * ```typescript\n   * const token = await messaging.requestPermission();\n   * if (token) {\n   *   console.log('Token FCM:', token);\n   *   // Enviar a backend\n   * } else {\n   *   console.log('Permiso denegado o no soportado');\n   * }\n   * ```\n   */\n  async requestPermission(): Promise<string | null> {\n    if (!await this.isSupported()) {\n      console.warn('FCM no está soportado en este navegador');\n      return null;\n    }\n\n    try {\n      // Solicitar permiso de notificaciones\n      const permission = await Notification.requestPermission();\n\n      this.stateSubject.next({\n        ...this.stateSubject.value,\n        permission: permission as NotificationPermission,\n      });\n\n      if (permission !== 'granted') {\n        console.warn('Permiso de notificaciones denegado');\n        return null;\n      }\n\n      // Obtener token FCM\n      const token = await this.getToken();\n\n      if (token) {\n        // Configurar listener de mensajes\n        this.setupMessageListener();\n      }\n\n      return token;\n    } catch (error) {\n      console.error('Error solicitando permiso de notificaciones:', error);\n      return null;\n    }\n  }\n\n  /**\n   * Obtiene el token FCM actual (sin solicitar permiso).\n   *\n   * @returns Token FCM si está disponible, null si no\n   *\n   * @example\n   * ```typescript\n   * const token = await messaging.getToken();\n   * ```\n   */\n  async getToken(): Promise<string | null> {\n    const messaging = this.getMessagingInstance();\n    if (!messaging) {\n      return null;\n    }\n\n    const vapidKey = this.config.messagingVapidKey;\n    if (!vapidKey) {\n      console.warn('VAPID key no configurada. FCM no funcionará.');\n      return null;\n    }\n\n    try {\n      // Registrar SW personalizado antes de obtener token\n      // Esto es necesario cuando usamos un SW custom (firebase-messaging-sw.js)\n      const registration = await navigator.serviceWorker.register('/firebase-messaging-sw.js');\n      // Esperar a que el SW esté activo\n      await navigator.serviceWorker.ready;\n\n      const token = await getToken(messaging, {\n        vapidKey,\n        serviceWorkerRegistration: registration,\n      });\n\n      this.stateSubject.next({\n        ...this.stateSubject.value,\n        token,\n      });\n\n      return token;\n    } catch (error) {\n      console.error('Error obteniendo token FCM:', error);\n      return null;\n    }\n  }\n\n  /**\n   * Elimina el token FCM actual (unsubscribe de notificaciones).\n   *\n   * @example\n   * ```typescript\n   * await messaging.deleteToken();\n   * console.log('Token eliminado, no recibirá más notificaciones');\n   * ```\n   */\n  async deleteToken(): Promise<void> {\n    const messaging = this.getMessagingInstance();\n    if (!messaging) {\n      return;\n    }\n\n    try {\n      await deleteToken(messaging);\n\n      this.stateSubject.next({\n        ...this.stateSubject.value,\n        token: null,\n      });\n\n      // Limpiar listener de mensajes\n      if (this.unsubscribeOnMessage) {\n        this.unsubscribeOnMessage();\n        this.unsubscribeOnMessage = undefined;\n      }\n    } catch (error) {\n      console.error('Error eliminando token FCM:', error);\n      throw new Error('No se pudo eliminar el token de notificaciones');\n    }\n  }\n\n  // ===========================================================================\n  // MENSAJES\n  // ===========================================================================\n\n  /**\n   * Observable de mensajes recibidos en foreground.\n   *\n   * IMPORTANTE: Los mensajes en background son manejados por el Service Worker.\n   *\n   * @returns Observable que emite cuando llega un mensaje en foreground\n   *\n   * @example\n   * ```typescript\n   * messaging.onMessage().subscribe(payload => {\n   *   console.log('Mensaje recibido:', payload);\n   *   // Mostrar notificación custom o actualizar UI\n   * });\n   * ```\n   */\n  onMessage(): Observable<NotificationPayload> {\n    return this.messageSubject.asObservable();\n  }\n\n  /**\n   * Configura el listener de mensajes en foreground\n   */\n  private setupMessageListener(): void {\n    const messaging = this.getMessagingInstance();\n    if (!messaging || this.unsubscribeOnMessage) {\n      return;\n    }\n\n    this.unsubscribeOnMessage = onMessage(messaging, (payload) => {\n      const notification: NotificationPayload = {\n        title: payload.notification?.title,\n        body: payload.notification?.body,\n        image: payload.notification?.image,\n        data: payload.data as Record<string, string>,\n        messageId: payload.messageId,\n      };\n\n      this.messageSubject.next(notification);\n    });\n  }\n\n  // ===========================================================================\n  // ESTADO Y UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Obtiene el estado actual del permiso de notificaciones.\n   *\n   * @returns 'granted' | 'denied' | 'default'\n   *\n   * @example\n   * ```typescript\n   * const permission = messaging.getPermissionState();\n   * if (permission === 'granted') {\n   *   // Ya tiene permiso\n   * } else if (permission === 'default') {\n   *   // Puede solicitar permiso\n   * } else {\n   *   // Denegado, debe habilitar manualmente\n   * }\n   * ```\n   */\n  getPermissionState(): NotificationPermission {\n    if (!isPlatformBrowser(this.platformId)) {\n      return 'default';\n    }\n\n    if (!('Notification' in window)) {\n      return 'denied';\n    }\n\n    return Notification.permission as NotificationPermission;\n  }\n\n  /**\n   * Verifica si FCM está soportado en el navegador actual.\n   *\n   * @returns true si FCM está soportado\n   *\n   * @example\n   * ```typescript\n   * if (await messaging.isSupported()) {\n   *   // Puede usar notificaciones push\n   * } else {\n   *   // Navegador no soporta o no tiene Service Worker\n   * }\n   * ```\n   */\n  async isSupported(): Promise<boolean> {\n    return this.checkSupport();\n  }\n\n  /**\n   * Obtiene el token actual sin hacer request.\n   *\n   * @returns Token almacenado o null\n   */\n  get currentToken(): string | null {\n    return this.stateSubject.value.token;\n  }\n\n  /**\n   * Observable del estado completo del servicio de messaging.\n   */\n  get state$(): Observable<MessagingState> {\n    return this.stateSubject.asObservable();\n  }\n\n  /**\n   * Verifica si el usuario ya otorgó permiso de notificaciones.\n   */\n  get hasPermission(): boolean {\n    return this.stateSubject.value.permission === 'granted';\n  }\n\n  // ===========================================================================\n  // DEEP LINKING / NAVEGACIÓN\n  // ===========================================================================\n\n  /**\n   * Observable de clicks en notificaciones.\n   *\n   * Emite cuando el usuario hace click en una notificación (foreground o background).\n   * Usa este observable para navegar a la página correspondiente.\n   *\n   * @returns Observable que emite NotificationClickEvent\n   *\n   * @example\n   * ```typescript\n   * @Component({...})\n   * export class AppComponent {\n   *   private messaging = inject(MessagingService);\n   *   private router = inject(Router);\n   *\n   *   constructor() {\n   *     this.messaging.onNotificationClick().subscribe(event => {\n   *       if (event.action.route) {\n   *         this.router.navigate([event.action.route], {\n   *           queryParams: event.action.queryParams\n   *         });\n   *       }\n   *     });\n   *   }\n   * }\n   * ```\n   */\n  onNotificationClick(): Observable<NotificationClickEvent> {\n    return this.notificationClickSubject.asObservable();\n  }\n\n  /**\n   * Extrae la acción de navegación de los datos de una notificación.\n   *\n   * Busca campos específicos en el payload de datos:\n   * - `route`: Ruta interna de la app (ej: '/orders/123')\n   * - `url`: URL externa (ej: 'https://example.com')\n   * - `action_type`: Tipo de acción personalizada\n   * - Campos con prefijo `action_`: Datos adicionales\n   *\n   * @param data - Datos del payload de la notificación\n   * @returns Acción de navegación extraída\n   *\n   * @example\n   * ```typescript\n   * // Payload desde el backend:\n   * // { route: '/orders/123', action_type: 'view_order', action_orderId: '123' }\n   *\n   * const action = messaging.extractActionFromData(notification.data);\n   * // { route: '/orders/123', actionType: 'view_order', actionData: { orderId: '123' } }\n   * ```\n   */\n  extractActionFromData(data?: Record<string, string>): NotificationAction {\n    if (!data) {\n      return {};\n    }\n\n    const action: NotificationAction = {};\n\n    // Ruta interna\n    if (data['route']) {\n      action.route = data['route'];\n    }\n\n    // URL externa\n    if (data['url']) {\n      action.url = data['url'];\n    }\n\n    // Tipo de acción\n    if (data['action_type']) {\n      action.actionType = data['action_type'];\n    }\n\n    // Query params (puede venir como JSON string)\n    if (data['query_params']) {\n      try {\n        action.queryParams = JSON.parse(data['query_params']);\n      } catch {\n        // Si no es JSON válido, intentar parsear como key=value\n        action.queryParams = this.parseQueryString(data['query_params']);\n      }\n    }\n\n    // Datos adicionales con prefijo action_\n    const actionData: Record<string, unknown> = {};\n    for (const [key, value] of Object.entries(data)) {\n      if (key.startsWith('action_') && key !== 'action_type') {\n        const cleanKey = key.replace('action_', '');\n        // Intentar parsear JSON si es posible\n        try {\n          actionData[cleanKey] = JSON.parse(value);\n        } catch {\n          actionData[cleanKey] = value;\n        }\n      }\n    }\n\n    if (Object.keys(actionData).length > 0) {\n      action.actionData = actionData;\n    }\n\n    return action;\n  }\n\n  /**\n   * Emite manualmente un evento de click en notificación.\n   *\n   * Útil para manejar clicks en notificaciones foreground donde\n   * la app decide mostrar un banner custom.\n   *\n   * @param notification - Payload de la notificación\n   *\n   * @example\n   * ```typescript\n   * messaging.onMessage().subscribe(notification => {\n   *   // Mostrar banner custom\n   *   this.showBanner(notification, () => {\n   *     // Usuario hizo click en el banner\n   *     messaging.handleNotificationClick(notification);\n   *   });\n   * });\n   * ```\n   */\n  handleNotificationClick(notification: NotificationPayload): void {\n    const action = this.extractActionFromData(notification.data);\n\n    this.notificationClickSubject.next({\n      notification,\n      action,\n      timestamp: new Date(),\n    });\n  }\n\n  /**\n   * Verifica si una notificación tiene acción de navegación.\n   *\n   * @param data - Datos del payload\n   * @returns true si tiene route o url\n   */\n  hasNavigationAction(data?: Record<string, string>): boolean {\n    if (!data) return false;\n    return !!(data['route'] || data['url']);\n  }\n\n  /**\n   * Parsea un query string en un objeto.\n   */\n  private parseQueryString(queryString: string): Record<string, string> {\n    const params: Record<string, string> = {};\n\n    if (!queryString) return params;\n\n    // Remover ? inicial si existe\n    const cleanQuery = queryString.startsWith('?') ? queryString.slice(1) : queryString;\n\n    for (const pair of cleanQuery.split('&')) {\n      const [key, value] = pair.split('=');\n      if (key) {\n        params[decodeURIComponent(key)] = decodeURIComponent(value || '');\n      }\n    }\n\n    return params;\n  }\n}\n"]}
|
|
573
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"messaging.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/messaging.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Messaging Service (FCM)\n *\n * Servicio para Firebase Cloud Messaging (Push Notifications).\n * Permite solicitar permisos, obtener tokens, escuchar mensajes y manejar\n * navegación (deep linking) cuando el usuario toca una notificación.\n */\n\nimport { Inject, Injectable, Injector, NgZone, PLATFORM_ID } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { Messaging, getToken, deleteToken, onMessage } from '@angular/fire/messaging';\nimport { Observable, Subject, BehaviorSubject } from 'rxjs';\n\nimport { VALTECH_FIREBASE_CONFIG } from './config';\nimport {\n  NotificationAction,\n  NotificationClickEvent,\n  NotificationPayload,\n  NotificationPermission,\n  ValtechFirebaseConfig,\n} from './types';\n\n/**\n * Estado interno del servicio de messaging\n */\ninterface MessagingState {\n  token: string | null;\n  permission: NotificationPermission;\n  isSupported: boolean;\n}\n\n/**\n * Servicio para Firebase Cloud Messaging (FCM).\n *\n * Permite recibir notificaciones push en la aplicación web.\n * Requiere VAPID key configurada en ValtechFirebaseConfig.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class NotificationComponent {\n *   private messaging = inject(MessagingService);\n *\n *   token = signal<string | null>(null);\n *\n *   async enableNotifications() {\n *     // Solicitar permiso y obtener token\n *     const token = await this.messaging.requestPermission();\n *\n *     if (token) {\n *       this.token.set(token);\n *       // Enviar token a tu backend para almacenarlo\n *       await this.backend.registerDeviceToken(token);\n *     }\n *   }\n *\n *   // Escuchar mensajes en foreground\n *   messages$ = this.messaging.onMessage();\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class MessagingService {\n  private messageSubject = new Subject<NotificationPayload>();\n  private notificationClickSubject = new Subject<NotificationClickEvent>();\n  private stateSubject = new BehaviorSubject<MessagingState>({\n    token: null,\n    permission: 'default',\n    isSupported: false,\n  });\n\n  private unsubscribeOnMessage?: () => void;\n\n  constructor(\n    private injector: Injector,\n    @Inject(VALTECH_FIREBASE_CONFIG) private config: ValtechFirebaseConfig,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    private ngZone: NgZone\n  ) {\n    this.initializeMessaging();\n  }\n\n  /**\n   * Obtiene la instancia de Messaging de forma perezosa.\n   * Esto evita el error de APP_INITIALIZER de AngularFire.\n   */\n  private getMessagingInstance(): Messaging | null {\n    try {\n      return this.injector.get(Messaging, null);\n    } catch {\n      return null;\n    }\n  }\n\n  // ===========================================================================\n  // INICIALIZACIÓN\n  // ===========================================================================\n\n  /**\n   * Inicializa el servicio de messaging\n   */\n  private async initializeMessaging(): Promise<void> {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    const supported = await this.checkSupport();\n    const permission = this.getPermissionState();\n\n    this.stateSubject.next({\n      ...this.stateSubject.value,\n      isSupported: supported,\n      permission,\n    });\n\n    // Si ya tiene permiso, configurar listeners\n    if (supported && permission === 'granted') {\n      this.setupMessageListener();\n    }\n\n    // Escuchar mensajes del Service Worker (clicks en notificaciones background)\n    this.setupServiceWorkerListener();\n  }\n\n  /**\n   * Configura listener para mensajes del Service Worker.\n   * Recibe eventos cuando el usuario hace click en una notificación background.\n   */\n  private setupServiceWorkerListener(): void {\n    if (!isPlatformBrowser(this.platformId) || !('serviceWorker' in navigator)) {\n      return;\n    }\n\n    navigator.serviceWorker.addEventListener('message', (event) => {\n      // Verificar que es un mensaje de notificación click\n      if (event.data?.type === 'NOTIFICATION_CLICK') {\n        this.ngZone.run(() => {\n          const notification = event.data.notification as NotificationPayload;\n          const action = this.extractActionFromData(notification.data);\n\n          this.notificationClickSubject.next({\n            notification,\n            action,\n            timestamp: new Date(),\n          });\n        });\n      }\n    });\n  }\n\n  /**\n   * Verifica si FCM está soportado en el navegador actual\n   */\n  private async checkSupport(): Promise<boolean> {\n    if (!isPlatformBrowser(this.platformId)) {\n      return false;\n    }\n\n    // Verificar APIs necesarias\n    if (!('Notification' in window)) {\n      return false;\n    }\n\n    if (!('serviceWorker' in navigator)) {\n      return false;\n    }\n\n    // Verificar que messaging esté disponible\n    if (!this.getMessagingInstance()) {\n      return false;\n    }\n\n    return true;\n  }\n\n  // ===========================================================================\n  // PERMISOS Y TOKEN\n  // ===========================================================================\n\n  /**\n   * Solicita permiso de notificaciones y obtiene el token FCM.\n   *\n   * @returns Token FCM si se otorgó permiso, null si se denegó\n   *\n   * @example\n   * ```typescript\n   * const token = await messaging.requestPermission();\n   * if (token) {\n   *   console.log('Token FCM:', token);\n   *   // Enviar a backend\n   * } else {\n   *   console.log('Permiso denegado o no soportado');\n   * }\n   * ```\n   */\n  async requestPermission(): Promise<string | null> {\n    if (!await this.isSupported()) {\n      console.warn('FCM no está soportado en este navegador');\n      return null;\n    }\n\n    try {\n      // Solicitar permiso de notificaciones\n      const permission = await Notification.requestPermission();\n\n      this.stateSubject.next({\n        ...this.stateSubject.value,\n        permission: permission as NotificationPermission,\n      });\n\n      if (permission !== 'granted') {\n        console.warn('Permiso de notificaciones denegado');\n        return null;\n      }\n\n      // Obtener token FCM\n      const token = await this.getToken();\n\n      if (token) {\n        // Configurar listener de mensajes\n        this.setupMessageListener();\n      }\n\n      return token;\n    } catch (error) {\n      console.error('Error solicitando permiso de notificaciones:', error);\n      return null;\n    }\n  }\n\n  /**\n   * Obtiene el token FCM actual (sin solicitar permiso).\n   *\n   * @returns Token FCM si está disponible, null si no\n   *\n   * @example\n   * ```typescript\n   * const token = await messaging.getToken();\n   * ```\n   */\n  async getToken(): Promise<string | null> {\n    const messaging = this.getMessagingInstance();\n    if (!messaging) {\n      return null;\n    }\n\n    const vapidKey = this.config.messagingVapidKey;\n    if (!vapidKey) {\n      console.warn('VAPID key no configurada. FCM no funcionará.');\n      return null;\n    }\n\n    try {\n      // Registrar SW personalizado antes de obtener token\n      // Esto es necesario cuando usamos un SW custom (firebase-messaging-sw.js)\n      const registration = await navigator.serviceWorker.register('/firebase-messaging-sw.js');\n      // Esperar a que el SW esté activo\n      await navigator.serviceWorker.ready;\n\n      const token = await getToken(messaging, {\n        vapidKey,\n        serviceWorkerRegistration: registration,\n      });\n\n      this.stateSubject.next({\n        ...this.stateSubject.value,\n        token,\n      });\n\n      return token;\n    } catch (error) {\n      console.error('Error obteniendo token FCM:', error);\n      return null;\n    }\n  }\n\n  /**\n   * Elimina el token FCM actual (unsubscribe de notificaciones).\n   *\n   * @example\n   * ```typescript\n   * await messaging.deleteToken();\n   * console.log('Token eliminado, no recibirá más notificaciones');\n   * ```\n   */\n  async deleteToken(): Promise<void> {\n    const messaging = this.getMessagingInstance();\n    if (!messaging) {\n      return;\n    }\n\n    try {\n      await deleteToken(messaging);\n\n      this.stateSubject.next({\n        ...this.stateSubject.value,\n        token: null,\n      });\n\n      // Limpiar listener de mensajes\n      if (this.unsubscribeOnMessage) {\n        this.unsubscribeOnMessage();\n        this.unsubscribeOnMessage = undefined;\n      }\n    } catch (error) {\n      console.error('Error eliminando token FCM:', error);\n      throw new Error('No se pudo eliminar el token de notificaciones');\n    }\n  }\n\n  // ===========================================================================\n  // MENSAJES\n  // ===========================================================================\n\n  /**\n   * Observable de mensajes recibidos en foreground.\n   *\n   * IMPORTANTE: Los mensajes en background son manejados por el Service Worker.\n   *\n   * @returns Observable que emite cuando llega un mensaje en foreground\n   *\n   * @example\n   * ```typescript\n   * messaging.onMessage().subscribe(payload => {\n   *   console.log('Mensaje recibido:', payload);\n   *   // Mostrar notificación custom o actualizar UI\n   * });\n   * ```\n   */\n  onMessage(): Observable<NotificationPayload> {\n    return this.messageSubject.asObservable();\n  }\n\n  /**\n   * Configura el listener de mensajes en foreground\n   */\n  private setupMessageListener(): void {\n    const messaging = this.getMessagingInstance();\n    if (!messaging || this.unsubscribeOnMessage) {\n      return;\n    }\n\n    this.unsubscribeOnMessage = onMessage(messaging, (payload) => {\n      const notification: NotificationPayload = {\n        title: payload.notification?.title,\n        body: payload.notification?.body,\n        image: payload.notification?.image,\n        data: payload.data as Record<string, string>,\n        messageId: payload.messageId,\n      };\n\n      this.messageSubject.next(notification);\n    });\n  }\n\n  // ===========================================================================\n  // ESTADO Y UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Obtiene el estado actual del permiso de notificaciones.\n   *\n   * @returns 'granted' | 'denied' | 'default'\n   *\n   * @example\n   * ```typescript\n   * const permission = messaging.getPermissionState();\n   * if (permission === 'granted') {\n   *   // Ya tiene permiso\n   * } else if (permission === 'default') {\n   *   // Puede solicitar permiso\n   * } else {\n   *   // Denegado, debe habilitar manualmente\n   * }\n   * ```\n   */\n  getPermissionState(): NotificationPermission {\n    if (!isPlatformBrowser(this.platformId)) {\n      return 'default';\n    }\n\n    if (!('Notification' in window)) {\n      return 'denied';\n    }\n\n    return Notification.permission as NotificationPermission;\n  }\n\n  /**\n   * Verifica si FCM está soportado en el navegador actual.\n   * Usa lógica inteligente para evitar falsos negativos por timing del Injector.\n   *\n   * @returns true si FCM está soportado\n   *\n   * @example\n   * ```typescript\n   * if (await messaging.isSupported()) {\n   *   // Puede usar notificaciones push\n   * } else {\n   *   // Navegador no soporta o no tiene Service Worker\n   * }\n   * ```\n   */\n  async isSupported(): Promise<boolean> {\n    // Si ya tenemos token en estado, claramente FCM funciona\n    if (this.stateSubject.value.token) {\n      return true;\n    }\n\n    // Si ya calculamos soporte exitosamente, usarlo\n    if (this.stateSubject.value.isSupported) {\n      return true;\n    }\n\n    // Verificaciones básicas que no dependen del Injector\n    if (!isPlatformBrowser(this.platformId)) {\n      return false;\n    }\n    if (!('Notification' in window) || !('serviceWorker' in navigator)) {\n      return false;\n    }\n\n    // Si el navegador soporta las APIs pero el Injector no está listo,\n    // podríamos tener un falso negativo. Intentar checkSupport de todas formas.\n    return this.checkSupport();\n  }\n\n  /**\n   * Obtiene el token actual sin hacer request.\n   *\n   * @returns Token almacenado o null\n   */\n  get currentToken(): string | null {\n    return this.stateSubject.value.token;\n  }\n\n  /**\n   * Observable del estado completo del servicio de messaging.\n   */\n  get state$(): Observable<MessagingState> {\n    return this.stateSubject.asObservable();\n  }\n\n  /**\n   * Verifica si el usuario ya otorgó permiso de notificaciones.\n   */\n  get hasPermission(): boolean {\n    return this.stateSubject.value.permission === 'granted';\n  }\n\n  /**\n   * Hidrata el estado del token desde un valor externo (ej: localStorage).\n   * Útil cuando el token ya existe pero el MessagingService no lo tiene en memoria.\n   *\n   * @param token Token FCM a setear\n   *\n   * @example\n   * ```typescript\n   * const storedToken = localStorage.getItem('fcm_token');\n   * if (storedToken) {\n   *   messaging.setTokenFromStorage(storedToken);\n   * }\n   * ```\n   */\n  setTokenFromStorage(token: string): void {\n    if (!token) return;\n\n    this.stateSubject.next({\n      ...this.stateSubject.value,\n      token,\n      isSupported: true, // Si hay token, claramente FCM funciona\n    });\n  }\n\n  // ===========================================================================\n  // DEEP LINKING / NAVEGACIÓN\n  // ===========================================================================\n\n  /**\n   * Observable de clicks en notificaciones.\n   *\n   * Emite cuando el usuario hace click en una notificación (foreground o background).\n   * Usa este observable para navegar a la página correspondiente.\n   *\n   * @returns Observable que emite NotificationClickEvent\n   *\n   * @example\n   * ```typescript\n   * @Component({...})\n   * export class AppComponent {\n   *   private messaging = inject(MessagingService);\n   *   private router = inject(Router);\n   *\n   *   constructor() {\n   *     this.messaging.onNotificationClick().subscribe(event => {\n   *       if (event.action.route) {\n   *         this.router.navigate([event.action.route], {\n   *           queryParams: event.action.queryParams\n   *         });\n   *       }\n   *     });\n   *   }\n   * }\n   * ```\n   */\n  onNotificationClick(): Observable<NotificationClickEvent> {\n    return this.notificationClickSubject.asObservable();\n  }\n\n  /**\n   * Extrae la acción de navegación de los datos de una notificación.\n   *\n   * Busca campos específicos en el payload de datos:\n   * - `route`: Ruta interna de la app (ej: '/orders/123')\n   * - `url`: URL externa (ej: 'https://example.com')\n   * - `action_type`: Tipo de acción personalizada\n   * - Campos con prefijo `action_`: Datos adicionales\n   *\n   * @param data - Datos del payload de la notificación\n   * @returns Acción de navegación extraída\n   *\n   * @example\n   * ```typescript\n   * // Payload desde el backend:\n   * // { route: '/orders/123', action_type: 'view_order', action_orderId: '123' }\n   *\n   * const action = messaging.extractActionFromData(notification.data);\n   * // { route: '/orders/123', actionType: 'view_order', actionData: { orderId: '123' } }\n   * ```\n   */\n  extractActionFromData(data?: Record<string, string>): NotificationAction {\n    if (!data) {\n      return {};\n    }\n\n    const action: NotificationAction = {};\n\n    // Ruta interna\n    if (data['route']) {\n      action.route = data['route'];\n    }\n\n    // URL externa\n    if (data['url']) {\n      action.url = data['url'];\n    }\n\n    // Tipo de acción\n    if (data['action_type']) {\n      action.actionType = data['action_type'];\n    }\n\n    // Query params (puede venir como JSON string)\n    if (data['query_params']) {\n      try {\n        action.queryParams = JSON.parse(data['query_params']);\n      } catch {\n        // Si no es JSON válido, intentar parsear como key=value\n        action.queryParams = this.parseQueryString(data['query_params']);\n      }\n    }\n\n    // Datos adicionales con prefijo action_\n    const actionData: Record<string, unknown> = {};\n    for (const [key, value] of Object.entries(data)) {\n      if (key.startsWith('action_') && key !== 'action_type') {\n        const cleanKey = key.replace('action_', '');\n        // Intentar parsear JSON si es posible\n        try {\n          actionData[cleanKey] = JSON.parse(value);\n        } catch {\n          actionData[cleanKey] = value;\n        }\n      }\n    }\n\n    if (Object.keys(actionData).length > 0) {\n      action.actionData = actionData;\n    }\n\n    return action;\n  }\n\n  /**\n   * Emite manualmente un evento de click en notificación.\n   *\n   * Útil para manejar clicks en notificaciones foreground donde\n   * la app decide mostrar un banner custom.\n   *\n   * @param notification - Payload de la notificación\n   *\n   * @example\n   * ```typescript\n   * messaging.onMessage().subscribe(notification => {\n   *   // Mostrar banner custom\n   *   this.showBanner(notification, () => {\n   *     // Usuario hizo click en el banner\n   *     messaging.handleNotificationClick(notification);\n   *   });\n   * });\n   * ```\n   */\n  handleNotificationClick(notification: NotificationPayload): void {\n    const action = this.extractActionFromData(notification.data);\n\n    this.notificationClickSubject.next({\n      notification,\n      action,\n      timestamp: new Date(),\n    });\n  }\n\n  /**\n   * Verifica si una notificación tiene acción de navegación.\n   *\n   * @param data - Datos del payload\n   * @returns true si tiene route o url\n   */\n  hasNavigationAction(data?: Record<string, string>): boolean {\n    if (!data) return false;\n    return !!(data['route'] || data['url']);\n  }\n\n  /**\n   * Parsea un query string en un objeto.\n   */\n  private parseQueryString(queryString: string): Record<string, string> {\n    const params: Record<string, string> = {};\n\n    if (!queryString) return params;\n\n    // Remover ? inicial si existe\n    const cleanQuery = queryString.startsWith('?') ? queryString.slice(1) : queryString;\n\n    for (const pair of cleanQuery.split('&')) {\n      const [key, value] = pair.split('=');\n      if (key) {\n        params[decodeURIComponent(key)] = decodeURIComponent(value || '');\n      }\n    }\n\n    return params;\n  }\n}\n"]}
|
|
@@ -23871,6 +23871,7 @@ class MessagingService {
|
|
|
23871
23871
|
}
|
|
23872
23872
|
/**
|
|
23873
23873
|
* Verifica si FCM está soportado en el navegador actual.
|
|
23874
|
+
* Usa lógica inteligente para evitar falsos negativos por timing del Injector.
|
|
23874
23875
|
*
|
|
23875
23876
|
* @returns true si FCM está soportado
|
|
23876
23877
|
*
|
|
@@ -23884,6 +23885,23 @@ class MessagingService {
|
|
|
23884
23885
|
* ```
|
|
23885
23886
|
*/
|
|
23886
23887
|
async isSupported() {
|
|
23888
|
+
// Si ya tenemos token en estado, claramente FCM funciona
|
|
23889
|
+
if (this.stateSubject.value.token) {
|
|
23890
|
+
return true;
|
|
23891
|
+
}
|
|
23892
|
+
// Si ya calculamos soporte exitosamente, usarlo
|
|
23893
|
+
if (this.stateSubject.value.isSupported) {
|
|
23894
|
+
return true;
|
|
23895
|
+
}
|
|
23896
|
+
// Verificaciones básicas que no dependen del Injector
|
|
23897
|
+
if (!isPlatformBrowser(this.platformId)) {
|
|
23898
|
+
return false;
|
|
23899
|
+
}
|
|
23900
|
+
if (!('Notification' in window) || !('serviceWorker' in navigator)) {
|
|
23901
|
+
return false;
|
|
23902
|
+
}
|
|
23903
|
+
// Si el navegador soporta las APIs pero el Injector no está listo,
|
|
23904
|
+
// podríamos tener un falso negativo. Intentar checkSupport de todas formas.
|
|
23887
23905
|
return this.checkSupport();
|
|
23888
23906
|
}
|
|
23889
23907
|
/**
|
|
@@ -23906,6 +23924,29 @@ class MessagingService {
|
|
|
23906
23924
|
get hasPermission() {
|
|
23907
23925
|
return this.stateSubject.value.permission === 'granted';
|
|
23908
23926
|
}
|
|
23927
|
+
/**
|
|
23928
|
+
* Hidrata el estado del token desde un valor externo (ej: localStorage).
|
|
23929
|
+
* Útil cuando el token ya existe pero el MessagingService no lo tiene en memoria.
|
|
23930
|
+
*
|
|
23931
|
+
* @param token Token FCM a setear
|
|
23932
|
+
*
|
|
23933
|
+
* @example
|
|
23934
|
+
* ```typescript
|
|
23935
|
+
* const storedToken = localStorage.getItem('fcm_token');
|
|
23936
|
+
* if (storedToken) {
|
|
23937
|
+
* messaging.setTokenFromStorage(storedToken);
|
|
23938
|
+
* }
|
|
23939
|
+
* ```
|
|
23940
|
+
*/
|
|
23941
|
+
setTokenFromStorage(token) {
|
|
23942
|
+
if (!token)
|
|
23943
|
+
return;
|
|
23944
|
+
this.stateSubject.next({
|
|
23945
|
+
...this.stateSubject.value,
|
|
23946
|
+
token,
|
|
23947
|
+
isSupported: true, // Si hay token, claramente FCM funciona
|
|
23948
|
+
});
|
|
23949
|
+
}
|
|
23909
23950
|
// ===========================================================================
|
|
23910
23951
|
// DEEP LINKING / NAVEGACIÓN
|
|
23911
23952
|
// ===========================================================================
|