valyrian.js 8.0.14 → 8.1.0
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 +4 -0
- package/dist/lib/pulses/index.d.ts.map +1 -1
- package/dist/lib/redux-devtools/index.d.ts +15 -0
- package/dist/lib/redux-devtools/index.d.ts.map +1 -0
- package/dist/pulses/index.js +32 -1
- 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 +32 -1
- package/dist/pulses/index.mjs.map +2 -2
- package/dist/redux-devtools/index.js +81 -0
- package/dist/redux-devtools/index.js.map +7 -0
- package/dist/redux-devtools/index.min.js +1 -0
- package/dist/redux-devtools/index.min.js.map +1 -0
- package/dist/redux-devtools/index.mjs +60 -0
- package/dist/redux-devtools/index.mjs.map +7 -0
- package/lib/pulses/index.ts +54 -6
- package/lib/redux-devtools/index.ts +86 -0
- package/package.json +1 -1
|
@@ -11,9 +11,13 @@ type StorePulses<PulsesType extends Pulses<any>> = {
|
|
|
11
11
|
};
|
|
12
12
|
export declare function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(initialState: StateType, pulses: PulsesType): StorePulses<PulsesType> & {
|
|
13
13
|
state: ProxyState<StateType>;
|
|
14
|
+
on: (event: string, callback: Function) => void;
|
|
15
|
+
off: (event: string, callback: Function) => void;
|
|
14
16
|
};
|
|
15
17
|
export declare function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(initialState: StateType, pulses: PulsesType): StorePulses<PulsesType> & {
|
|
16
18
|
state: ProxyState<StateType>;
|
|
19
|
+
on: (event: string, callback: Function) => void;
|
|
20
|
+
off: (event: string, callback: Function) => void;
|
|
17
21
|
};
|
|
18
22
|
export declare function createEffect(effect: Function): void;
|
|
19
23
|
export declare function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void];
|
|
@@ -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,CAAC,KAAK,EAAE,SAAS,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE,KAAK,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;AAEnH,MAAM,MAAM,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,SAAS,EAAE,GAAG,CAAC,GAAG;IAAE,MAAM,CAAC,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAA;CAAE,CAAC,CAAC;AAEzG,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,KAAK,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,MAAM,IAAI,KAAK,MAAM,CAAC,GACvF,CAAC,GAAG,IAAI,EAAE,IAAI,KAAK,CAAC,GACpB,KAAK;CACV,CAAC;
|
|
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,CAAC,KAAK,EAAE,SAAS,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE,KAAK,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;AAEnH,MAAM,MAAM,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,SAAS,EAAE,GAAG,CAAC,GAAG;IAAE,MAAM,CAAC,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAA;CAAE,CAAC,CAAC;AAEzG,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,KAAK,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,MAAM,IAAI,KAAK,MAAM,CAAC,GACvF,CAAC,GAAG,IAAI,EAAE,IAAI,KAAK,CAAC,GACpB,KAAK;CACV,CAAC;AA2OF,wBAAgB,gBAAgB,CAAC,SAAS,SAAS,KAAK,EAAE,UAAU,SAAS,MAAM,CAAC,SAAS,CAAC,EAC5F,YAAY,EAAE,SAAS,EACvB,MAAM,EAAE,UAAU,GACjB,WAAW,CAAC,UAAU,CAAC,GAAG;IAC3B,KAAK,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC;IAC7B,EAAE,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI,CAAC;IAChD,GAAG,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI,CAAC;CAClD,CAEA;AAED,wBAAgB,kBAAkB,CAAC,SAAS,SAAS,KAAK,EAAE,UAAU,SAAS,MAAM,CAAC,SAAS,CAAC,EAC9F,YAAY,EAAE,SAAS,EACvB,MAAM,EAAE,UAAU,GACjB,WAAW,CAAC,UAAU,CAAC,GAAG;IAC3B,KAAK,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC;IAC7B,EAAE,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI,CAAC;IAChD,GAAG,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI,CAAC;CAClD,CAKA;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"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { FluxStore } from "valyrian.js/flux-store";
|
|
2
|
+
declare global {
|
|
3
|
+
interface Window {
|
|
4
|
+
__REDUX_DEVTOOLS_EXTENSION__: any;
|
|
5
|
+
}
|
|
6
|
+
}
|
|
7
|
+
interface DevToolsOptions {
|
|
8
|
+
name?: string;
|
|
9
|
+
[key: string]: any;
|
|
10
|
+
}
|
|
11
|
+
export declare function connectFluxStore(store: FluxStore, options?: DevToolsOptions): void;
|
|
12
|
+
export declare function connectPulseStore(store: any, options?: DevToolsOptions): void;
|
|
13
|
+
export declare function connectPulse(pulse: any, options?: DevToolsOptions): any;
|
|
14
|
+
export {};
|
|
15
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/redux-devtools/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,MAAM,wBAAwB,CAAC;AAEnD,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,MAAM;QACd,4BAA4B,EAAE,GAAG,CAAC;KACnC;CACF;AAED,UAAU,eAAe;IACvB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAC;CACpB;AAUD,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,SAAS,EAAE,OAAO,GAAE,eAAoB,QAqB/E;AAED,wBAAgB,iBAAiB,CAAC,KAAK,EAAE,GAAG,EAAE,OAAO,GAAE,eAAoB,QAe1E;AAED,wBAAgB,YAAY,CAAC,KAAK,EAAE,GAAG,EAAE,OAAO,GAAE,eAAoB,OAuBrE"}
|
package/dist/pulses/index.js
CHANGED
|
@@ -169,17 +169,48 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
169
169
|
return pulseMethod;
|
|
170
170
|
}
|
|
171
171
|
syncState(localState);
|
|
172
|
+
const listeners = {};
|
|
173
|
+
const trigger = (event, ...args) => {
|
|
174
|
+
if (listeners[event]) {
|
|
175
|
+
listeners[event].forEach((callback) => callback(...args));
|
|
176
|
+
}
|
|
177
|
+
};
|
|
172
178
|
const pulsesProxy = new Proxy(boundPulses, {
|
|
173
179
|
get: (pulses2, prop) => {
|
|
174
180
|
if (prop === "state") {
|
|
175
181
|
return proxyState;
|
|
176
182
|
}
|
|
183
|
+
if (prop === "on") {
|
|
184
|
+
return (event, callback) => {
|
|
185
|
+
if (!listeners[event]) {
|
|
186
|
+
listeners[event] = [];
|
|
187
|
+
}
|
|
188
|
+
listeners[event].push(callback);
|
|
189
|
+
};
|
|
190
|
+
}
|
|
191
|
+
if (prop === "off") {
|
|
192
|
+
return (event, callback) => {
|
|
193
|
+
if (listeners[event]) {
|
|
194
|
+
listeners[event] = listeners[event].filter((cb) => cb !== callback);
|
|
195
|
+
}
|
|
196
|
+
};
|
|
197
|
+
}
|
|
177
198
|
if (!(prop in pulses2)) {
|
|
178
199
|
throw new Error(`Pulse '${prop}' does not exist`);
|
|
179
200
|
}
|
|
180
201
|
const pulseMethod = pulses2[prop];
|
|
181
202
|
if (typeof pulseMethod === "function") {
|
|
182
|
-
return
|
|
203
|
+
return (...args) => {
|
|
204
|
+
const result = pulseMethod.apply(pulsesProxy, 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
|
+
};
|
|
183
214
|
}
|
|
184
215
|
return pulseMethod;
|
|
185
216
|
}
|
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> & { state: ProxyState<StateType> } {\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]);\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 pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\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 pulseMethod.bind(pulsesProxy);\n }\n\n return pulseMethod;\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & { state: ProxyState<StateType> };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n): StorePulses<PulsesType> & { state: ProxyState<StateType> } {\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> & { state: ProxyState<StateType> } {\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;AAU1D,IAAM,cAA0B,CAAC;AAWjC,SAAS,wBAAwB,aAA4B,uBAAkD;AAC7G,QAAM,eAAe,wBAAQ;AAC7B,MAAI,CAAC,gBAAgB,sBAAsB,IAAI,aAAa,GAAG,GAAG;AAChE;AAAA,EACF;AAEA,MAAI,YAAY;AAChB,MAAI,SAAS,aAAa,IAAI;AAC9B,SAAO,QAAQ;AACb,QAAI,sBAAsB,IAAI,MAAM,GAAG;AACrC,kBAAY;AACZ;AAAA,IACF;AACA,aAAS,OAAO;AAAA,EAClB;AAGA,MAAI,CAAC,WAAW;AACd,UAAM,MAAM,aAAa;AACzB,UAAM,eAAe,MAAM;AACzB,uCAAY,IAAI,KAAK;AACrB,UAAI,CAAC,IAAI,eAAe;AACtB,oBAAY,OAAO,YAAY;AAC/B,8BAAsB,OAAO,GAAG;AAAA,MAClC;AAAA,IACF;AACA,gBAAY,IAAI,YAAY;AAC5B,0BAAsB,IAAI,GAAG;AAAA,EAC/B;AACF;AAEA,SAAS,YACP,cACA,QACA,YAAY,
|
|
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": "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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:()=>
|
|
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,l=(s=o,((o,s,l,u)=>{if(s&&"object"==typeof s||"function"==typeof s)for(let a of n(s))r.call(o,a)||a===l||e(o,a,{get:()=>s[a],enumerable:!(u=t(s,a))||u.enumerable});return o})(e({},"__esModule",{value:!0}),s)),u=require("valyrian.js"),a=require("valyrian.js/utils"),c=[];function i(e,t){const n=u.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,u.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 l=("function"==typeof e?e():e)||{};function u(){if(n)throw new Error("You need to call a pulse to modify the state")}let f=null,h=0;const p=new Proxy(l,{get:(e,t)=>{if(f)return f[t];const n=c[c.length-1];return n&&!r.has(n)&&r.add(n),i(r,o),e[t]},set:(e,t,n)=>(u(),Reflect.set(e,t,n),!0),deleteProperty:(e,t)=>(u(),Reflect.deleteProperty(e,t),!0)});function d(e){for(const t in e)l[t]=n?(0,a.deepFreeze)(e[t]):e[t];for(const t in l)t in e||Reflect.deleteProperty(l,t)}let y=null;function w(e,t=!1){h--,(0,a.hasChanged)(l,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)(l)),Reflect.set(this,"$flush",async()=>{w(f,!0),f=(0,a.deepCloneUnfreeze)(l),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),Reflect.set(this,"$flush",r),e)).catch(t=>{throw console.error(`Error in pulse '${e}':`,t),Reflect.set(this,"$flush",r),t}):(w(f),Reflect.set(this,"$flush",r),o)}catch(t){throw console.error(`Error in pulse '${e}':`,t),Reflect.set(this,"$flush",r),t}}}d(l);const P={},b=(e,...t)=>{P[e]&&P[e].forEach(e=>e(...t))},g=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"function"==typeof n?(...e)=>{const r=n.apply(g,e);return r instanceof Promise?r.then(n=>(b("pulse",t,e),n)):(b("pulse",t,e),r)}:n}});return g}function h(e,t){return f(e,t,!0)}function p(e,t){return console.warn("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."),f(e,t,!1)}function d(e){const t=()=>{try{c.push(t),e()}finally{c.pop()}};t()}function y(e){let t=e;const n=new Set,r=new WeakSet,o=()=>{n.forEach(e=>e())};return[()=>{const e=c[c.length-1];return e&&!n.has(e)&&n.add(e),i(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=l:self.ValyrianPulses=l})();//# 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","pulsesProxy","pulseMethod","bind","warn","effect","runEffect","push","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> & { state: ProxyState<StateType> } {\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]);\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 pulsesProxy = new Proxy(boundPulses, {\n    get: (pulses, prop: string) => {\n      if (prop === \"state\") {\n        return proxyState;\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 pulseMethod.bind(pulsesProxy);\n      }\n\n      return pulseMethod;\n    }\n  });\n\n  return pulsesProxy as StorePulses<PulsesType> & { state: ProxyState<StateType> };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n  initialState: StateType,\n  pulses: PulsesType\n): StorePulses<PulsesType> & { state: ProxyState<StateType> } {\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> & { state: ProxyState<StateType> } {\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","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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"}
|
package/dist/pulses/index.mjs
CHANGED
|
@@ -142,17 +142,48 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
142
142
|
return pulseMethod;
|
|
143
143
|
}
|
|
144
144
|
syncState(localState);
|
|
145
|
+
const listeners = {};
|
|
146
|
+
const trigger = (event, ...args) => {
|
|
147
|
+
if (listeners[event]) {
|
|
148
|
+
listeners[event].forEach((callback) => callback(...args));
|
|
149
|
+
}
|
|
150
|
+
};
|
|
145
151
|
const pulsesProxy = new Proxy(boundPulses, {
|
|
146
152
|
get: (pulses2, prop) => {
|
|
147
153
|
if (prop === "state") {
|
|
148
154
|
return proxyState;
|
|
149
155
|
}
|
|
156
|
+
if (prop === "on") {
|
|
157
|
+
return (event, callback) => {
|
|
158
|
+
if (!listeners[event]) {
|
|
159
|
+
listeners[event] = [];
|
|
160
|
+
}
|
|
161
|
+
listeners[event].push(callback);
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
if (prop === "off") {
|
|
165
|
+
return (event, callback) => {
|
|
166
|
+
if (listeners[event]) {
|
|
167
|
+
listeners[event] = listeners[event].filter((cb) => cb !== callback);
|
|
168
|
+
}
|
|
169
|
+
};
|
|
170
|
+
}
|
|
150
171
|
if (!(prop in pulses2)) {
|
|
151
172
|
throw new Error(`Pulse '${prop}' does not exist`);
|
|
152
173
|
}
|
|
153
174
|
const pulseMethod = pulses2[prop];
|
|
154
175
|
if (typeof pulseMethod === "function") {
|
|
155
|
-
return
|
|
176
|
+
return (...args) => {
|
|
177
|
+
const result = pulseMethod.apply(pulsesProxy, 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
|
+
};
|
|
156
187
|
}
|
|
157
188
|
return pulseMethod;
|
|
158
189
|
}
|
|
@@ -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> & { state: ProxyState<StateType> } {\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]);\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 pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\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 pulseMethod.bind(pulsesProxy);\n }\n\n return pulseMethod;\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & { state: ProxyState<StateType> };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(\n initialState: StateType,\n pulses: PulsesType\n): StorePulses<PulsesType> & { state: ProxyState<StateType> } {\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> & { state: ProxyState<StateType> } {\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;AAU1D,IAAM,cAA0B,CAAC;AAWjC,SAAS,wBAAwB,aAA4B,uBAAkD;AAC7G,QAAM,eAAe,QAAQ;AAC7B,MAAI,CAAC,gBAAgB,sBAAsB,IAAI,aAAa,GAAG,GAAG;AAChE;AAAA,EACF;AAEA,MAAI,YAAY;AAChB,MAAI,SAAS,aAAa,IAAI;AAC9B,SAAO,QAAQ;AACb,QAAI,sBAAsB,IAAI,MAAM,GAAG;AACrC,kBAAY;AACZ;AAAA,IACF;AACA,aAAS,OAAO;AAAA,EAClB;AAGA,MAAI,CAAC,WAAW;AACd,UAAM,MAAM,aAAa;AACzB,UAAM,eAAe,MAAM;AACzB,kBAAY,IAAI,KAAK;AACrB,UAAI,CAAC,IAAI,eAAe;AACtB,oBAAY,OAAO,YAAY;AAC/B,8BAAsB,OAAO,GAAG;AAAA,MAClC;AAAA,IACF;AACA,gBAAY,IAAI,YAAY;AAC5B,0BAAsB,IAAI,GAAG;AAAA,EAC/B;AACF;AAEA,SAAS,YACP,cACA,QACA,YAAY,
|
|
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": "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",
|
|
6
6
|
"names": ["pulses"]
|
|
7
7
|
}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __export = (target, all) => {
|
|
7
|
+
for (var name in all)
|
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
+
};
|
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
+
for (let key of __getOwnPropNames(from))
|
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
+
|
|
20
|
+
// lib/redux-devtools/index.ts
|
|
21
|
+
var index_exports = {};
|
|
22
|
+
__export(index_exports, {
|
|
23
|
+
connectFluxStore: () => connectFluxStore,
|
|
24
|
+
connectPulse: () => connectPulse,
|
|
25
|
+
connectPulseStore: () => connectPulseStore
|
|
26
|
+
});
|
|
27
|
+
module.exports = __toCommonJS(index_exports);
|
|
28
|
+
function getDevTools() {
|
|
29
|
+
const isBrowser = typeof window !== "undefined";
|
|
30
|
+
if (isBrowser && window.__REDUX_DEVTOOLS_EXTENSION__) {
|
|
31
|
+
return window.__REDUX_DEVTOOLS_EXTENSION__;
|
|
32
|
+
}
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
function connectFluxStore(store, options = {}) {
|
|
36
|
+
const devTools = getDevTools();
|
|
37
|
+
if (!devTools) {
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
const name = options.name || "FluxStore";
|
|
41
|
+
const dt = devTools.connect({ name, ...options });
|
|
42
|
+
dt.init(store.state);
|
|
43
|
+
store.on("commit", (_, mutation, ...args) => {
|
|
44
|
+
dt.send({ type: mutation, payload: args }, store.state);
|
|
45
|
+
});
|
|
46
|
+
store.on("registerModule", (_, namespace) => {
|
|
47
|
+
dt.send({ type: `[Module] Register: ${namespace}` }, store.state);
|
|
48
|
+
});
|
|
49
|
+
store.on("unregisterModule", (_, namespace) => {
|
|
50
|
+
dt.send({ type: `[Module] Unregister: ${namespace}` }, store.state);
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
function connectPulseStore(store, options = {}) {
|
|
54
|
+
const devTools = getDevTools();
|
|
55
|
+
if (!devTools) {
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
const name = options.name || "PulseStore";
|
|
59
|
+
const dt = devTools.connect({ name, ...options });
|
|
60
|
+
dt.init(store.state);
|
|
61
|
+
if (store.on) {
|
|
62
|
+
store.on("pulse", (pulse, args) => {
|
|
63
|
+
dt.send({ type: pulse, payload: args }, store.state);
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
function connectPulse(pulse, options = {}) {
|
|
68
|
+
const devTools = getDevTools();
|
|
69
|
+
if (!devTools) {
|
|
70
|
+
return pulse;
|
|
71
|
+
}
|
|
72
|
+
const name = options.name || "Pulse";
|
|
73
|
+
const dt = devTools.connect({ name, ...options });
|
|
74
|
+
const [read, write, run] = pulse;
|
|
75
|
+
dt.init(read());
|
|
76
|
+
const newWrite = (newValue) => {
|
|
77
|
+
write(newValue);
|
|
78
|
+
dt.send({ type: "update", payload: newValue }, read());
|
|
79
|
+
};
|
|
80
|
+
return [read, newWrite, run];
|
|
81
|
+
}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../lib/redux-devtools/index.ts"],
|
|
4
|
+
"sourcesContent": ["/* eslint-disable no-console */\nimport { FluxStore } from \"valyrian.js/flux-store\";\n\ndeclare global {\n interface Window {\n __REDUX_DEVTOOLS_EXTENSION__: any;\n }\n}\n\ninterface DevToolsOptions {\n name?: string;\n [key: string]: any;\n}\n\nfunction getDevTools() {\n const isBrowser = typeof window !== \"undefined\";\n if (isBrowser && window.__REDUX_DEVTOOLS_EXTENSION__) {\n return window.__REDUX_DEVTOOLS_EXTENSION__;\n }\n return null;\n}\n\nexport function connectFluxStore(store: FluxStore, options: DevToolsOptions = {}) {\n const devTools = getDevTools();\n if (!devTools) {\n return;\n }\n\n const name = options.name || \"FluxStore\";\n const dt = devTools.connect({ name, ...options });\n dt.init(store.state);\n\n store.on(\"commit\", (_: any, mutation: string, ...args: any[]) => {\n dt.send({ type: mutation, payload: args }, store.state);\n });\n\n store.on(\"registerModule\", (_: any, namespace: string) => {\n dt.send({ type: `[Module] Register: ${namespace}` }, store.state);\n });\n\n store.on(\"unregisterModule\", (_: any, namespace: string) => {\n dt.send({ type: `[Module] Unregister: ${namespace}` }, store.state);\n });\n}\n\nexport function connectPulseStore(store: any, options: DevToolsOptions = {}) {\n const devTools = getDevTools();\n if (!devTools) {\n return;\n }\n\n const name = options.name || \"PulseStore\";\n const dt = devTools.connect({ name, ...options });\n dt.init(store.state);\n\n if (store.on) {\n store.on(\"pulse\", (pulse: string, args: any[]) => {\n dt.send({ type: pulse, payload: args }, store.state);\n });\n }\n}\n\nexport function connectPulse(pulse: any, options: DevToolsOptions = {}) {\n // Pulse is [read, write, runSubscribers]\n // We can't easily hook into the write function without wrapping it.\n // But the user asked for \"simple\" and \"using existing apis\".\n // If we want to debug individual pulses, we might need to wrap them.\n\n const devTools = getDevTools();\n if (!devTools) {\n return pulse;\n }\n\n const name = options.name || \"Pulse\";\n const dt = devTools.connect({ name, ...options });\n const [read, write, run] = pulse;\n \n dt.init(read());\n\n const newWrite = (newValue: any) => {\n write(newValue);\n dt.send({ type: \"update\", payload: newValue }, read());\n };\n\n return [read, newWrite, run];\n}\n"],
|
|
5
|
+
"mappings": ";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAcA,SAAS,cAAc;AACrB,QAAM,YAAY,OAAO,WAAW;AACpC,MAAI,aAAa,OAAO,8BAA8B;AACpD,WAAO,OAAO;AAAA,EAChB;AACA,SAAO;AACT;AAEO,SAAS,iBAAiB,OAAkB,UAA2B,CAAC,GAAG;AAChF,QAAM,WAAW,YAAY;AAC7B,MAAI,CAAC,UAAU;AACb;AAAA,EACF;AAEA,QAAM,OAAO,QAAQ,QAAQ;AAC7B,QAAM,KAAK,SAAS,QAAQ,EAAE,MAAM,GAAG,QAAQ,CAAC;AAChD,KAAG,KAAK,MAAM,KAAK;AAEnB,QAAM,GAAG,UAAU,CAAC,GAAQ,aAAqB,SAAgB;AAC/D,OAAG,KAAK,EAAE,MAAM,UAAU,SAAS,KAAK,GAAG,MAAM,KAAK;AAAA,EACxD,CAAC;AAED,QAAM,GAAG,kBAAkB,CAAC,GAAQ,cAAsB;AACxD,OAAG,KAAK,EAAE,MAAM,sBAAsB,SAAS,GAAG,GAAG,MAAM,KAAK;AAAA,EAClE,CAAC;AAED,QAAM,GAAG,oBAAoB,CAAC,GAAQ,cAAsB;AAC1D,OAAG,KAAK,EAAE,MAAM,wBAAwB,SAAS,GAAG,GAAG,MAAM,KAAK;AAAA,EACpE,CAAC;AACH;AAEO,SAAS,kBAAkB,OAAY,UAA2B,CAAC,GAAG;AAC3E,QAAM,WAAW,YAAY;AAC7B,MAAI,CAAC,UAAU;AACb;AAAA,EACF;AAEA,QAAM,OAAO,QAAQ,QAAQ;AAC7B,QAAM,KAAK,SAAS,QAAQ,EAAE,MAAM,GAAG,QAAQ,CAAC;AAChD,KAAG,KAAK,MAAM,KAAK;AAEnB,MAAI,MAAM,IAAI;AACZ,UAAM,GAAG,SAAS,CAAC,OAAe,SAAgB;AAChD,SAAG,KAAK,EAAE,MAAM,OAAO,SAAS,KAAK,GAAG,MAAM,KAAK;AAAA,IACrD,CAAC;AAAA,EACH;AACF;AAEO,SAAS,aAAa,OAAY,UAA2B,CAAC,GAAG;AAMtE,QAAM,WAAW,YAAY;AAC7B,MAAI,CAAC,UAAU;AACb,WAAO;AAAA,EACT;AAEA,QAAM,OAAO,QAAQ,QAAQ;AAC7B,QAAM,KAAK,SAAS,QAAQ,EAAE,MAAM,GAAG,QAAQ,CAAC;AAChD,QAAM,CAAC,MAAM,OAAO,GAAG,IAAI;AAE3B,KAAG,KAAK,KAAK,CAAC;AAEd,QAAM,WAAW,CAAC,aAAkB;AAClC,UAAM,QAAQ;AACd,OAAG,KAAK,EAAE,MAAM,UAAU,SAAS,SAAS,GAAG,KAAK,CAAC;AAAA,EACvD;AAEA,SAAO,CAAC,MAAM,UAAU,GAAG;AAC7B;",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
(()=>{"use strict";var e=Object.defineProperty,t=Object.getOwnPropertyDescriptor,n=Object.getOwnPropertyNames,o=Object.prototype.hasOwnProperty,r={};((t,n)=>{for(var o in n)e(t,o,{get:n[o],enumerable:!0})})(r,{connectFluxStore:()=>a,connectPulse:()=>l,connectPulseStore:()=>i});var u,s=(u=r,((r,u,s,c)=>{if(u&&"object"==typeof u||"function"==typeof u)for(let a of n(u))o.call(r,a)||a===s||e(r,a,{get:()=>u[a],enumerable:!(c=t(u,a))||c.enumerable});return r})(e({},"__esModule",{value:!0}),u));function c(){return"undefined"!=typeof window&&window.__REDUX_DEVTOOLS_EXTENSION__?window.__REDUX_DEVTOOLS_EXTENSION__:null}function a(e,t={}){const n=c();if(!n)return;const o=t.name||"FluxStore",r=n.connect({name:o,...t});r.init(e.state),e.on("commit",(t,n,...o)=>{r.send({type:n,payload:o},e.state)}),e.on("registerModule",(t,n)=>{r.send({type:`[Module] Register: ${n}`},e.state)}),e.on("unregisterModule",(t,n)=>{r.send({type:`[Module] Unregister: ${n}`},e.state)})}function i(e,t={}){const n=c();if(!n)return;const o=t.name||"PulseStore",r=n.connect({name:o,...t});r.init(e.state),e.on&&e.on("pulse",(t,n)=>{r.send({type:t,payload:n},e.state)})}function l(e,t={}){const n=c();if(!n)return e;const o=t.name||"Pulse",r=n.connect({name:o,...t}),[u,s,a]=e;r.init(u());return[u,e=>{s(e),r.send({type:"update",payload:e},u())},a]}"undefined"!=typeof module?module.exports=s:self.ValyrianReduxDevtools=s})();//# sourceMappingURL=index.min.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJpbmRleF9leHBvcnRzIiwiX19leHBvcnQiLCJjb25uZWN0Rmx1eFN0b3JlIiwiY29ubmVjdFB1bHNlIiwiY29ubmVjdFB1bHNlU3RvcmUiLCJfX0VYUE9SVFNfXyIsImdldERldlRvb2xzIiwid2luZG93IiwiX19SRURVWF9ERVZUT09MU19FWFRFTlNJT05fXyIsInN0b3JlIiwib3B0aW9ucyIsImRldlRvb2xzIiwibmFtZSIsImR0IiwiY29ubmVjdCIsImluaXQiLCJzdGF0ZSIsIm9uIiwiXyIsIm11dGF0aW9uIiwiYXJncyIsInNlbmQiLCJ0eXBlIiwicGF5bG9hZCIsIm5hbWVzcGFjZSIsInB1bHNlIiwicmVhZCIsIndyaXRlIiwicnVuIiwibmV3VmFsdWUiXSwic291cmNlcyI6WyIuLi8uLi9saWIvcmVkdXgtZGV2dG9vbHMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgbm8tY29uc29sZSAqL1xuaW1wb3J0IHsgRmx1eFN0b3JlIH0gZnJvbSBcInZhbHlyaWFuLmpzL2ZsdXgtc3RvcmVcIjtcblxuZGVjbGFyZSBnbG9iYWwge1xuICBpbnRlcmZhY2UgV2luZG93IHtcbiAgICBfX1JFRFVYX0RFVlRPT0xTX0VYVEVOU0lPTl9fOiBhbnk7XG4gIH1cbn1cblxuaW50ZXJmYWNlIERldlRvb2xzT3B0aW9ucyB7XG4gIG5hbWU/OiBzdHJpbmc7XG4gIFtrZXk6IHN0cmluZ106IGFueTtcbn1cblxuZnVuY3Rpb24gZ2V0RGV2VG9vbHMoKSB7XG4gIGNvbnN0IGlzQnJvd3NlciA9IHR5cGVvZiB3aW5kb3cgIT09IFwidW5kZWZpbmVkXCI7XG4gIGlmIChpc0Jyb3dzZXIgJiYgd2luZG93Ll9fUkVEVVhfREVWVE9PTFNfRVhURU5TSU9OX18pIHtcbiAgICByZXR1cm4gd2luZG93Ll9fUkVEVVhfREVWVE9PTFNfRVhURU5TSU9OX187XG4gIH1cbiAgcmV0dXJuIG51bGw7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjb25uZWN0Rmx1eFN0b3JlKHN0b3JlOiBGbHV4U3RvcmUsIG9wdGlvbnM6IERldlRvb2xzT3B0aW9ucyA9IHt9KSB7XG4gIGNvbnN0IGRldlRvb2xzID0gZ2V0RGV2VG9vbHMoKTtcbiAgaWYgKCFkZXZUb29scykge1xuICAgIHJldHVybjtcbiAgfVxuXG4gIGNvbnN0IG5hbWUgPSBvcHRpb25zLm5hbWUgfHwgXCJGbHV4U3RvcmVcIjtcbiAgY29uc3QgZHQgPSBkZXZUb29scy5jb25uZWN0KHsgbmFtZSwgLi4ub3B0aW9ucyB9KTtcbiAgZHQuaW5pdChzdG9yZS5zdGF0ZSk7XG5cbiAgc3RvcmUub24oXCJjb21taXRcIiwgKF86IGFueSwgbXV0YXRpb246IHN0cmluZywgLi4uYXJnczogYW55W10pID0+IHtcbiAgICBkdC5zZW5kKHsgdHlwZTogbXV0YXRpb24sIHBheWxvYWQ6IGFyZ3MgfSwgc3RvcmUuc3RhdGUpO1xuICB9KTtcblxuICBzdG9yZS5vbihcInJlZ2lzdGVyTW9kdWxlXCIsIChfOiBhbnksIG5hbWVzcGFjZTogc3RyaW5nKSA9PiB7XG4gICAgZHQuc2VuZCh7IHR5cGU6IGBbTW9kdWxlXSBSZWdpc3RlcjogJHtuYW1lc3BhY2V9YCB9LCBzdG9yZS5zdGF0ZSk7XG4gIH0pO1xuXG4gIHN0b3JlLm9uKFwidW5yZWdpc3Rlck1vZHVsZVwiLCAoXzogYW55LCBuYW1lc3BhY2U6IHN0cmluZykgPT4ge1xuICAgIGR0LnNlbmQoeyB0eXBlOiBgW01vZHVsZV0gVW5yZWdpc3RlcjogJHtuYW1lc3BhY2V9YCB9LCBzdG9yZS5zdGF0ZSk7XG4gIH0pO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY29ubmVjdFB1bHNlU3RvcmUoc3RvcmU6IGFueSwgb3B0aW9uczogRGV2VG9vbHNPcHRpb25zID0ge30pIHtcbiAgY29uc3QgZGV2VG9vbHMgPSBnZXREZXZUb29scygpO1xuICBpZiAoIWRldlRvb2xzKSB7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgY29uc3QgbmFtZSA9IG9wdGlvbnMubmFtZSB8fCBcIlB1bHNlU3RvcmVcIjtcbiAgY29uc3QgZHQgPSBkZXZUb29scy5jb25uZWN0KHsgbmFtZSwgLi4ub3B0aW9ucyB9KTtcbiAgZHQuaW5pdChzdG9yZS5zdGF0ZSk7XG5cbiAgaWYgKHN0b3JlLm9uKSB7XG4gICAgc3RvcmUub24oXCJwdWxzZVwiLCAocHVsc2U6IHN0cmluZywgYXJnczogYW55W10pID0+IHtcbiAgICAgIGR0LnNlbmQoeyB0eXBlOiBwdWxzZSwgcGF5bG9hZDogYXJncyB9LCBzdG9yZS5zdGF0ZSk7XG4gICAgfSk7XG4gIH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNvbm5lY3RQdWxzZShwdWxzZTogYW55LCBvcHRpb25zOiBEZXZUb29sc09wdGlvbnMgPSB7fSkge1xuICAvLyBQdWxzZSBpcyBbcmVhZCwgd3JpdGUsIHJ1blN1YnNjcmliZXJzXVxuICAvLyBXZSBjYW4ndCBlYXNpbHkgaG9vayBpbnRvIHRoZSB3cml0ZSBmdW5jdGlvbiB3aXRob3V0IHdyYXBwaW5nIGl0LlxuICAvLyBCdXQgdGhlIHVzZXIgYXNrZWQgZm9yIFwic2ltcGxlXCIgYW5kIFwidXNpbmcgZXhpc3RpbmcgYXBpc1wiLlxuICAvLyBJZiB3ZSB3YW50IHRvIGRlYnVnIGluZGl2aWR1YWwgcHVsc2VzLCB3ZSBtaWdodCBuZWVkIHRvIHdyYXAgdGhlbS5cblxuICBjb25zdCBkZXZUb29scyA9IGdldERldlRvb2xzKCk7XG4gIGlmICghZGV2VG9vbHMpIHtcbiAgICByZXR1cm4gcHVsc2U7XG4gIH1cblxuICBjb25zdCBuYW1lID0gb3B0aW9ucy5uYW1lIHx8IFwiUHVsc2VcIjtcbiAgY29uc3QgZHQgPSBkZXZUb29scy5jb25uZWN0KHsgbmFtZSwgLi4ub3B0aW9ucyB9KTtcbiAgY29uc3QgW3JlYWQsIHdyaXRlLCBydW5dID0gcHVsc2U7XG4gIFxuICBkdC5pbml0KHJlYWQoKSk7XG5cbiAgY29uc3QgbmV3V3JpdGUgPSAobmV3VmFsdWU6IGFueSkgPT4ge1xuICAgIHdyaXRlKG5ld1ZhbHVlKTtcbiAgICBkdC5zZW5kKHsgdHlwZTogXCJ1cGRhdGVcIiwgcGF5bG9hZDogbmV3VmFsdWUgfSwgcmVhZCgpKTtcbiAgfTtcblxuICByZXR1cm4gW3JlYWQsIG5ld1dyaXRlLCBydW5dO1xufVxuIl0sIm1hcHBpbmdzIjoiZ0pBQUFBLEVBQUEsRyx5REFBQUMsQ0FBQUQsRUFBQSxDQUFBRSxpQkFBQSxJQUFBQSxFQUFBQyxhQUFBLElBQUFBLEVBQUFDLGtCQUFBLElBQUFBLElBQUEsSSxFQUFBQyxHLEVBQUFMLEUsME1BY0EsU0FBU00sSUFFUCxNQURvQyxvQkFBWEMsUUFDUkEsT0FBT0MsNkJBQ2ZELE9BQU9DLDZCQUVULElBQ1QsQ0FFTyxTQUFTTixFQUFpQk8sRUFBa0JDLEVBQTJCLENBQUMsR0FDN0UsTUFBTUMsRUFBV0wsSUFDakIsSUFBS0ssRUFDSCxPQUdGLE1BQU1DLEVBQU9GLEVBQVFFLE1BQVEsWUFDdkJDLEVBQUtGLEVBQVNHLFFBQVEsQ0FBRUYsVUFBU0YsSUFDdkNHLEVBQUdFLEtBQUtOLEVBQU1PLE9BRWRQLEVBQU1RLEdBQUcsU0FBVSxDQUFDQyxFQUFRQyxLQUFxQkMsS0FDL0NQLEVBQUdRLEtBQUssQ0FBRUMsS0FBTUgsRUFBVUksUUFBU0gsR0FBUVgsRUFBTU8sU0FHbkRQLEVBQU1RLEdBQUcsaUJBQWtCLENBQUNDLEVBQVFNLEtBQ2xDWCxFQUFHUSxLQUFLLENBQUVDLEtBQU0sc0JBQXNCRSxLQUFlZixFQUFNTyxTQUc3RFAsRUFBTVEsR0FBRyxtQkFBb0IsQ0FBQ0MsRUFBUU0sS0FDcENYLEVBQUdRLEtBQUssQ0FBRUMsS0FBTSx3QkFBd0JFLEtBQWVmLEVBQU1PLFFBRWpFLENBRU8sU0FBU1osRUFBa0JLLEVBQVlDLEVBQTJCLENBQUMsR0FDeEUsTUFBTUMsRUFBV0wsSUFDakIsSUFBS0ssRUFDSCxPQUdGLE1BQU1DLEVBQU9GLEVBQVFFLE1BQVEsYUFDdkJDLEVBQUtGLEVBQVNHLFFBQVEsQ0FBRUYsVUFBU0YsSUFDdkNHLEVBQUdFLEtBQUtOLEVBQU1PLE9BRVZQLEVBQU1RLElBQ1JSLEVBQU1RLEdBQUcsUUFBUyxDQUFDUSxFQUFlTCxLQUNoQ1AsRUFBR1EsS0FBSyxDQUFFQyxLQUFNRyxFQUFPRixRQUFTSCxHQUFRWCxFQUFNTyxRQUdwRCxDQUVPLFNBQVNiLEVBQWFzQixFQUFZZixFQUEyQixDQUFDLEdBTW5FLE1BQU1DLEVBQVdMLElBQ2pCLElBQUtLLEVBQ0gsT0FBT2MsRUFHVCxNQUFNYixFQUFPRixFQUFRRSxNQUFRLFFBQ3ZCQyxFQUFLRixFQUFTRyxRQUFRLENBQUVGLFVBQVNGLEtBQ2hDZ0IsRUFBTUMsRUFBT0MsR0FBT0gsRUFFM0JaLEVBQUdFLEtBQUtXLEtBT1IsTUFBTyxDQUFDQSxFQUxVRyxJQUNoQkYsRUFBTUUsR0FDTmhCLEVBQUdRLEtBQUssQ0FBRUMsS0FBTSxTQUFVQyxRQUFTTSxHQUFZSCxNQUd6QkUsRUFDMUIsQyJ9
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
// lib/redux-devtools/index.ts
|
|
2
|
+
function getDevTools() {
|
|
3
|
+
const isBrowser = typeof window !== "undefined";
|
|
4
|
+
if (isBrowser && window.__REDUX_DEVTOOLS_EXTENSION__) {
|
|
5
|
+
return window.__REDUX_DEVTOOLS_EXTENSION__;
|
|
6
|
+
}
|
|
7
|
+
return null;
|
|
8
|
+
}
|
|
9
|
+
function connectFluxStore(store, options = {}) {
|
|
10
|
+
const devTools = getDevTools();
|
|
11
|
+
if (!devTools) {
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
const name = options.name || "FluxStore";
|
|
15
|
+
const dt = devTools.connect({ name, ...options });
|
|
16
|
+
dt.init(store.state);
|
|
17
|
+
store.on("commit", (_, mutation, ...args) => {
|
|
18
|
+
dt.send({ type: mutation, payload: args }, store.state);
|
|
19
|
+
});
|
|
20
|
+
store.on("registerModule", (_, namespace) => {
|
|
21
|
+
dt.send({ type: `[Module] Register: ${namespace}` }, store.state);
|
|
22
|
+
});
|
|
23
|
+
store.on("unregisterModule", (_, namespace) => {
|
|
24
|
+
dt.send({ type: `[Module] Unregister: ${namespace}` }, store.state);
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
function connectPulseStore(store, options = {}) {
|
|
28
|
+
const devTools = getDevTools();
|
|
29
|
+
if (!devTools) {
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
const name = options.name || "PulseStore";
|
|
33
|
+
const dt = devTools.connect({ name, ...options });
|
|
34
|
+
dt.init(store.state);
|
|
35
|
+
if (store.on) {
|
|
36
|
+
store.on("pulse", (pulse, args) => {
|
|
37
|
+
dt.send({ type: pulse, payload: args }, store.state);
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
function connectPulse(pulse, options = {}) {
|
|
42
|
+
const devTools = getDevTools();
|
|
43
|
+
if (!devTools) {
|
|
44
|
+
return pulse;
|
|
45
|
+
}
|
|
46
|
+
const name = options.name || "Pulse";
|
|
47
|
+
const dt = devTools.connect({ name, ...options });
|
|
48
|
+
const [read, write, run] = pulse;
|
|
49
|
+
dt.init(read());
|
|
50
|
+
const newWrite = (newValue) => {
|
|
51
|
+
write(newValue);
|
|
52
|
+
dt.send({ type: "update", payload: newValue }, read());
|
|
53
|
+
};
|
|
54
|
+
return [read, newWrite, run];
|
|
55
|
+
}
|
|
56
|
+
export {
|
|
57
|
+
connectFluxStore,
|
|
58
|
+
connectPulse,
|
|
59
|
+
connectPulseStore
|
|
60
|
+
};
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../lib/redux-devtools/index.ts"],
|
|
4
|
+
"sourcesContent": ["/* eslint-disable no-console */\nimport { FluxStore } from \"valyrian.js/flux-store\";\n\ndeclare global {\n interface Window {\n __REDUX_DEVTOOLS_EXTENSION__: any;\n }\n}\n\ninterface DevToolsOptions {\n name?: string;\n [key: string]: any;\n}\n\nfunction getDevTools() {\n const isBrowser = typeof window !== \"undefined\";\n if (isBrowser && window.__REDUX_DEVTOOLS_EXTENSION__) {\n return window.__REDUX_DEVTOOLS_EXTENSION__;\n }\n return null;\n}\n\nexport function connectFluxStore(store: FluxStore, options: DevToolsOptions = {}) {\n const devTools = getDevTools();\n if (!devTools) {\n return;\n }\n\n const name = options.name || \"FluxStore\";\n const dt = devTools.connect({ name, ...options });\n dt.init(store.state);\n\n store.on(\"commit\", (_: any, mutation: string, ...args: any[]) => {\n dt.send({ type: mutation, payload: args }, store.state);\n });\n\n store.on(\"registerModule\", (_: any, namespace: string) => {\n dt.send({ type: `[Module] Register: ${namespace}` }, store.state);\n });\n\n store.on(\"unregisterModule\", (_: any, namespace: string) => {\n dt.send({ type: `[Module] Unregister: ${namespace}` }, store.state);\n });\n}\n\nexport function connectPulseStore(store: any, options: DevToolsOptions = {}) {\n const devTools = getDevTools();\n if (!devTools) {\n return;\n }\n\n const name = options.name || \"PulseStore\";\n const dt = devTools.connect({ name, ...options });\n dt.init(store.state);\n\n if (store.on) {\n store.on(\"pulse\", (pulse: string, args: any[]) => {\n dt.send({ type: pulse, payload: args }, store.state);\n });\n }\n}\n\nexport function connectPulse(pulse: any, options: DevToolsOptions = {}) {\n // Pulse is [read, write, runSubscribers]\n // We can't easily hook into the write function without wrapping it.\n // But the user asked for \"simple\" and \"using existing apis\".\n // If we want to debug individual pulses, we might need to wrap them.\n\n const devTools = getDevTools();\n if (!devTools) {\n return pulse;\n }\n\n const name = options.name || \"Pulse\";\n const dt = devTools.connect({ name, ...options });\n const [read, write, run] = pulse;\n \n dt.init(read());\n\n const newWrite = (newValue: any) => {\n write(newValue);\n dt.send({ type: \"update\", payload: newValue }, read());\n };\n\n return [read, newWrite, run];\n}\n"],
|
|
5
|
+
"mappings": ";AAcA,SAAS,cAAc;AACrB,QAAM,YAAY,OAAO,WAAW;AACpC,MAAI,aAAa,OAAO,8BAA8B;AACpD,WAAO,OAAO;AAAA,EAChB;AACA,SAAO;AACT;AAEO,SAAS,iBAAiB,OAAkB,UAA2B,CAAC,GAAG;AAChF,QAAM,WAAW,YAAY;AAC7B,MAAI,CAAC,UAAU;AACb;AAAA,EACF;AAEA,QAAM,OAAO,QAAQ,QAAQ;AAC7B,QAAM,KAAK,SAAS,QAAQ,EAAE,MAAM,GAAG,QAAQ,CAAC;AAChD,KAAG,KAAK,MAAM,KAAK;AAEnB,QAAM,GAAG,UAAU,CAAC,GAAQ,aAAqB,SAAgB;AAC/D,OAAG,KAAK,EAAE,MAAM,UAAU,SAAS,KAAK,GAAG,MAAM,KAAK;AAAA,EACxD,CAAC;AAED,QAAM,GAAG,kBAAkB,CAAC,GAAQ,cAAsB;AACxD,OAAG,KAAK,EAAE,MAAM,sBAAsB,SAAS,GAAG,GAAG,MAAM,KAAK;AAAA,EAClE,CAAC;AAED,QAAM,GAAG,oBAAoB,CAAC,GAAQ,cAAsB;AAC1D,OAAG,KAAK,EAAE,MAAM,wBAAwB,SAAS,GAAG,GAAG,MAAM,KAAK;AAAA,EACpE,CAAC;AACH;AAEO,SAAS,kBAAkB,OAAY,UAA2B,CAAC,GAAG;AAC3E,QAAM,WAAW,YAAY;AAC7B,MAAI,CAAC,UAAU;AACb;AAAA,EACF;AAEA,QAAM,OAAO,QAAQ,QAAQ;AAC7B,QAAM,KAAK,SAAS,QAAQ,EAAE,MAAM,GAAG,QAAQ,CAAC;AAChD,KAAG,KAAK,MAAM,KAAK;AAEnB,MAAI,MAAM,IAAI;AACZ,UAAM,GAAG,SAAS,CAAC,OAAe,SAAgB;AAChD,SAAG,KAAK,EAAE,MAAM,OAAO,SAAS,KAAK,GAAG,MAAM,KAAK;AAAA,IACrD,CAAC;AAAA,EACH;AACF;AAEO,SAAS,aAAa,OAAY,UAA2B,CAAC,GAAG;AAMtE,QAAM,WAAW,YAAY;AAC7B,MAAI,CAAC,UAAU;AACb,WAAO;AAAA,EACT;AAEA,QAAM,OAAO,QAAQ,QAAQ;AAC7B,QAAM,KAAK,SAAS,QAAQ,EAAE,MAAM,GAAG,QAAQ,CAAC;AAChD,QAAM,CAAC,MAAM,OAAO,GAAG,IAAI;AAE3B,KAAG,KAAK,KAAK,CAAC;AAEd,QAAM,WAAW,CAAC,aAAkB;AAClC,UAAM,QAAQ;AACd,OAAG,KAAK,EAAE,MAAM,UAAU,SAAS,SAAS,GAAG,KAAK,CAAC;AAAA,EACvD;AAEA,SAAO,CAAC,MAAM,UAAU,GAAG;AAC7B;",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
package/lib/pulses/index.ts
CHANGED
|
@@ -56,7 +56,11 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
56
56
|
initialState: StateType | (() => StateType) | null,
|
|
57
57
|
pulses: PulsesType,
|
|
58
58
|
immutable = false
|
|
59
|
-
): StorePulses<PulsesType> & {
|
|
59
|
+
): StorePulses<PulsesType> & {
|
|
60
|
+
state: ProxyState<StateType>;
|
|
61
|
+
on: (event: string, callback: Function) => void;
|
|
62
|
+
off: (event: string, callback: Function) => void;
|
|
63
|
+
} {
|
|
60
64
|
const subscribers = new Set<Function>();
|
|
61
65
|
const domWithVnodesToUpdate = new WeakSet<DomElement>();
|
|
62
66
|
|
|
@@ -160,7 +164,7 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
160
164
|
const emptyFlush = async () => {};
|
|
161
165
|
|
|
162
166
|
try {
|
|
163
|
-
const pulseResult = pulses[key].apply(this, [currentState, ...args]);
|
|
167
|
+
const pulseResult = pulses[key].apply(this, [currentState, ...args] as any);
|
|
164
168
|
if (pulseResult instanceof Promise) {
|
|
165
169
|
return pulseResult
|
|
166
170
|
.then((resolvedValue) => {
|
|
@@ -190,38 +194,82 @@ function createStore<StateType extends State, PulsesType extends Pulses<StateTyp
|
|
|
190
194
|
|
|
191
195
|
syncState(localState);
|
|
192
196
|
|
|
197
|
+
const listeners: Record<string, Function[]> = {};
|
|
198
|
+
const trigger = (event: string, ...args: any[]) => {
|
|
199
|
+
if (listeners[event]) {
|
|
200
|
+
listeners[event].forEach((callback) => callback(...args));
|
|
201
|
+
}
|
|
202
|
+
};
|
|
203
|
+
|
|
193
204
|
const pulsesProxy = new Proxy(boundPulses, {
|
|
194
205
|
get: (pulses, prop: string) => {
|
|
195
206
|
if (prop === "state") {
|
|
196
207
|
return proxyState;
|
|
197
208
|
}
|
|
209
|
+
if (prop === "on") {
|
|
210
|
+
return (event: string, callback: Function) => {
|
|
211
|
+
if (!listeners[event]) {
|
|
212
|
+
listeners[event] = [];
|
|
213
|
+
}
|
|
214
|
+
listeners[event].push(callback);
|
|
215
|
+
};
|
|
216
|
+
}
|
|
217
|
+
if (prop === "off") {
|
|
218
|
+
return (event: string, callback: Function) => {
|
|
219
|
+
if (listeners[event]) {
|
|
220
|
+
listeners[event] = listeners[event].filter((cb) => cb !== callback);
|
|
221
|
+
}
|
|
222
|
+
};
|
|
223
|
+
}
|
|
198
224
|
if (!(prop in pulses)) {
|
|
199
225
|
throw new Error(`Pulse '${prop}' does not exist`);
|
|
200
226
|
}
|
|
201
227
|
const pulseMethod = pulses[prop];
|
|
202
228
|
|
|
203
229
|
if (typeof pulseMethod === "function") {
|
|
204
|
-
return
|
|
230
|
+
return (...args: any[]) => {
|
|
231
|
+
const result = pulseMethod.apply(pulsesProxy, args as any);
|
|
232
|
+
if (result instanceof Promise) {
|
|
233
|
+
return result.then((r) => {
|
|
234
|
+
trigger("pulse", prop, args);
|
|
235
|
+
return r;
|
|
236
|
+
});
|
|
237
|
+
}
|
|
238
|
+
trigger("pulse", prop, args);
|
|
239
|
+
return result;
|
|
240
|
+
};
|
|
205
241
|
}
|
|
206
242
|
|
|
207
243
|
return pulseMethod;
|
|
208
244
|
}
|
|
209
245
|
});
|
|
210
246
|
|
|
211
|
-
return pulsesProxy as StorePulses<PulsesType> & {
|
|
247
|
+
return pulsesProxy as StorePulses<PulsesType> & {
|
|
248
|
+
state: ProxyState<StateType>;
|
|
249
|
+
on: (event: string, callback: Function) => void;
|
|
250
|
+
off: (event: string, callback: Function) => void;
|
|
251
|
+
};
|
|
212
252
|
}
|
|
213
253
|
|
|
214
254
|
export function createPulseStore<StateType extends State, PulsesType extends Pulses<StateType>>(
|
|
215
255
|
initialState: StateType,
|
|
216
256
|
pulses: PulsesType
|
|
217
|
-
): StorePulses<PulsesType> & {
|
|
257
|
+
): StorePulses<PulsesType> & {
|
|
258
|
+
state: ProxyState<StateType>;
|
|
259
|
+
on: (event: string, callback: Function) => void;
|
|
260
|
+
off: (event: string, callback: Function) => void;
|
|
261
|
+
} {
|
|
218
262
|
return createStore(initialState, pulses, true);
|
|
219
263
|
}
|
|
220
264
|
|
|
221
265
|
export function createMutableStore<StateType extends State, PulsesType extends Pulses<StateType>>(
|
|
222
266
|
initialState: StateType,
|
|
223
267
|
pulses: PulsesType
|
|
224
|
-
): StorePulses<PulsesType> & {
|
|
268
|
+
): StorePulses<PulsesType> & {
|
|
269
|
+
state: ProxyState<StateType>;
|
|
270
|
+
on: (event: string, callback: Function) => void;
|
|
271
|
+
off: (event: string, callback: Function) => void;
|
|
272
|
+
} {
|
|
225
273
|
console.warn(
|
|
226
274
|
"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."
|
|
227
275
|
);
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
/* eslint-disable no-console */
|
|
2
|
+
import { FluxStore } from "valyrian.js/flux-store";
|
|
3
|
+
|
|
4
|
+
declare global {
|
|
5
|
+
interface Window {
|
|
6
|
+
__REDUX_DEVTOOLS_EXTENSION__: any;
|
|
7
|
+
}
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
interface DevToolsOptions {
|
|
11
|
+
name?: string;
|
|
12
|
+
[key: string]: any;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
function getDevTools() {
|
|
16
|
+
const isBrowser = typeof window !== "undefined";
|
|
17
|
+
if (isBrowser && window.__REDUX_DEVTOOLS_EXTENSION__) {
|
|
18
|
+
return window.__REDUX_DEVTOOLS_EXTENSION__;
|
|
19
|
+
}
|
|
20
|
+
return null;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export function connectFluxStore(store: FluxStore, options: DevToolsOptions = {}) {
|
|
24
|
+
const devTools = getDevTools();
|
|
25
|
+
if (!devTools) {
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
const name = options.name || "FluxStore";
|
|
30
|
+
const dt = devTools.connect({ name, ...options });
|
|
31
|
+
dt.init(store.state);
|
|
32
|
+
|
|
33
|
+
store.on("commit", (_: any, mutation: string, ...args: any[]) => {
|
|
34
|
+
dt.send({ type: mutation, payload: args }, store.state);
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
store.on("registerModule", (_: any, namespace: string) => {
|
|
38
|
+
dt.send({ type: `[Module] Register: ${namespace}` }, store.state);
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
store.on("unregisterModule", (_: any, namespace: string) => {
|
|
42
|
+
dt.send({ type: `[Module] Unregister: ${namespace}` }, store.state);
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export function connectPulseStore(store: any, options: DevToolsOptions = {}) {
|
|
47
|
+
const devTools = getDevTools();
|
|
48
|
+
if (!devTools) {
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
const name = options.name || "PulseStore";
|
|
53
|
+
const dt = devTools.connect({ name, ...options });
|
|
54
|
+
dt.init(store.state);
|
|
55
|
+
|
|
56
|
+
if (store.on) {
|
|
57
|
+
store.on("pulse", (pulse: string, args: any[]) => {
|
|
58
|
+
dt.send({ type: pulse, payload: args }, store.state);
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export function connectPulse(pulse: any, options: DevToolsOptions = {}) {
|
|
64
|
+
// Pulse is [read, write, runSubscribers]
|
|
65
|
+
// We can't easily hook into the write function without wrapping it.
|
|
66
|
+
// But the user asked for "simple" and "using existing apis".
|
|
67
|
+
// If we want to debug individual pulses, we might need to wrap them.
|
|
68
|
+
|
|
69
|
+
const devTools = getDevTools();
|
|
70
|
+
if (!devTools) {
|
|
71
|
+
return pulse;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
const name = options.name || "Pulse";
|
|
75
|
+
const dt = devTools.connect({ name, ...options });
|
|
76
|
+
const [read, write, run] = pulse;
|
|
77
|
+
|
|
78
|
+
dt.init(read());
|
|
79
|
+
|
|
80
|
+
const newWrite = (newValue: any) => {
|
|
81
|
+
write(newValue);
|
|
82
|
+
dt.send({ type: "update", payload: newValue }, read());
|
|
83
|
+
};
|
|
84
|
+
|
|
85
|
+
return [read, newWrite, run];
|
|
86
|
+
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "valyrian.js",
|
|
3
|
-
"version": "8.0
|
|
3
|
+
"version": "8.1.0",
|
|
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>",
|