-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathpulsar.js
More file actions
98 lines (89 loc) · 3.18 KB
/
pulsar.js
File metadata and controls
98 lines (89 loc) · 3.18 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
/**
* @file pulsar.js
* @description Un sistema de gestión de estado reactivo, modular y atómico.
* Basado en el patrón Observador, diseñado para simplicidad y rendimiento.
* A Radiant Store
*/
class Pulsar {
/**
* Crea una nueva instancia de almacenamiento (Pulsar).
* @param {Object} initialState - El estado inicial de la tienda.
*/
constructor(initialState) {
// Estado interno: La fuente de verdad.
this.state = initialState;
// Colección de suscriptores. Usamos un Set para:
// 1. Evitar duplicados automáticamente.
// 2. Optimizar la adición y eliminación de oyentes (O(1)).
this.listeners = new Set();
}
/**
* Suscribe una función (listener) a los cambios de estado.
*
* @param {Function} listener - Función que se ejecutará cuando el estado cambie.
* @param {boolean} [immediate=false] - Si es true, ejecuta el listener inmediatamente tras suscribirse.
* @returns {Function} Función de limpieza (cleanup) para cancelar la suscripción.
*/
subscribe(listener, immediate = false) {
this.listeners.add(listener);
// Ejecución inmediata opcional: Útil para sincronizar la UI con el estado actual
// al momento de montar un componente, sin esperar al primer cambio.
if (immediate) {
try {
listener();
} catch (error) {
console.error('[Pulsar] Error en ejecución inmediata del listener:', error);
}
}
// Retornamos una función de cleanup.
// Patrón funcional: const unsubscribe = Pulsar.subscribe(...)
return () => this.listeners.delete(listener);
}
/**
* Obtiene una instantánea del estado actual.
*
* @returns {Object} Una copia superficial (shallow copy) del estado.
* @note Se retorna una copia para promover la inmutabilidad y evitar
* que referencias externas modifiquen el estado interno sin usar setState.
*/
getState() {
return { ...this.state };
}
/**
* Método interno para notificar a todos los suscriptores.
* Itera sobre el Set de listeners y ejecuta cada uno.
* @private
*/
_notify() {
this.listeners.forEach(listener => {
try {
listener();
} catch (error) {
// Robustez: Un error en un listener no debe detener la propagación
// a los demás ni romper el flujo de la aplicación.
console.error('[Pulsar] Error notificado en listener:', error);
}
});
}
/**
* Actualiza el estado y emite una señal de cambio a los suscriptores.
* Realiza una fusión superficial (shallow merge) del estado nuevo con el anterior.
*
* @param {Object} newState - Objeto con las propiedades a actualizar.
*/
setState(newState) {
// Inmutabilidad: Creamos un nuevo objeto en lugar de mutar 'this.state'.
this.state = { ...this.state, ...newState };
this._notify();
}
}
/**
* Factory Function para crear Pulsars independientes.
* Promueve la modularidad creando instancias aisladas para diferentes dominios (Auth, UI, Data).
*
* @param {Object} initialState - Estado inicial para esta instancia.
* @returns {Pulsar} Nueva instancia de Pulsar.
*/
export function createStatePulsar(initialState) {
return new Pulsar(initialState);
}