valyrian.js 8.1.6 → 8.1.7
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/dist/lib/pulses/index.d.ts +6 -5
- package/dist/lib/pulses/index.d.ts.map +1 -1
- package/dist/pulses/index.js +20 -22
- package/dist/pulses/index.js.map +2 -2
- package/dist/pulses/index.min.js +1 -1
- package/dist/pulses/index.min.js.map +1 -1
- package/dist/pulses/index.mjs +20 -22
- package/dist/pulses/index.mjs.map +2 -2
- package/lib/pulses/index.ts +36 -44
- package/package.json +1 -1
|
@@ -1,13 +1,14 @@
|
|
|
1
1
|
type State = Record<string, any>;
|
|
2
|
-
export type
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
2
|
+
export type PulseContext = {
|
|
3
|
+
$flush: () => Promise<void>;
|
|
4
|
+
};
|
|
5
|
+
export type Pulse<StateType, TReturn = unknown> = (this: PulseContext, state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;
|
|
6
|
+
export type Pulses<StateType> = Record<string, Pulse<StateType, any>>;
|
|
6
7
|
type ProxyState<StateType> = StateType & {
|
|
7
8
|
[key: string]: any;
|
|
8
9
|
};
|
|
9
10
|
type StorePulses<PulsesType extends Pulses<any>> = {
|
|
10
|
-
[K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R ? (...args: Args) => R : never;
|
|
11
|
+
[K in keyof PulsesType]: PulsesType[K] extends (this: any, state: any, ...args: infer Args) => infer R ? (...args: Args) => R : never;
|
|
11
12
|
};
|
|
12
13
|
export declare function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(initialState: StateType, pulses: PulsesType): StorePulses<PulsesType> & {
|
|
13
14
|
state: ProxyState<StateType>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/pulses/index.ts"],"names":[],"mappings":"AAIA,KAAK,KAAK,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;AAEjC,MAAM,MAAM,KAAK,CAAC,SAAS,EAAE,OAAO,GAAG,OAAO,IAAI,
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/pulses/index.ts"],"names":[],"mappings":"AAIA,KAAK,KAAK,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;AAEjC,MAAM,MAAM,YAAY,GAAG;IACzB,MAAM,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC7B,CAAC;AAEF,MAAM,MAAM,KAAK,CAAC,SAAS,EAAE,OAAO,GAAG,OAAO,IAAI,CAChD,IAAI,EAAE,YAAY,EAClB,KAAK,EAAE,SAAS,EAChB,GAAG,IAAI,EAAE,GAAG,EAAE,KACX,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;AAEhC,MAAM,MAAM,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,CAAC;AAEtE,KAAK,UAAU,CAAC,SAAS,IAAI,SAAS,GAAG;IAAE,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAA;CAAE,CAAC;AAIhE,KAAK,WAAW,CAAC,UAAU,SAAS,MAAM,CAAC,GAAG,CAAC,IAAI;KAChD,CAAC,IAAI,MAAM,UAAU,GAAG,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,IAAI,EAAE,GAAG,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,MAAM,IAAI,KAAK,MAAM,CAAC,GAClG,CAAC,GAAG,IAAI,EAAE,IAAI,KAAK,CAAC,GACpB,KAAK;CACV,CAAC;AAmOF,wBAAgB,gBAAgB,CAAC,SAAS,SAAS,KAAK,EAAE,UAAU,SAAS,MAAM,CAAC,SAAS,CAAC,EAC5F,YAAY,EAAE,SAAS,EACvB,MAAM,EAAE,UAAU;;QA/Ld,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;SAC1C,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;EAiMjD;AAED,wBAAgB,kBAAkB,CAAC,SAAS,SAAS,KAAK,EAAE,UAAU,SAAS,MAAM,CAAC,SAAS,CAAC,EAC9F,YAAY,EAAE,SAAS,EACvB,MAAM,EAAE,UAAU;;QAtMd,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;SAC1C,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;EA2MjD;AAED,wBAAgB,YAAY,CAAC,MAAM,EAAE,QAAQ,QAU5C;AAED,wBAAgB,WAAW,CAAC,CAAC,EAAE,YAAY,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,IAAI,EAAE,MAAM,IAAI,CAAC,CA4BlH"}
|
package/dist/pulses/index.js
CHANGED
|
@@ -135,12 +135,13 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
135
135
|
if (currentState === null) {
|
|
136
136
|
currentState = (0, import_utils.deepCloneUnfreeze)(localState);
|
|
137
137
|
}
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
138
|
+
this.$flush = async () => {
|
|
139
|
+
if (currentState) {
|
|
140
|
+
setState(currentState, true);
|
|
141
|
+
currentState = (0, import_utils.deepCloneUnfreeze)(localState);
|
|
142
|
+
}
|
|
141
143
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
142
144
|
};
|
|
143
|
-
Reflect.set(this, "$flush", $flush);
|
|
144
145
|
const emptyFlush = async () => {
|
|
145
146
|
};
|
|
146
147
|
try {
|
|
@@ -148,16 +149,16 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
148
149
|
if (pulseResult instanceof Promise) {
|
|
149
150
|
return pulseResult.then((resolvedValue) => {
|
|
150
151
|
setState(currentState);
|
|
151
|
-
|
|
152
|
+
this.$flush = emptyFlush;
|
|
152
153
|
return resolvedValue;
|
|
153
154
|
}).catch((error) => {
|
|
154
155
|
console.error(`Error in pulse '${key}':`, error);
|
|
155
|
-
|
|
156
|
+
this.$flush = emptyFlush;
|
|
156
157
|
throw error;
|
|
157
158
|
});
|
|
158
159
|
} else {
|
|
159
160
|
setState(currentState);
|
|
160
|
-
|
|
161
|
+
this.$flush = emptyFlush;
|
|
161
162
|
return pulseResult;
|
|
162
163
|
}
|
|
163
164
|
} catch (error) {
|
|
@@ -199,20 +200,17 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
199
200
|
throw new Error(`Pulse '${prop}' does not exist`);
|
|
200
201
|
}
|
|
201
202
|
const pulseMethod = pulses2[prop];
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
};
|
|
214
|
-
}
|
|
215
|
-
return pulseMethod;
|
|
203
|
+
return (...args) => {
|
|
204
|
+
const result = pulseMethod.apply(pulseMethod, args);
|
|
205
|
+
if (result instanceof Promise) {
|
|
206
|
+
return result.then((r) => {
|
|
207
|
+
trigger("pulse", prop, args);
|
|
208
|
+
return r;
|
|
209
|
+
});
|
|
210
|
+
}
|
|
211
|
+
trigger("pulse", prop, args);
|
|
212
|
+
return result;
|
|
213
|
+
};
|
|
216
214
|
}
|
|
217
215
|
});
|
|
218
216
|
return pulsesProxy;
|
|
@@ -222,7 +220,7 @@ function createPulseStore(initialState, pulses) {
|
|
|
222
220
|
}
|
|
223
221
|
function createMutableStore(initialState, pulses) {
|
|
224
222
|
console.warn(
|
|
225
|
-
"Warning: You are working with a mutable state.
|
|
223
|
+
"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render."
|
|
226
224
|
);
|
|
227
225
|
return createStore(initialState, pulses, false);
|
|
228
226
|
}
|
package/dist/pulses/index.js.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../lib/pulses/index.ts"],
|
|
4
|
-
"sourcesContent": ["/* eslint-disable no-console */\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type Pulse<StateType, TReturn = unknown> = (state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;\n\nexport type Pulses<StateType> = Record<string, Pulse<StateType, any> & { $flush?: () => Promise<void> }>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType extends Pulses<any>> = {\n [K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R\n ? (...args: Args) => R\n : never;\n};\n\n/** Funci\u00F3n auxiliar para registrar la suscripci\u00F3n al nodo DOM.\n * Retorna true si se agreg\u00F3 la suscripci\u00F3n, o false si se encontr\u00F3 un nodo padre ya suscrito.\n */\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n const currentVnode = current.vnode as VnodeWithDom;\n if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n return;\n }\n\n let hasParent = false;\n let parent = currentVnode.dom.parentElement as DomElement;\n while (parent) {\n if (domWithVnodesToUpdate.has(parent)) {\n hasParent = true;\n break;\n }\n parent = parent.parentElement as DomElement;\n }\n\n // Si no hay nodo padre registrado, se crea la suscripci\u00F3n.\n if (!hasParent) {\n const dom = currentVnode.dom;\n const subscription = () => {\n updateVnode(dom.vnode);\n if (!dom.parentElement) {\n subscribers.delete(subscription);\n domWithVnodesToUpdate.delete(dom);\n }\n };\n subscribers.add(subscription);\n domWithVnodesToUpdate.add(dom);\n }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType | (() => StateType) | null,\n pulses: PulsesType,\n immutable = false\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const boundPulses: Pulses<StateType> = {};\n for (const key in pulses) {\n if (typeof pulses[key] !== \"function\") {\n throw new Error(`Pulse '${key}' must be a function`);\n }\n if (key === \"state\") {\n throw new Error(`A pulse cannot be named 'state'`);\n }\n boundPulses[key] = getPulseMethod(key);\n }\n\n const localState: StateType =\n (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n function isMutable() {\n if (immutable) {\n throw new Error(\"You need to call a pulse to modify the state\");\n }\n }\n\n let currentState: StateType | null = null;\n let pulseCallCount = 0;\n\n const proxyState = new Proxy(localState, {\n get: (state, prop: string) => {\n // If we are in a pulse, we return the value of the cloned state.\n if (currentState) {\n return currentState[prop];\n }\n\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n return state[prop];\n },\n set: (state, prop: string, value: any) => {\n isMutable();\n Reflect.set(state, prop, value);\n return true;\n },\n deleteProperty: (state, prop: string) => {\n isMutable();\n Reflect.deleteProperty(state, prop);\n return true;\n }\n });\n\n function syncState(newState: StateType) {\n for (const key in newState) {\n localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n }\n for (const key in localState) {\n if (!(key in newState)) {\n Reflect.deleteProperty(localState, key);\n }\n }\n }\n\n let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n function debouncedUpdate() {\n if (debounceTimeout) {\n clearTimeout(debounceTimeout);\n }\n debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n }\n\n function setState(newState: StateType, flush = false) {\n pulseCallCount--;\n if (!hasChanged(localState, newState)) {\n return;\n }\n if (pulseCallCount > 0 && !flush) {\n return;\n }\n syncState(newState);\n currentState = null;\n debouncedUpdate();\n }\n\n function getPulseMethod(key: string) {\n function pulseMethod(this: Pulse<StateType, any> & { $flush: () => Promise<void> }, ...args: any[]) {\n pulseCallCount++;\n if (currentState === null) {\n currentState = deepCloneUnfreeze(localState);\n }\n\n const $flush = async () => {\n setState(currentState as StateType, true);\n currentState = deepCloneUnfreeze(localState);\n await new Promise((resolve) => setTimeout(resolve, 0));\n };\n\n Reflect.set(this, \"$flush\", $flush);\n const emptyFlush = async () => {};\n\n try {\n const pulseResult = pulses[key].apply(this, [currentState, ...args] as any);\n if (pulseResult instanceof Promise) {\n return pulseResult\n .then((resolvedValue) => {\n setState(currentState as StateType);\n Reflect.set(this, \"$flush\", emptyFlush);\n return resolvedValue;\n })\n .catch((error) => {\n console.error(`Error in pulse '${key}':`, error);\n Reflect.set(this, \"$flush\", emptyFlush);\n throw error;\n });\n } else {\n setState(currentState);\n Reflect.set(this, \"$flush\", emptyFlush);\n return pulseResult;\n }\n } catch (error) {\n console.error(`Error in pulse '${key}':`, error);\n Reflect.set(this, \"$flush\", emptyFlush);\n throw error;\n }\n }\n\n return pulseMethod;\n }\n\n syncState(localState);\n\n const listeners: Record<string, Function[]> = {};\n const trigger = (event: string, ...args: any[]) => {\n if (listeners[event]) {\n listeners[event].forEach((callback) => callback(...args));\n }\n };\n\n const pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\n }\n if (prop === \"on\") {\n return (event: string, callback: Function) => {\n if (!listeners[event]) {\n listeners[event] = [];\n }\n listeners[event].push(callback);\n };\n }\n if (prop === \"off\") {\n return (event: string, callback: Function) => {\n if (listeners[event]) {\n listeners[event] = listeners[event].filter((cb) => cb !== callback);\n }\n };\n }\n if (!(prop in pulses)) {\n throw new Error(`Pulse '${prop}' does not exist`);\n }\n const pulseMethod = pulses[prop];\n\n if (typeof pulseMethod === \"function\") {\n return (...args: any[]) => {\n const result = pulseMethod.apply(pulsesProxy, args as any);\n if (result instanceof Promise) {\n return result.then((r) => {\n trigger(\"pulse\", prop, args);\n return r;\n });\n }\n trigger(\"pulse\", prop, args);\n return result;\n };\n }\n\n return pulseMethod;\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n console.warn(\n \"Warning: You are working with a mutable state. This can lead to unpredictable behavior. All state changes made outside of a pulse will not trigger a re-render.\"\n );\n return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n const runEffect = () => {\n try {\n effectStack.push(runEffect);\n effect();\n } finally {\n effectStack.pop();\n }\n };\n runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n let value = initialValue;\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const runSubscribers = () => {\n subscribers.forEach((subscriber) => subscriber());\n };\n\n const read = (): T => {\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n return value;\n };\n\n const write = (newValue: T | ((current: T) => T)): void => {\n const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n if (!hasChanged(value, resolvedValue)) {\n return;\n }\n value = resolvedValue;\n runSubscribers();\n };\n\n return [read, write, runSubscribers];\n}\n"],
|
|
5
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA,sBAA+D;AAC/D,mBAA0D;
|
|
4
|
+
"sourcesContent": ["/* eslint-disable no-console */\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type PulseContext = {\n $flush: () => Promise<void>;\n};\n\nexport type Pulse<StateType, TReturn = unknown> = (\n this: PulseContext,\n state: StateType,\n ...args: any[]\n) => TReturn | Promise<TReturn>;\n\nexport type Pulses<StateType> = Record<string, Pulse<StateType, any>>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType extends Pulses<any>> = {\n [K in keyof PulsesType]: PulsesType[K] extends (this: any, state: any, ...args: infer Args) => infer R\n ? (...args: Args) => R\n : never;\n};\n\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n const currentVnode = current.vnode as VnodeWithDom;\n if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n return;\n }\n\n let hasParent = false;\n let parent = currentVnode.dom.parentElement as DomElement;\n while (parent) {\n if (domWithVnodesToUpdate.has(parent)) {\n hasParent = true;\n break;\n }\n parent = parent.parentElement as DomElement;\n }\n\n if (!hasParent) {\n const dom = currentVnode.dom;\n const subscription = () => {\n updateVnode(dom.vnode);\n if (!dom.parentElement) {\n subscribers.delete(subscription);\n domWithVnodesToUpdate.delete(dom);\n }\n };\n subscribers.add(subscription);\n domWithVnodesToUpdate.add(dom);\n }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType | (() => StateType) | null,\n pulses: PulsesType,\n immutable = false\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const boundPulses: Record<string, Function> = {};\n for (const key in pulses) {\n if (typeof pulses[key] !== \"function\") {\n throw new Error(`Pulse '${key}' must be a function`);\n }\n if (key === \"state\") {\n throw new Error(`A pulse cannot be named 'state'`);\n }\n boundPulses[key] = getPulseMethod(key);\n }\n\n const localState: StateType =\n (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n function isMutable() {\n if (immutable) {\n throw new Error(\"You need to call a pulse to modify the state\");\n }\n }\n\n let currentState: StateType | null = null;\n let pulseCallCount = 0;\n\n const proxyState = new Proxy(localState, {\n get: (state, prop: string) => {\n if (currentState) {\n return currentState[prop];\n }\n\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n return state[prop];\n },\n set: (state, prop: string, value: any) => {\n isMutable();\n Reflect.set(state, prop, value);\n return true;\n },\n deleteProperty: (state, prop: string) => {\n isMutable();\n Reflect.deleteProperty(state, prop);\n return true;\n }\n });\n\n function syncState(newState: StateType) {\n for (const key in newState) {\n localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n }\n for (const key in localState) {\n if (!(key in newState)) {\n Reflect.deleteProperty(localState, key);\n }\n }\n }\n\n let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n function debouncedUpdate() {\n if (debounceTimeout) {\n clearTimeout(debounceTimeout);\n }\n debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n }\n\n function setState(newState: StateType, flush = false) {\n pulseCallCount--;\n if (!hasChanged(localState, newState)) {\n return;\n }\n if (pulseCallCount > 0 && !flush) {\n return;\n }\n syncState(newState);\n currentState = null;\n debouncedUpdate();\n }\n\n function getPulseMethod(key: string) {\n function pulseMethod(this: PulseContext, ...args: any[]) {\n pulseCallCount++;\n if (currentState === null) {\n currentState = deepCloneUnfreeze(localState);\n }\n\n this.$flush = async () => {\n if (currentState) {\n setState(currentState, true);\n currentState = deepCloneUnfreeze(localState);\n }\n await new Promise((resolve) => setTimeout(resolve, 0));\n };\n\n const emptyFlush = async () => {};\n\n try {\n const pulseResult = pulses[key].apply(this, [currentState, ...args]);\n if (pulseResult instanceof Promise) {\n return pulseResult\n .then((resolvedValue) => {\n setState(currentState as StateType);\n this.$flush = emptyFlush;\n return resolvedValue;\n })\n .catch((error) => {\n console.error(`Error in pulse '${key}':`, error);\n this.$flush = emptyFlush;\n throw error;\n });\n } else {\n setState(currentState);\n this.$flush = emptyFlush;\n return pulseResult;\n }\n } catch (error) {\n console.error(`Error in pulse '${key}':`, error);\n Reflect.set(this, \"$flush\", emptyFlush);\n throw error;\n }\n }\n\n return pulseMethod;\n }\n\n syncState(localState);\n\n const listeners: Record<string, Function[]> = {};\n const trigger = (event: string, ...args: any[]) => {\n if (listeners[event]) {\n listeners[event].forEach((callback) => callback(...args));\n }\n };\n\n const pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\n }\n if (prop === \"on\") {\n return (event: string, callback: Function) => {\n if (!listeners[event]) {\n listeners[event] = [];\n }\n listeners[event].push(callback);\n };\n }\n if (prop === \"off\") {\n return (event: string, callback: Function) => {\n if (listeners[event]) {\n listeners[event] = listeners[event].filter((cb) => cb !== callback);\n }\n };\n }\n if (!(prop in pulses)) {\n throw new Error(`Pulse '${prop}' does not exist`);\n }\n const pulseMethod = pulses[prop];\n\n return (...args: any[]) => {\n const result = pulseMethod.apply(pulseMethod, args);\n if (result instanceof Promise) {\n return result.then((r) => {\n trigger(\"pulse\", prop, args);\n return r;\n });\n }\n trigger(\"pulse\", prop, args);\n return result;\n };\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n) {\n return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n) {\n console.warn(\n \"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render.\"\n );\n return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n const runEffect = () => {\n try {\n effectStack.push(runEffect);\n effect();\n } finally {\n effectStack.pop();\n }\n };\n runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n let value = initialValue;\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const runSubscribers = () => {\n subscribers.forEach((subscriber) => subscriber());\n };\n\n const read = (): T => {\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n return value;\n };\n\n const write = (newValue: T | ((current: T) => T)): void => {\n const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n if (!hasChanged(value, resolvedValue)) {\n return;\n }\n value = resolvedValue;\n runSubscribers();\n };\n\n return [read, write, runSubscribers];\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["pulses"]
|
|
7
7
|
}
|
package/dist/pulses/index.min.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
(()=>{"use strict";var e=Object.defineProperty,t=Object.getOwnPropertyDescriptor,n=Object.getOwnPropertyNames,r=Object.prototype.hasOwnProperty,o={};((t,n)=>{for(var r in n)e(t,r,{get:n[r],enumerable:!0})})(o,{createEffect:()=>d,createMutableStore:()=>p,createPulse:()=>y,createPulseStore:()=>h});var s,
|
|
1
|
+
(()=>{"use strict";var e=Object.defineProperty,t=Object.getOwnPropertyDescriptor,n=Object.getOwnPropertyNames,r=Object.prototype.hasOwnProperty,o={};((t,n)=>{for(var r in n)e(t,r,{get:n[r],enumerable:!0})})(o,{createEffect:()=>d,createMutableStore:()=>p,createPulse:()=>y,createPulseStore:()=>h});var s,u=(s=o,((o,s,u,l)=>{if(s&&"object"==typeof s||"function"==typeof s)for(let a of n(s))r.call(o,a)||a===u||e(o,a,{get:()=>s[a],enumerable:!(l=t(s,a))||l.enumerable});return o})(e({},"__esModule",{value:!0}),s)),l=require("valyrian.js"),a=require("valyrian.js/utils"),i=[];function c(e,t){const n=l.current.vnode;if(!n||t.has(n.dom))return;let r=!1,o=n.dom.parentElement;for(;o;){if(t.has(o)){r=!0;break}o=o.parentElement}if(!r){const r=n.dom,o=()=>{(0,l.updateVnode)(r.vnode),r.parentElement||(e.delete(o),t.delete(r))};e.add(o),t.add(r)}}function f(e,t,n=!1){const r=new Set,o=new WeakSet,s={};for(const e in t){if("function"!=typeof t[e])throw new Error(`Pulse '${e}' must be a function`);if("state"===e)throw new Error("A pulse cannot be named 'state'");s[e]=m(e)}const u=("function"==typeof e?e():e)||{};function l(){if(n)throw new Error("You need to call a pulse to modify the state")}let f=null,h=0;const p=new Proxy(u,{get:(e,t)=>{if(f)return f[t];const n=i[i.length-1];return n&&!r.has(n)&&r.add(n),c(r,o),e[t]},set:(e,t,n)=>(l(),Reflect.set(e,t,n),!0),deleteProperty:(e,t)=>(l(),Reflect.deleteProperty(e,t),!0)});function d(e){for(const t in e)u[t]=n?(0,a.deepFreeze)(e[t]):e[t];for(const t in u)t in e||Reflect.deleteProperty(u,t)}let y=null;function w(e,t=!1){h--,(0,a.hasChanged)(u,e)&&(h>0&&!t||(d(e),f=null,y&&clearTimeout(y),y=setTimeout(()=>r.forEach(e=>e()),0)))}function m(e){return function(...n){h++,null===f&&(f=(0,a.deepCloneUnfreeze)(u)),this.$flush=async()=>{f&&(w(f,!0),f=(0,a.deepCloneUnfreeze)(u)),await new Promise(e=>setTimeout(e,0))};const r=async()=>{};try{const o=t[e].apply(this,[f,...n]);return o instanceof Promise?o.then(e=>(w(f),this.$flush=r,e)).catch(t=>{throw console.error(`Error in pulse '${e}':`,t),this.$flush=r,t}):(w(f),this.$flush=r,o)}catch(t){throw console.error(`Error in pulse '${e}':`,t),Reflect.set(this,"$flush",r),t}}}d(u);const P={},g=(e,...t)=>{P[e]&&P[e].forEach(e=>e(...t))};return new Proxy(s,{get:(e,t)=>{if("state"===t)return p;if("on"===t)return(e,t)=>{P[e]||(P[e]=[]),P[e].push(t)};if("off"===t)return(e,t)=>{P[e]&&(P[e]=P[e].filter(e=>e!==t))};if(!(t in e))throw new Error(`Pulse '${t}' does not exist`);const n=e[t];return(...e)=>{const r=n.apply(n,e);return r instanceof Promise?r.then(n=>(g("pulse",t,e),n)):(g("pulse",t,e),r)}}})}function h(e,t){return f(e,t,!0)}function p(e,t){return console.warn("Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render."),f(e,t,!1)}function d(e){const t=()=>{try{i.push(t),e()}finally{i.pop()}};t()}function y(e){let t=e;const n=new Set,r=new WeakSet,o=()=>{n.forEach(e=>e())};return[()=>{const e=i[i.length-1];return e&&!n.has(e)&&n.add(e),c(n,r),t},e=>{const n="function"==typeof e?e(t):e;(0,a.hasChanged)(t,n)&&(t=n,o())},o]}"undefined"!=typeof module?module.exports=u:self.ValyrianPulses=u})();//# sourceMappingURL=index.min.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["index_exports","__export","createEffect","createMutableStore","createPulse","createPulseStore","__EXPORTS__","import_valyrian","require","import_utils","effectStack","registerDomSubscription","subscribers","domWithVnodesToUpdate","currentVnode","current","vnode","has","dom","hasParent","parent","parentElement","subscription","updateVnode","delete","add","createStore","initialState","pulses","immutable","Set","WeakSet","boundPulses","key","Error","getPulseMethod","localState","isMutable","currentState","pulseCallCount","proxyState","Proxy","get","state","prop","currentEffect","length","set","value","Reflect","deleteProperty","syncState","newState","deepFreeze","debounceTimeout","setState","flush","hasChanged","clearTimeout","setTimeout","forEach","subscriber","args","deepCloneUnfreeze","this","async","Promise","resolve","emptyFlush","pulseResult","apply","then","resolvedValue","catch","error","console","listeners","trigger","event","callback","pulsesProxy","push","filter","cb","pulseMethod","result","r","warn","effect","runEffect","pop","initialValue","runSubscribers","newValue"],"sources":["../../lib/pulses/index.ts"],"sourcesContent":["/* eslint-disable no-console */\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type Pulse<StateType, TReturn = unknown> = (state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;\n\nexport type Pulses<StateType> = Record<string, Pulse<StateType, any> & { $flush?: () => Promise<void> }>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType extends Pulses<any>> = {\n  [K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R\n    ? (...args: Args) => R\n    : never;\n};\n\n/** Función auxiliar para registrar la suscripción al nodo DOM.\n *  Retorna true si se agregó la suscripción, o false si se encontró un nodo padre ya suscrito.\n */\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n  const currentVnode = current.vnode as VnodeWithDom;\n  if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n    return;\n  }\n\n  let hasParent = false;\n  let parent = currentVnode.dom.parentElement as DomElement;\n  while (parent) {\n    if (domWithVnodesToUpdate.has(parent)) {\n      hasParent = true;\n      break;\n    }\n    parent = parent.parentElement as DomElement;\n  }\n\n  // Si no hay nodo padre registrado, se crea la suscripción.\n  if (!hasParent) {\n    const dom = currentVnode.dom;\n    const subscription = () => {\n      updateVnode(dom.vnode);\n      if (!dom.parentElement) {\n        subscribers.delete(subscription);\n        domWithVnodesToUpdate.delete(dom);\n      }\n    };\n    subscribers.add(subscription);\n    domWithVnodesToUpdate.add(dom);\n  }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n  initialState: StateType | (() => StateType) | null,\n  pulses: PulsesType,\n  immutable = false\n): StorePulses<PulsesType> & {\n  state: ProxyState<StateType>;\n  on: (event: string, callback: Function) => void;\n  off: (event: string, callback: Function) => void;\n} {\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const boundPulses: Pulses<StateType> = {};\n  for (const key in pulses) {\n    if (typeof pulses[key] !== \"function\") {\n      throw new Error(`Pulse '${key}' must be a function`);\n    }\n    if (key === \"state\") {\n      throw new Error(`A pulse cannot be named 'state'`);\n    }\n    boundPulses[key] = getPulseMethod(key);\n  }\n\n  const localState: StateType =\n    (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n  function isMutable() {\n    if (immutable) {\n      throw new Error(\"You need to call a pulse to modify the state\");\n    }\n  }\n\n  let currentState: StateType | null = null;\n  let pulseCallCount = 0;\n\n  const proxyState = new Proxy(localState, {\n    get: (state, prop: string) => {\n      // If we are in a pulse, we return the value of the cloned state.\n      if (currentState) {\n        return currentState[prop];\n      }\n\n      const currentEffect = effectStack[effectStack.length - 1];\n      if (currentEffect && !subscribers.has(currentEffect)) {\n        subscribers.add(currentEffect);\n      }\n\n      registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n      return state[prop];\n    },\n    set: (state, prop: string, value: any) => {\n      isMutable();\n      Reflect.set(state, prop, value);\n      return true;\n    },\n    deleteProperty: (state, prop: string) => {\n      isMutable();\n      Reflect.deleteProperty(state, prop);\n      return true;\n    }\n  });\n\n  function syncState(newState: StateType) {\n    for (const key in newState) {\n      localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n    }\n    for (const key in localState) {\n      if (!(key in newState)) {\n        Reflect.deleteProperty(localState, key);\n      }\n    }\n  }\n\n  let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n  function debouncedUpdate() {\n    if (debounceTimeout) {\n      clearTimeout(debounceTimeout);\n    }\n    debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n  }\n\n  function setState(newState: StateType, flush = false) {\n    pulseCallCount--;\n    if (!hasChanged(localState, newState)) {\n      return;\n    }\n    if (pulseCallCount > 0 && !flush) {\n      return;\n    }\n    syncState(newState);\n    currentState = null;\n    debouncedUpdate();\n  }\n\n  function getPulseMethod(key: string) {\n    function pulseMethod(this: Pulse<StateType, any> & { $flush: () => Promise<void> }, ...args: any[]) {\n      pulseCallCount++;\n      if (currentState === null) {\n        currentState = deepCloneUnfreeze(localState);\n      }\n\n      const $flush = async () => {\n        setState(currentState as StateType, true);\n        currentState = deepCloneUnfreeze(localState);\n        await new Promise((resolve) => setTimeout(resolve, 0));\n      };\n\n      Reflect.set(this, \"$flush\", $flush);\n      const emptyFlush = async () => {};\n\n      try {\n        const pulseResult = pulses[key].apply(this, [currentState, ...args] as any);\n        if (pulseResult instanceof Promise) {\n          return pulseResult\n            .then((resolvedValue) => {\n              setState(currentState as StateType);\n              Reflect.set(this, \"$flush\", emptyFlush);\n              return resolvedValue;\n            })\n            .catch((error) => {\n              console.error(`Error in pulse '${key}':`, error);\n              Reflect.set(this, \"$flush\", emptyFlush);\n              throw error;\n            });\n        } else {\n          setState(currentState);\n          Reflect.set(this, \"$flush\", emptyFlush);\n          return pulseResult;\n        }\n      } catch (error) {\n        console.error(`Error in pulse '${key}':`, error);\n        Reflect.set(this, \"$flush\", emptyFlush);\n        throw error;\n      }\n    }\n\n    return pulseMethod;\n  }\n\n  syncState(localState);\n\n  const listeners: Record<string, Function[]> = {};\n  const trigger = (event: string, ...args: any[]) => {\n    if (listeners[event]) {\n      listeners[event].forEach((callback) => callback(...args));\n    }\n  };\n\n  const pulsesProxy = new Proxy(boundPulses, {\n    get: (pulses, prop: string) => {\n      if (prop === \"state\") {\n        return proxyState;\n      }\n      if (prop === \"on\") {\n        return (event: string, callback: Function) => {\n          if (!listeners[event]) {\n            listeners[event] = [];\n          }\n          listeners[event].push(callback);\n        };\n      }\n      if (prop === \"off\") {\n        return (event: string, callback: Function) => {\n          if (listeners[event]) {\n            listeners[event] = listeners[event].filter((cb) => cb !== callback);\n          }\n        };\n      }\n      if (!(prop in pulses)) {\n        throw new Error(`Pulse '${prop}' does not exist`);\n      }\n      const pulseMethod = pulses[prop];\n\n      if (typeof pulseMethod === \"function\") {\n        return (...args: any[]) => {\n          const result = pulseMethod.apply(pulsesProxy, args as any);\n          if (result instanceof Promise) {\n            return result.then((r) => {\n              trigger(\"pulse\", prop, args);\n              return r;\n            });\n          }\n          trigger(\"pulse\", prop, args);\n          return result;\n        };\n      }\n\n      return pulseMethod;\n    }\n  });\n\n  return pulsesProxy as StorePulses<PulsesType> & {\n    state: ProxyState<StateType>;\n    on: (event: string, callback: Function) => void;\n    off: (event: string, callback: Function) => void;\n  };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n  initialState: StateType,\n  pulses: PulsesType\n): StorePulses<PulsesType> & {\n  state: ProxyState<StateType>;\n  on: (event: string, callback: Function) => void;\n  off: (event: string, callback: Function) => void;\n} {\n  return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n  initialState: StateType,\n  pulses: PulsesType\n): StorePulses<PulsesType> & {\n  state: ProxyState<StateType>;\n  on: (event: string, callback: Function) => void;\n  off: (event: string, callback: Function) => void;\n} {\n  console.warn(\n    \"Warning: You are working with a mutable state. This can lead to unpredictable behavior. All state changes made outside of a pulse will not trigger a re-render.\"\n  );\n  return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n  const runEffect = () => {\n    try {\n      effectStack.push(runEffect);\n      effect();\n    } finally {\n      effectStack.pop();\n    }\n  };\n  runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n  let value = initialValue;\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const runSubscribers = () => {\n    subscribers.forEach((subscriber) => subscriber());\n  };\n\n  const read = (): T => {\n    const currentEffect = effectStack[effectStack.length - 1];\n    if (currentEffect && !subscribers.has(currentEffect)) {\n      subscribers.add(currentEffect);\n    }\n    registerDomSubscription(subscribers, domWithVnodesToUpdate);\n    return value;\n  };\n\n  const write = (newValue: T | ((current: T) => T)): void => {\n    const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n    if (!hasChanged(value, resolvedValue)) {\n      return;\n    }\n    value = resolvedValue;\n    runSubscribers();\n  };\n\n  return [read, write, runSubscribers];\n}\n"],"mappings":"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"}
|
|
1
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["index_exports","__export","createEffect","createMutableStore","createPulse","createPulseStore","__EXPORTS__","import_valyrian","require","import_utils","effectStack","registerDomSubscription","subscribers","domWithVnodesToUpdate","currentVnode","current","vnode","has","dom","hasParent","parent","parentElement","subscription","updateVnode","delete","add","createStore","initialState","pulses","immutable","Set","WeakSet","boundPulses","key","Error","getPulseMethod","localState","isMutable","currentState","pulseCallCount","proxyState","Proxy","get","state","prop","currentEffect","length","set","value","Reflect","deleteProperty","syncState","newState","deepFreeze","debounceTimeout","setState","flush","hasChanged","clearTimeout","setTimeout","forEach","subscriber","args","deepCloneUnfreeze","this","$flush","async","Promise","resolve","emptyFlush","pulseResult","apply","then","resolvedValue","catch","error","console","listeners","trigger","event","callback","push","filter","cb","pulseMethod","result","r","warn","effect","runEffect","pop","initialValue","runSubscribers","newValue"],"sources":["../../lib/pulses/index.ts"],"sourcesContent":["/* eslint-disable no-console */\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type PulseContext = {\n  $flush: () => Promise<void>;\n};\n\nexport type Pulse<StateType, TReturn = unknown> = (\n  this: PulseContext,\n  state: StateType,\n  ...args: any[]\n) => TReturn | Promise<TReturn>;\n\nexport type Pulses<StateType> = Record<string, Pulse<StateType, any>>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType extends Pulses<any>> = {\n  [K in keyof PulsesType]: PulsesType[K] extends (this: any, state: any, ...args: infer Args) => infer R\n    ? (...args: Args) => R\n    : never;\n};\n\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n  const currentVnode = current.vnode as VnodeWithDom;\n  if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n    return;\n  }\n\n  let hasParent = false;\n  let parent = currentVnode.dom.parentElement as DomElement;\n  while (parent) {\n    if (domWithVnodesToUpdate.has(parent)) {\n      hasParent = true;\n      break;\n    }\n    parent = parent.parentElement as DomElement;\n  }\n\n  if (!hasParent) {\n    const dom = currentVnode.dom;\n    const subscription = () => {\n      updateVnode(dom.vnode);\n      if (!dom.parentElement) {\n        subscribers.delete(subscription);\n        domWithVnodesToUpdate.delete(dom);\n      }\n    };\n    subscribers.add(subscription);\n    domWithVnodesToUpdate.add(dom);\n  }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n  initialState: StateType | (() => StateType) | null,\n  pulses: PulsesType,\n  immutable = false\n): StorePulses<PulsesType> & {\n  state: ProxyState<StateType>;\n  on: (event: string, callback: Function) => void;\n  off: (event: string, callback: Function) => void;\n} {\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const boundPulses: Record<string, Function> = {};\n  for (const key in pulses) {\n    if (typeof pulses[key] !== \"function\") {\n      throw new Error(`Pulse '${key}' must be a function`);\n    }\n    if (key === \"state\") {\n      throw new Error(`A pulse cannot be named 'state'`);\n    }\n    boundPulses[key] = getPulseMethod(key);\n  }\n\n  const localState: StateType =\n    (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n  function isMutable() {\n    if (immutable) {\n      throw new Error(\"You need to call a pulse to modify the state\");\n    }\n  }\n\n  let currentState: StateType | null = null;\n  let pulseCallCount = 0;\n\n  const proxyState = new Proxy(localState, {\n    get: (state, prop: string) => {\n      if (currentState) {\n        return currentState[prop];\n      }\n\n      const currentEffect = effectStack[effectStack.length - 1];\n      if (currentEffect && !subscribers.has(currentEffect)) {\n        subscribers.add(currentEffect);\n      }\n\n      registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n      return state[prop];\n    },\n    set: (state, prop: string, value: any) => {\n      isMutable();\n      Reflect.set(state, prop, value);\n      return true;\n    },\n    deleteProperty: (state, prop: string) => {\n      isMutable();\n      Reflect.deleteProperty(state, prop);\n      return true;\n    }\n  });\n\n  function syncState(newState: StateType) {\n    for (const key in newState) {\n      localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n    }\n    for (const key in localState) {\n      if (!(key in newState)) {\n        Reflect.deleteProperty(localState, key);\n      }\n    }\n  }\n\n  let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n  function debouncedUpdate() {\n    if (debounceTimeout) {\n      clearTimeout(debounceTimeout);\n    }\n    debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n  }\n\n  function setState(newState: StateType, flush = false) {\n    pulseCallCount--;\n    if (!hasChanged(localState, newState)) {\n      return;\n    }\n    if (pulseCallCount > 0 && !flush) {\n      return;\n    }\n    syncState(newState);\n    currentState = null;\n    debouncedUpdate();\n  }\n\n  function getPulseMethod(key: string) {\n    function pulseMethod(this: PulseContext, ...args: any[]) {\n      pulseCallCount++;\n      if (currentState === null) {\n        currentState = deepCloneUnfreeze(localState);\n      }\n\n      this.$flush = async () => {\n        if (currentState) {\n          setState(currentState, true);\n          currentState = deepCloneUnfreeze(localState);\n        }\n        await new Promise((resolve) => setTimeout(resolve, 0));\n      };\n\n      const emptyFlush = async () => {};\n\n      try {\n        const pulseResult = pulses[key].apply(this, [currentState, ...args]);\n        if (pulseResult instanceof Promise) {\n          return pulseResult\n            .then((resolvedValue) => {\n              setState(currentState as StateType);\n              this.$flush = emptyFlush;\n              return resolvedValue;\n            })\n            .catch((error) => {\n              console.error(`Error in pulse '${key}':`, error);\n              this.$flush = emptyFlush;\n              throw error;\n            });\n        } else {\n          setState(currentState);\n          this.$flush = emptyFlush;\n          return pulseResult;\n        }\n      } catch (error) {\n        console.error(`Error in pulse '${key}':`, error);\n        Reflect.set(this, \"$flush\", emptyFlush);\n        throw error;\n      }\n    }\n\n    return pulseMethod;\n  }\n\n  syncState(localState);\n\n  const listeners: Record<string, Function[]> = {};\n  const trigger = (event: string, ...args: any[]) => {\n    if (listeners[event]) {\n      listeners[event].forEach((callback) => callback(...args));\n    }\n  };\n\n  const pulsesProxy = new Proxy(boundPulses, {\n    get: (pulses, prop: string) => {\n      if (prop === \"state\") {\n        return proxyState;\n      }\n      if (prop === \"on\") {\n        return (event: string, callback: Function) => {\n          if (!listeners[event]) {\n            listeners[event] = [];\n          }\n          listeners[event].push(callback);\n        };\n      }\n      if (prop === \"off\") {\n        return (event: string, callback: Function) => {\n          if (listeners[event]) {\n            listeners[event] = listeners[event].filter((cb) => cb !== callback);\n          }\n        };\n      }\n      if (!(prop in pulses)) {\n        throw new Error(`Pulse '${prop}' does not exist`);\n      }\n      const pulseMethod = pulses[prop];\n\n      return (...args: any[]) => {\n        const result = pulseMethod.apply(pulseMethod, args);\n        if (result instanceof Promise) {\n          return result.then((r) => {\n            trigger(\"pulse\", prop, args);\n            return r;\n          });\n        }\n        trigger(\"pulse\", prop, args);\n        return result;\n      };\n    }\n  });\n\n  return pulsesProxy as StorePulses<PulsesType> & {\n    state: ProxyState<StateType>;\n    on: (event: string, callback: Function) => void;\n    off: (event: string, callback: Function) => void;\n  };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n  initialState: StateType,\n  pulses: PulsesType\n) {\n  return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n  initialState: StateType,\n  pulses: PulsesType\n) {\n  console.warn(\n    \"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render.\"\n  );\n  return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n  const runEffect = () => {\n    try {\n      effectStack.push(runEffect);\n      effect();\n    } finally {\n      effectStack.pop();\n    }\n  };\n  runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n  let value = initialValue;\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const runSubscribers = () => {\n    subscribers.forEach((subscriber) => subscriber());\n  };\n\n  const read = (): T => {\n    const currentEffect = effectStack[effectStack.length - 1];\n    if (currentEffect && !subscribers.has(currentEffect)) {\n      subscribers.add(currentEffect);\n    }\n    registerDomSubscription(subscribers, domWithVnodesToUpdate);\n    return value;\n  };\n\n  const write = (newValue: T | ((current: T) => T)): void => {\n    const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n    if (!hasChanged(value, resolvedValue)) {\n      return;\n    }\n    value = resolvedValue;\n    runSubscribers();\n  };\n\n  return [read, write, runSubscribers];\n}\n"],"mappings":"gJAAAA,EAAA,G,yDAAAC,CAAAD,EAAA,CAAAE,aAAA,IAAAA,EAAAC,mBAAA,IAAAA,EAAAC,YAAA,IAAAA,EAAAC,iBAAA,IAAAA,IAAA,I,EAAAC,G,EAAAN,E,0MACAO,EAA+DC,QAAA,eAC/DC,EAA0DD,QAAA,qBAkBpDE,EAA0B,GAQhC,SAASC,EAAwBC,EAA4BC,GAC3D,MAAMC,EAAeP,EAAAQ,QAAQC,MAC7B,IAAKF,GAAgBD,EAAsBI,IAAIH,EAAaI,KAC1D,OAGF,IAAIC,GAAY,EACZC,EAASN,EAAaI,IAAIG,cAC9B,KAAOD,GAAQ,CACb,GAAIP,EAAsBI,IAAIG,GAAS,CACrCD,GAAY,EACZ,KACF,CACAC,EAASA,EAAOC,aAClB,CAEA,IAAKF,EAAW,CACd,MAAMD,EAAMJ,EAAaI,IACnBI,EAAe,MACnB,EAAAf,EAAAgB,aAAYL,EAAIF,OACXE,EAAIG,gBACPT,EAAYY,OAAOF,GACnBT,EAAsBW,OAAON,KAGjCN,EAAYa,IAAIH,GAChBT,EAAsBY,IAAIP,EAC5B,CACF,CAEA,SAASQ,EACPC,EACAC,EACAC,GAAY,GAMZ,MAAMjB,EAAc,IAAIkB,IAClBjB,EAAwB,IAAIkB,QAE5BC,EAAwC,CAAC,EAC/C,UAAWC,KAAOL,EAAQ,CACxB,GAA2B,mBAAhBA,EAAOK,GAChB,MAAM,IAAIC,MAAM,UAAUD,yBAE5B,GAAY,UAARA,EACF,MAAM,IAAIC,MAAM,mCAElBF,EAAYC,GAAOE,EAAeF,EACpC,CAEA,MAAMG,GACqB,mBAAjBT,EAA8BA,IAAiBA,IAAkB,CAAC,EAE5E,SAASU,IACP,GAAIR,EACF,MAAM,IAAIK,MAAM,+CAEpB,CAEA,IAAII,EAAiC,KACjCC,EAAiB,EAErB,MAAMC,EAAa,IAAIC,MAAML,EAAY,CACvCM,IAAK,CAACC,EAAOC,KACX,GAAIN,EACF,OAAOA,EAAaM,GAGtB,MAAMC,EAAgBnC,EAAYA,EAAYoC,OAAS,GAOvD,OANID,IAAkBjC,EAAYK,IAAI4B,IACpCjC,EAAYa,IAAIoB,GAGlBlC,EAAwBC,EAAaC,GAE9B8B,EAAMC,IAEfG,IAAK,CAACJ,EAAOC,EAAcI,KACzBX,IACAY,QAAQF,IAAIJ,EAAOC,EAAMI,IAClB,GAETE,eAAgB,CAACP,EAAOC,KACtBP,IACAY,QAAQC,eAAeP,EAAOC,IACvB,KAIX,SAASO,EAAUC,GACjB,UAAWnB,KAAOmB,EAChBhB,EAAWH,GAAOJ,GAAA,EAAYpB,EAAA4C,YAAWD,EAASnB,IAAQmB,EAASnB,GAErE,UAAWA,KAAOG,EACVH,KAAOmB,GACXH,QAAQC,eAAed,EAAYH,EAGzC,CAEA,IAAIqB,EAAwD,KAQ5D,SAASC,EAASH,EAAqBI,GAAQ,GAC7CjB,KACI,EAAC9B,EAAAgD,YAAWrB,EAAYgB,KAGxBb,EAAiB,IAAMiB,IAG3BL,EAAUC,GACVd,EAAe,KAfXgB,GACFI,aAAaJ,GAEfA,EAAkBK,WAAW,IAAM/C,EAAYgD,QAASC,GAAeA,KAAe,IAcxF,CAEA,SAAS1B,EAAeF,GA2CtB,OA1CA,YAA4C6B,GAC1CvB,IACqB,OAAjBD,IACFA,GAAA,EAAe7B,EAAAsD,mBAAkB3B,IAGnC4B,KAAKC,OAASC,UACR5B,IACFiB,EAASjB,GAAc,GACvBA,GAAA,EAAe7B,EAAAsD,mBAAkB3B,UAE7B,IAAI+B,QAASC,GAAYT,WAAWS,EAAS,KAGrD,MAAMC,EAAaH,YAEnB,IACE,MAAMI,EAAc1C,EAAOK,GAAKsC,MAAMP,KAAM,CAAC1B,KAAiBwB,IAC9D,OAAIQ,aAAuBH,QAClBG,EACJE,KAAMC,IACLlB,EAASjB,GACT0B,KAAKC,OAASI,EACPI,IAERC,MAAOC,IAGN,MAFAC,QAAQD,MAAM,mBAAmB1C,MAAS0C,GAC1CX,KAAKC,OAASI,EACRM,KAGVpB,EAASjB,GACT0B,KAAKC,OAASI,EACPC,EAEX,OAASK,GAGP,MAFAC,QAAQD,MAAM,mBAAmB1C,MAAS0C,GAC1C1B,QAAQF,IAAIiB,KAAM,SAAUK,GACtBM,CACR,CACF,CAGF,CAEAxB,EAAUf,GAEV,MAAMyC,EAAwC,CAAC,EACzCC,EAAU,CAACC,KAAkBjB,KAC7Be,EAAUE,IACZF,EAAUE,GAAOnB,QAASoB,GAAaA,KAAYlB,KA2CvD,OAvCoB,IAAIrB,MAAMT,EAAa,CACzCU,IAAK,CAACd,EAAQgB,KACZ,GAAa,UAATA,EACF,OAAOJ,EAET,GAAa,OAATI,EACF,MAAO,CAACmC,EAAeC,KAChBH,EAAUE,KACbF,EAAUE,GAAS,IAErBF,EAAUE,GAAOE,KAAKD,IAG1B,GAAa,QAATpC,EACF,MAAO,CAACmC,EAAeC,KACjBH,EAAUE,KACZF,EAAUE,GAASF,EAAUE,GAAOG,OAAQC,GAAOA,IAAOH,KAIhE,KAAMpC,KAAQhB,GACZ,MAAM,IAAIM,MAAM,UAAUU,qBAE5B,MAAMwC,EAAcxD,EAAOgB,GAE3B,MAAO,IAAIkB,KACT,MAAMuB,EAASD,EAAYb,MAAMa,EAAatB,GAC9C,OAAIuB,aAAkBlB,QACbkB,EAAOb,KAAMc,IAClBR,EAAQ,QAASlC,EAAMkB,GAChBwB,KAGXR,EAAQ,QAASlC,EAAMkB,GAChBuB,MAUf,CAEO,SAAShF,EACdsB,EACAC,GAEA,OAAOF,EAAYC,EAAcC,GAAQ,EAC3C,CAEO,SAASzB,EACdwB,EACAC,GAKA,OAHAgD,QAAQW,KACN,0HAEK7D,EAAYC,EAAcC,GAAQ,EAC3C,CAEO,SAAS1B,EAAasF,GAC3B,MAAMC,EAAY,KAChB,IACE/E,EAAYuE,KAAKQ,GACjBD,GACF,SACE9E,EAAYgF,KACd,GAEFD,GACF,CAEO,SAASrF,EAAeuF,GAC7B,IAAI3C,EAAQ2C,EACZ,MAAM/E,EAAc,IAAIkB,IAClBjB,EAAwB,IAAIkB,QAE5B6D,EAAiB,KACrBhF,EAAYgD,QAASC,GAAeA,MAqBtC,MAAO,CAlBM,KACX,MAAMhB,EAAgBnC,EAAYA,EAAYoC,OAAS,GAKvD,OAJID,IAAkBjC,EAAYK,IAAI4B,IACpCjC,EAAYa,IAAIoB,GAElBlC,EAAwBC,EAAaC,GAC9BmC,GAGM6C,IACb,MAAMpB,EAAoC,mBAAboB,EAA2BA,EAA+B7C,GAAS6C,GAC5F,EAACpF,EAAAgD,YAAWT,EAAOyB,KAGvBzB,EAAQyB,EACRmB,MAGmBA,EACvB,C"}
|
package/dist/pulses/index.mjs
CHANGED
|
@@ -108,12 +108,13 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
108
108
|
if (currentState === null) {
|
|
109
109
|
currentState = deepCloneUnfreeze(localState);
|
|
110
110
|
}
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
111
|
+
this.$flush = async () => {
|
|
112
|
+
if (currentState) {
|
|
113
|
+
setState(currentState, true);
|
|
114
|
+
currentState = deepCloneUnfreeze(localState);
|
|
115
|
+
}
|
|
114
116
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
115
117
|
};
|
|
116
|
-
Reflect.set(this, "$flush", $flush);
|
|
117
118
|
const emptyFlush = async () => {
|
|
118
119
|
};
|
|
119
120
|
try {
|
|
@@ -121,16 +122,16 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
121
122
|
if (pulseResult instanceof Promise) {
|
|
122
123
|
return pulseResult.then((resolvedValue) => {
|
|
123
124
|
setState(currentState);
|
|
124
|
-
|
|
125
|
+
this.$flush = emptyFlush;
|
|
125
126
|
return resolvedValue;
|
|
126
127
|
}).catch((error) => {
|
|
127
128
|
console.error(`Error in pulse '${key}':`, error);
|
|
128
|
-
|
|
129
|
+
this.$flush = emptyFlush;
|
|
129
130
|
throw error;
|
|
130
131
|
});
|
|
131
132
|
} else {
|
|
132
133
|
setState(currentState);
|
|
133
|
-
|
|
134
|
+
this.$flush = emptyFlush;
|
|
134
135
|
return pulseResult;
|
|
135
136
|
}
|
|
136
137
|
} catch (error) {
|
|
@@ -172,20 +173,17 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
172
173
|
throw new Error(`Pulse '${prop}' does not exist`);
|
|
173
174
|
}
|
|
174
175
|
const pulseMethod = pulses2[prop];
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
};
|
|
187
|
-
}
|
|
188
|
-
return pulseMethod;
|
|
176
|
+
return (...args) => {
|
|
177
|
+
const result = pulseMethod.apply(pulseMethod, args);
|
|
178
|
+
if (result instanceof Promise) {
|
|
179
|
+
return result.then((r) => {
|
|
180
|
+
trigger("pulse", prop, args);
|
|
181
|
+
return r;
|
|
182
|
+
});
|
|
183
|
+
}
|
|
184
|
+
trigger("pulse", prop, args);
|
|
185
|
+
return result;
|
|
186
|
+
};
|
|
189
187
|
}
|
|
190
188
|
});
|
|
191
189
|
return pulsesProxy;
|
|
@@ -195,7 +193,7 @@ function createPulseStore(initialState, pulses) {
|
|
|
195
193
|
}
|
|
196
194
|
function createMutableStore(initialState, pulses) {
|
|
197
195
|
console.warn(
|
|
198
|
-
"Warning: You are working with a mutable state.
|
|
196
|
+
"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render."
|
|
199
197
|
);
|
|
200
198
|
return createStore(initialState, pulses, false);
|
|
201
199
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../lib/pulses/index.ts"],
|
|
4
|
-
"sourcesContent": ["/* eslint-disable no-console */\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type Pulse<StateType, TReturn = unknown> = (state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;\n\nexport type Pulses<StateType> = Record<string, Pulse<StateType, any> & { $flush?: () => Promise<void> }>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType extends Pulses<any>> = {\n [K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R\n ? (...args: Args) => R\n : never;\n};\n\n/** Funci\u00F3n auxiliar para registrar la suscripci\u00F3n al nodo DOM.\n * Retorna true si se agreg\u00F3 la suscripci\u00F3n, o false si se encontr\u00F3 un nodo padre ya suscrito.\n */\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n const currentVnode = current.vnode as VnodeWithDom;\n if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n return;\n }\n\n let hasParent = false;\n let parent = currentVnode.dom.parentElement as DomElement;\n while (parent) {\n if (domWithVnodesToUpdate.has(parent)) {\n hasParent = true;\n break;\n }\n parent = parent.parentElement as DomElement;\n }\n\n // Si no hay nodo padre registrado, se crea la suscripci\u00F3n.\n if (!hasParent) {\n const dom = currentVnode.dom;\n const subscription = () => {\n updateVnode(dom.vnode);\n if (!dom.parentElement) {\n subscribers.delete(subscription);\n domWithVnodesToUpdate.delete(dom);\n }\n };\n subscribers.add(subscription);\n domWithVnodesToUpdate.add(dom);\n }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType | (() => StateType) | null,\n pulses: PulsesType,\n immutable = false\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const boundPulses: Pulses<StateType> = {};\n for (const key in pulses) {\n if (typeof pulses[key] !== \"function\") {\n throw new Error(`Pulse '${key}' must be a function`);\n }\n if (key === \"state\") {\n throw new Error(`A pulse cannot be named 'state'`);\n }\n boundPulses[key] = getPulseMethod(key);\n }\n\n const localState: StateType =\n (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n function isMutable() {\n if (immutable) {\n throw new Error(\"You need to call a pulse to modify the state\");\n }\n }\n\n let currentState: StateType | null = null;\n let pulseCallCount = 0;\n\n const proxyState = new Proxy(localState, {\n get: (state, prop: string) => {\n // If we are in a pulse, we return the value of the cloned state.\n if (currentState) {\n return currentState[prop];\n }\n\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n return state[prop];\n },\n set: (state, prop: string, value: any) => {\n isMutable();\n Reflect.set(state, prop, value);\n return true;\n },\n deleteProperty: (state, prop: string) => {\n isMutable();\n Reflect.deleteProperty(state, prop);\n return true;\n }\n });\n\n function syncState(newState: StateType) {\n for (const key in newState) {\n localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n }\n for (const key in localState) {\n if (!(key in newState)) {\n Reflect.deleteProperty(localState, key);\n }\n }\n }\n\n let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n function debouncedUpdate() {\n if (debounceTimeout) {\n clearTimeout(debounceTimeout);\n }\n debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n }\n\n function setState(newState: StateType, flush = false) {\n pulseCallCount--;\n if (!hasChanged(localState, newState)) {\n return;\n }\n if (pulseCallCount > 0 && !flush) {\n return;\n }\n syncState(newState);\n currentState = null;\n debouncedUpdate();\n }\n\n function getPulseMethod(key: string) {\n function pulseMethod(this: Pulse<StateType, any> & { $flush: () => Promise<void> }, ...args: any[]) {\n pulseCallCount++;\n if (currentState === null) {\n currentState = deepCloneUnfreeze(localState);\n }\n\n const $flush = async () => {\n setState(currentState as StateType, true);\n currentState = deepCloneUnfreeze(localState);\n await new Promise((resolve) => setTimeout(resolve, 0));\n };\n\n Reflect.set(this, \"$flush\", $flush);\n const emptyFlush = async () => {};\n\n try {\n const pulseResult = pulses[key].apply(this, [currentState, ...args] as any);\n if (pulseResult instanceof Promise) {\n return pulseResult\n .then((resolvedValue) => {\n setState(currentState as StateType);\n Reflect.set(this, \"$flush\", emptyFlush);\n return resolvedValue;\n })\n .catch((error) => {\n console.error(`Error in pulse '${key}':`, error);\n Reflect.set(this, \"$flush\", emptyFlush);\n throw error;\n });\n } else {\n setState(currentState);\n Reflect.set(this, \"$flush\", emptyFlush);\n return pulseResult;\n }\n } catch (error) {\n console.error(`Error in pulse '${key}':`, error);\n Reflect.set(this, \"$flush\", emptyFlush);\n throw error;\n }\n }\n\n return pulseMethod;\n }\n\n syncState(localState);\n\n const listeners: Record<string, Function[]> = {};\n const trigger = (event: string, ...args: any[]) => {\n if (listeners[event]) {\n listeners[event].forEach((callback) => callback(...args));\n }\n };\n\n const pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\n }\n if (prop === \"on\") {\n return (event: string, callback: Function) => {\n if (!listeners[event]) {\n listeners[event] = [];\n }\n listeners[event].push(callback);\n };\n }\n if (prop === \"off\") {\n return (event: string, callback: Function) => {\n if (listeners[event]) {\n listeners[event] = listeners[event].filter((cb) => cb !== callback);\n }\n };\n }\n if (!(prop in pulses)) {\n throw new Error(`Pulse '${prop}' does not exist`);\n }\n const pulseMethod = pulses[prop];\n\n if (typeof pulseMethod === \"function\") {\n return (...args: any[]) => {\n const result = pulseMethod.apply(pulsesProxy, args as any);\n if (result instanceof Promise) {\n return result.then((r) => {\n trigger(\"pulse\", prop, args);\n return r;\n });\n }\n trigger(\"pulse\", prop, args);\n return result;\n };\n }\n\n return pulseMethod;\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n console.warn(\n \"Warning: You are working with a mutable state. This can lead to unpredictable behavior. All state changes made outside of a pulse will not trigger a re-render.\"\n );\n return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n const runEffect = () => {\n try {\n effectStack.push(runEffect);\n effect();\n } finally {\n effectStack.pop();\n }\n };\n runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n let value = initialValue;\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const runSubscribers = () => {\n subscribers.forEach((subscriber) => subscriber());\n };\n\n const read = (): T => {\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n return value;\n };\n\n const write = (newValue: T | ((current: T) => T)): void => {\n const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n if (!hasChanged(value, resolvedValue)) {\n return;\n }\n value = resolvedValue;\n runSubscribers();\n };\n\n return [read, write, runSubscribers];\n}\n"],
|
|
5
|
-
"mappings": ";AACA,SAAS,aAA2B,eAA2B;AAC/D,SAAS,mBAAmB,YAAY,kBAAkB;
|
|
4
|
+
"sourcesContent": ["/* eslint-disable no-console */\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type PulseContext = {\n $flush: () => Promise<void>;\n};\n\nexport type Pulse<StateType, TReturn = unknown> = (\n this: PulseContext,\n state: StateType,\n ...args: any[]\n) => TReturn | Promise<TReturn>;\n\nexport type Pulses<StateType> = Record<string, Pulse<StateType, any>>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType extends Pulses<any>> = {\n [K in keyof PulsesType]: PulsesType[K] extends (this: any, state: any, ...args: infer Args) => infer R\n ? (...args: Args) => R\n : never;\n};\n\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n const currentVnode = current.vnode as VnodeWithDom;\n if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n return;\n }\n\n let hasParent = false;\n let parent = currentVnode.dom.parentElement as DomElement;\n while (parent) {\n if (domWithVnodesToUpdate.has(parent)) {\n hasParent = true;\n break;\n }\n parent = parent.parentElement as DomElement;\n }\n\n if (!hasParent) {\n const dom = currentVnode.dom;\n const subscription = () => {\n updateVnode(dom.vnode);\n if (!dom.parentElement) {\n subscribers.delete(subscription);\n domWithVnodesToUpdate.delete(dom);\n }\n };\n subscribers.add(subscription);\n domWithVnodesToUpdate.add(dom);\n }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType | (() => StateType) | null,\n pulses: PulsesType,\n immutable = false\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const boundPulses: Record<string, Function> = {};\n for (const key in pulses) {\n if (typeof pulses[key] !== \"function\") {\n throw new Error(`Pulse '${key}' must be a function`);\n }\n if (key === \"state\") {\n throw new Error(`A pulse cannot be named 'state'`);\n }\n boundPulses[key] = getPulseMethod(key);\n }\n\n const localState: StateType =\n (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n function isMutable() {\n if (immutable) {\n throw new Error(\"You need to call a pulse to modify the state\");\n }\n }\n\n let currentState: StateType | null = null;\n let pulseCallCount = 0;\n\n const proxyState = new Proxy(localState, {\n get: (state, prop: string) => {\n if (currentState) {\n return currentState[prop];\n }\n\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n return state[prop];\n },\n set: (state, prop: string, value: any) => {\n isMutable();\n Reflect.set(state, prop, value);\n return true;\n },\n deleteProperty: (state, prop: string) => {\n isMutable();\n Reflect.deleteProperty(state, prop);\n return true;\n }\n });\n\n function syncState(newState: StateType) {\n for (const key in newState) {\n localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n }\n for (const key in localState) {\n if (!(key in newState)) {\n Reflect.deleteProperty(localState, key);\n }\n }\n }\n\n let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n function debouncedUpdate() {\n if (debounceTimeout) {\n clearTimeout(debounceTimeout);\n }\n debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n }\n\n function setState(newState: StateType, flush = false) {\n pulseCallCount--;\n if (!hasChanged(localState, newState)) {\n return;\n }\n if (pulseCallCount > 0 && !flush) {\n return;\n }\n syncState(newState);\n currentState = null;\n debouncedUpdate();\n }\n\n function getPulseMethod(key: string) {\n function pulseMethod(this: PulseContext, ...args: any[]) {\n pulseCallCount++;\n if (currentState === null) {\n currentState = deepCloneUnfreeze(localState);\n }\n\n this.$flush = async () => {\n if (currentState) {\n setState(currentState, true);\n currentState = deepCloneUnfreeze(localState);\n }\n await new Promise((resolve) => setTimeout(resolve, 0));\n };\n\n const emptyFlush = async () => {};\n\n try {\n const pulseResult = pulses[key].apply(this, [currentState, ...args]);\n if (pulseResult instanceof Promise) {\n return pulseResult\n .then((resolvedValue) => {\n setState(currentState as StateType);\n this.$flush = emptyFlush;\n return resolvedValue;\n })\n .catch((error) => {\n console.error(`Error in pulse '${key}':`, error);\n this.$flush = emptyFlush;\n throw error;\n });\n } else {\n setState(currentState);\n this.$flush = emptyFlush;\n return pulseResult;\n }\n } catch (error) {\n console.error(`Error in pulse '${key}':`, error);\n Reflect.set(this, \"$flush\", emptyFlush);\n throw error;\n }\n }\n\n return pulseMethod;\n }\n\n syncState(localState);\n\n const listeners: Record<string, Function[]> = {};\n const trigger = (event: string, ...args: any[]) => {\n if (listeners[event]) {\n listeners[event].forEach((callback) => callback(...args));\n }\n };\n\n const pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\n }\n if (prop === \"on\") {\n return (event: string, callback: Function) => {\n if (!listeners[event]) {\n listeners[event] = [];\n }\n listeners[event].push(callback);\n };\n }\n if (prop === \"off\") {\n return (event: string, callback: Function) => {\n if (listeners[event]) {\n listeners[event] = listeners[event].filter((cb) => cb !== callback);\n }\n };\n }\n if (!(prop in pulses)) {\n throw new Error(`Pulse '${prop}' does not exist`);\n }\n const pulseMethod = pulses[prop];\n\n return (...args: any[]) => {\n const result = pulseMethod.apply(pulseMethod, args);\n if (result instanceof Promise) {\n return result.then((r) => {\n trigger(\"pulse\", prop, args);\n return r;\n });\n }\n trigger(\"pulse\", prop, args);\n return result;\n };\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n) {\n return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n) {\n console.warn(\n \"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render.\"\n );\n return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n const runEffect = () => {\n try {\n effectStack.push(runEffect);\n effect();\n } finally {\n effectStack.pop();\n }\n };\n runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n let value = initialValue;\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const runSubscribers = () => {\n subscribers.forEach((subscriber) => subscriber());\n };\n\n const read = (): T => {\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n return value;\n };\n\n const write = (newValue: T | ((current: T) => T)): void => {\n const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n if (!hasChanged(value, resolvedValue)) {\n return;\n }\n value = resolvedValue;\n runSubscribers();\n };\n\n return [read, write, runSubscribers];\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["pulses"]
|
|
7
7
|
}
|
package/lib/pulses/index.ts
CHANGED
|
@@ -4,23 +4,28 @@ import { deepCloneUnfreeze, deepFreeze, hasChanged } from "valyrian.js/utils";
|
|
|
4
4
|
|
|
5
5
|
type State = Record<string, any>;
|
|
6
6
|
|
|
7
|
-
export type
|
|
7
|
+
export type PulseContext = {
|
|
8
|
+
$flush: () => Promise<void>;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
export type Pulse<StateType, TReturn = unknown> = (
|
|
12
|
+
this: PulseContext,
|
|
13
|
+
state: StateType,
|
|
14
|
+
...args: any[]
|
|
15
|
+
) => TReturn | Promise<TReturn>;
|
|
8
16
|
|
|
9
|
-
export type Pulses<StateType> = Record<string, Pulse<StateType, any
|
|
17
|
+
export type Pulses<StateType> = Record<string, Pulse<StateType, any>>;
|
|
10
18
|
|
|
11
19
|
type ProxyState<StateType> = StateType & { [key: string]: any };
|
|
12
20
|
|
|
13
21
|
const effectStack: Function[] = [];
|
|
14
22
|
|
|
15
23
|
type StorePulses<PulsesType extends Pulses<any>> = {
|
|
16
|
-
[K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R
|
|
24
|
+
[K in keyof PulsesType]: PulsesType[K] extends (this: any, state: any, ...args: infer Args) => infer R
|
|
17
25
|
? (...args: Args) => R
|
|
18
26
|
: never;
|
|
19
27
|
};
|
|
20
28
|
|
|
21
|
-
/** Función auxiliar para registrar la suscripción al nodo DOM.
|
|
22
|
-
* Retorna true si se agregó la suscripción, o false si se encontró un nodo padre ya suscrito.
|
|
23
|
-
*/
|
|
24
29
|
function registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {
|
|
25
30
|
const currentVnode = current.vnode as VnodeWithDom;
|
|
26
31
|
if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {
|
|
@@ -37,7 +42,6 @@ function registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpda
|
|
|
37
42
|
parent = parent.parentElement as DomElement;
|
|
38
43
|
}
|
|
39
44
|
|
|
40
|
-
// Si no hay nodo padre registrado, se crea la suscripción.
|
|
41
45
|
if (!hasParent) {
|
|
42
46
|
const dom = currentVnode.dom;
|
|
43
47
|
const subscription = () => {
|
|
@@ -64,7 +68,7 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
64
68
|
const subscribers = new Set<Function>();
|
|
65
69
|
const domWithVnodesToUpdate = new WeakSet<DomElement>();
|
|
66
70
|
|
|
67
|
-
const boundPulses:
|
|
71
|
+
const boundPulses: Record<string, Function> = {};
|
|
68
72
|
for (const key in pulses) {
|
|
69
73
|
if (typeof pulses[key] !== "function") {
|
|
70
74
|
throw new Error(`Pulse '${key}' must be a function`);
|
|
@@ -89,7 +93,6 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
89
93
|
|
|
90
94
|
const proxyState = new Proxy(localState, {
|
|
91
95
|
get: (state, prop: string) => {
|
|
92
|
-
// If we are in a pulse, we return the value of the cloned state.
|
|
93
96
|
if (currentState) {
|
|
94
97
|
return currentState[prop];
|
|
95
98
|
}
|
|
@@ -148,38 +151,39 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
148
151
|
}
|
|
149
152
|
|
|
150
153
|
function getPulseMethod(key: string) {
|
|
151
|
-
function pulseMethod(this:
|
|
154
|
+
function pulseMethod(this: PulseContext, ...args: any[]) {
|
|
152
155
|
pulseCallCount++;
|
|
153
156
|
if (currentState === null) {
|
|
154
157
|
currentState = deepCloneUnfreeze(localState);
|
|
155
158
|
}
|
|
156
159
|
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
+
this.$flush = async () => {
|
|
161
|
+
if (currentState) {
|
|
162
|
+
setState(currentState, true);
|
|
163
|
+
currentState = deepCloneUnfreeze(localState);
|
|
164
|
+
}
|
|
160
165
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
161
166
|
};
|
|
162
167
|
|
|
163
|
-
Reflect.set(this, "$flush", $flush);
|
|
164
168
|
const emptyFlush = async () => {};
|
|
165
169
|
|
|
166
170
|
try {
|
|
167
|
-
const pulseResult = pulses[key].apply(this, [currentState, ...args]
|
|
171
|
+
const pulseResult = pulses[key].apply(this, [currentState, ...args]);
|
|
168
172
|
if (pulseResult instanceof Promise) {
|
|
169
173
|
return pulseResult
|
|
170
174
|
.then((resolvedValue) => {
|
|
171
175
|
setState(currentState as StateType);
|
|
172
|
-
|
|
176
|
+
this.$flush = emptyFlush;
|
|
173
177
|
return resolvedValue;
|
|
174
178
|
})
|
|
175
179
|
.catch((error) => {
|
|
176
180
|
console.error(`Error in pulse '${key}':`, error);
|
|
177
|
-
|
|
181
|
+
this.$flush = emptyFlush;
|
|
178
182
|
throw error;
|
|
179
183
|
});
|
|
180
184
|
} else {
|
|
181
185
|
setState(currentState);
|
|
182
|
-
|
|
186
|
+
this.$flush = emptyFlush;
|
|
183
187
|
return pulseResult;
|
|
184
188
|
}
|
|
185
189
|
} catch (error) {
|
|
@@ -226,21 +230,17 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
226
230
|
}
|
|
227
231
|
const pulseMethod = pulses[prop];
|
|
228
232
|
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
};
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
return pulseMethod;
|
|
233
|
+
return (...args: any[]) => {
|
|
234
|
+
const result = pulseMethod.apply(pulseMethod, args);
|
|
235
|
+
if (result instanceof Promise) {
|
|
236
|
+
return result.then((r) => {
|
|
237
|
+
trigger("pulse", prop, args);
|
|
238
|
+
return r;
|
|
239
|
+
});
|
|
240
|
+
}
|
|
241
|
+
trigger("pulse", prop, args);
|
|
242
|
+
return result;
|
|
243
|
+
};
|
|
244
244
|
}
|
|
245
245
|
});
|
|
246
246
|
|
|
@@ -254,24 +254,16 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
254
254
|
export function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(
|
|
255
255
|
initialState: StateType,
|
|
256
256
|
pulses: PulsesType
|
|
257
|
-
)
|
|
258
|
-
state: ProxyState<StateType>;
|
|
259
|
-
on: (event: string, callback: Function) => void;
|
|
260
|
-
off: (event: string, callback: Function) => void;
|
|
261
|
-
} {
|
|
257
|
+
) {
|
|
262
258
|
return createStore(initialState, pulses, true);
|
|
263
259
|
}
|
|
264
260
|
|
|
265
261
|
export function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(
|
|
266
262
|
initialState: StateType,
|
|
267
263
|
pulses: PulsesType
|
|
268
|
-
)
|
|
269
|
-
state: ProxyState<StateType>;
|
|
270
|
-
on: (event: string, callback: Function) => void;
|
|
271
|
-
off: (event: string, callback: Function) => void;
|
|
272
|
-
} {
|
|
264
|
+
) {
|
|
273
265
|
console.warn(
|
|
274
|
-
"Warning: You are working with a mutable state.
|
|
266
|
+
"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render."
|
|
275
267
|
);
|
|
276
268
|
return createStore(initialState, pulses, false);
|
|
277
269
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "valyrian.js",
|
|
3
|
-
"version": "8.1.
|
|
3
|
+
"version": "8.1.7",
|
|
4
4
|
"description": "Lightweight steel to forge PWAs. (Minimal Frontend Framework with server side rendering and other capabilities)",
|
|
5
5
|
"repository": "git@github.com:Masquerade-Circus/valyrian.js.git",
|
|
6
6
|
"author": "Masquerade <christian@masquerade-circus.net>",
|