valyrian.js 8.1.8 → 8.1.9
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.map +1 -1
- package/dist/pulses/index.js +26 -17
- 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 +26 -17
- package/dist/pulses/index.mjs.map +2 -2
- package/lib/pulses/index.ts +33 -19
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/pulses/index.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/pulses/index.ts"],"names":[],"mappings":"AAKA,KAAK,KAAK,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;AAEjC,MAAM,MAAM,YAAY,GAAG;IACzB,MAAM,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC7B,CAAC;AAEF,MAAM,MAAM,KAAK,CAAC,SAAS,EAAE,OAAO,GAAG,OAAO,IAAI,CAAC,KAAK,EAAE,SAAS,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE,KAAK,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;AAEnH,KAAK,UAAU,CAAC,SAAS,IAAI,SAAS,GAAG;IAAE,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAA;CAAE,CAAC;AAIhE,KAAK,WAAW,CAAC,UAAU,IAAI;KAC5B,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;AAiPF,wBAAgB,gBAAgB,CAAC,SAAS,SAAS,KAAK,EAAE,UAAU,SAAS,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,EAChH,YAAY,EAAE,SAAS,EACvB,MAAM,EAAE,UAAU,GAAG,QAAQ,CAAC,UAAU,GAAG,YAAY,CAAC;;QA7MpD,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;SAC1C,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;EA+MjD;AAED,wBAAgB,kBAAkB,CAAC,SAAS,SAAS,KAAK,EAAE,UAAU,SAAS,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,EAClH,YAAY,EAAE,SAAS,EACvB,MAAM,EAAE,UAAU,GAAG,QAAQ,CAAC,UAAU,GAAG,YAAY,CAAC;;QApNpD,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;SAC1C,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,KAAK,IAAI;EAyNjD;AAED,wBAAgB,YAAY,CAAC,MAAM,EAAE,QAAQ,QAU5C;AAED,wBAAgB,WAAW,CAAC,CAAC,EAAE,YAAY,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,IAAI,EAAE,MAAM,IAAI,CAAC,CA4BlH"}
|
package/dist/pulses/index.js
CHANGED
|
@@ -126,46 +126,55 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
126
126
|
return;
|
|
127
127
|
}
|
|
128
128
|
syncState(newState);
|
|
129
|
-
|
|
129
|
+
if (!flush) {
|
|
130
|
+
currentState = null;
|
|
131
|
+
}
|
|
132
|
+
pulseCallCount = 0;
|
|
130
133
|
debouncedUpdate();
|
|
131
134
|
}
|
|
135
|
+
function unfreezeState() {
|
|
136
|
+
if (currentState === null) {
|
|
137
|
+
currentState = (0, import_utils.deepCloneUnfreeze)(localState);
|
|
138
|
+
}
|
|
139
|
+
return currentState;
|
|
140
|
+
}
|
|
132
141
|
function getPulseMethod(key) {
|
|
133
142
|
return function(...args) {
|
|
134
143
|
pulseCallCount++;
|
|
135
|
-
|
|
136
|
-
currentState = (0, import_utils.deepCloneUnfreeze)(localState);
|
|
137
|
-
}
|
|
144
|
+
const state = unfreezeState();
|
|
138
145
|
const context = Object.create(pulses);
|
|
139
146
|
context.$flush = async () => {
|
|
140
147
|
if (currentState) {
|
|
141
148
|
setState(currentState, true);
|
|
142
|
-
currentState = (0, import_utils.deepCloneUnfreeze)(localState);
|
|
143
149
|
}
|
|
144
|
-
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
145
150
|
};
|
|
146
151
|
const emptyFlush = async () => {
|
|
147
152
|
};
|
|
153
|
+
const handleError = (error) => {
|
|
154
|
+
console.error(`Error in pulse '${key}':`, error);
|
|
155
|
+
context.$flush = emptyFlush;
|
|
156
|
+
pulseCallCount--;
|
|
157
|
+
if (pulseCallCount <= 0) {
|
|
158
|
+
currentState = null;
|
|
159
|
+
pulseCallCount = 0;
|
|
160
|
+
}
|
|
161
|
+
throw error;
|
|
162
|
+
};
|
|
148
163
|
try {
|
|
149
|
-
const pulseResult = pulses[key].apply(context, [
|
|
164
|
+
const pulseResult = pulses[key].apply(context, [state, ...args]);
|
|
150
165
|
if (pulseResult instanceof Promise) {
|
|
151
166
|
return pulseResult.then((resolvedValue) => {
|
|
152
|
-
setState(
|
|
167
|
+
setState(state);
|
|
153
168
|
context.$flush = emptyFlush;
|
|
154
169
|
return resolvedValue;
|
|
155
|
-
}).catch(
|
|
156
|
-
console.error(`Error in pulse '${key}':`, error);
|
|
157
|
-
context.$flush = emptyFlush;
|
|
158
|
-
throw error;
|
|
159
|
-
});
|
|
170
|
+
}).catch(handleError);
|
|
160
171
|
} else {
|
|
161
|
-
setState(
|
|
172
|
+
setState(state);
|
|
162
173
|
context.$flush = emptyFlush;
|
|
163
174
|
return pulseResult;
|
|
164
175
|
}
|
|
165
176
|
} catch (error) {
|
|
166
|
-
|
|
167
|
-
context.$flush = emptyFlush;
|
|
168
|
-
throw error;
|
|
177
|
+
handleError(error);
|
|
169
178
|
}
|
|
170
179
|
};
|
|
171
180
|
}
|
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-
|
|
5
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;
|
|
4
|
+
"sourcesContent": ["/* eslint-disable @typescript-eslint/no-unsafe-function-type */\n\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type PulseContext = {\n $flush: () => Promise<void>;\n};\n\nexport type Pulse<StateType, TReturn = unknown> = (state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType> = {\n [K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R\n ? (...args: Args) => R\n : never;\n};\n\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n const currentVnode = current.vnode as VnodeWithDom;\n if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n return;\n }\n\n let hasParent = false;\n let parent = currentVnode.dom.parentElement as DomElement;\n while (parent) {\n if (domWithVnodesToUpdate.has(parent)) {\n hasParent = true;\n break;\n }\n parent = parent.parentElement as DomElement;\n }\n\n if (!hasParent) {\n const dom = currentVnode.dom;\n const subscription = () => {\n updateVnode(dom.vnode);\n if (!dom.parentElement) {\n subscribers.delete(subscription);\n domWithVnodesToUpdate.delete(dom);\n }\n };\n subscribers.add(subscription);\n domWithVnodesToUpdate.add(dom);\n }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n initialState: StateType | (() => StateType) | null,\n pulses: PulsesType & ThisType<PulsesType & PulseContext>,\n immutable = false\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const boundPulses: Record<string, Function> = {};\n for (const key in pulses) {\n if (typeof pulses[key] !== \"function\") {\n throw new Error(`Pulse '${key}' must be a function`);\n }\n if (key === \"state\") {\n throw new Error(`A pulse cannot be named 'state'`);\n }\n boundPulses[key] = getPulseMethod(key);\n }\n\n const localState: StateType =\n (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n function isMutable() {\n if (immutable) {\n throw new Error(\"You need to call a pulse to modify the state\");\n }\n }\n\n let currentState: StateType | null = null;\n let pulseCallCount = 0;\n\n const proxyState = new Proxy(localState, {\n get: (state, prop: string) => {\n if (currentState) {\n return currentState[prop];\n }\n\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n return state[prop];\n },\n set: (state, prop: string, value: any) => {\n isMutable();\n Reflect.set(state, prop, value);\n return true;\n },\n deleteProperty: (state, prop: string) => {\n isMutable();\n Reflect.deleteProperty(state, prop);\n return true;\n }\n });\n\n function syncState(newState: StateType) {\n for (const key in newState) {\n localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n }\n for (const key in localState) {\n if (!(key in newState)) {\n Reflect.deleteProperty(localState, key);\n }\n }\n }\n\n let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n function debouncedUpdate() {\n if (debounceTimeout) {\n clearTimeout(debounceTimeout);\n }\n debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n }\n\n function setState(newState: StateType, flush = false) {\n pulseCallCount--;\n if (!hasChanged(localState, newState)) {\n return;\n }\n if (pulseCallCount > 0 && !flush) {\n return;\n }\n syncState(newState);\n if (!flush) {\n currentState = null;\n }\n pulseCallCount = 0;\n debouncedUpdate();\n }\n\n function unfreezeState() {\n if (currentState === null) {\n currentState = deepCloneUnfreeze(localState);\n }\n return currentState;\n }\n\n function getPulseMethod(key: string) {\n return function (this: any, ...args: any[]) {\n pulseCallCount++;\n\n const state = unfreezeState();\n\n const context = Object.create(pulses);\n\n context.$flush = async () => {\n if (currentState) {\n setState(currentState, true);\n }\n };\n\n const emptyFlush = async () => {};\n\n const handleError = (error: any) => {\n console.error(`Error in pulse '${key}':`, error);\n context.$flush = emptyFlush;\n\n pulseCallCount--;\n if (pulseCallCount <= 0) {\n currentState = null;\n pulseCallCount = 0;\n }\n\n throw error;\n };\n\n try {\n const pulseResult = pulses[key].apply(context, [state, ...args]);\n if (pulseResult instanceof Promise) {\n return pulseResult\n .then((resolvedValue) => {\n setState(state as StateType);\n context.$flush = emptyFlush;\n return resolvedValue;\n })\n .catch(handleError);\n } else {\n setState(state);\n context.$flush = emptyFlush;\n return pulseResult;\n }\n } catch (error) {\n handleError(error);\n }\n };\n }\n\n syncState(localState);\n\n const listeners: Record<string, Function[]> = {};\n const trigger = (event: string, ...args: any[]) => {\n if (listeners[event]) {\n listeners[event].forEach((callback) => callback(...args));\n }\n };\n\n const pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\n }\n if (prop === \"on\") {\n return (event: string, callback: Function) => {\n if (!listeners[event]) {\n listeners[event] = [];\n }\n listeners[event].push(callback);\n };\n }\n if (prop === \"off\") {\n return (event: string, callback: Function) => {\n if (listeners[event]) {\n listeners[event] = listeners[event].filter((cb) => cb !== callback);\n }\n };\n }\n if (!(prop in pulses)) {\n throw new Error(`Pulse '${prop}' does not exist`);\n }\n const pulseMethod = pulses[prop];\n\n return (...args: any[]) => {\n const result = pulseMethod.apply(pulseMethod, args);\n if (result instanceof Promise) {\n return result.then((r) => {\n trigger(\"pulse\", prop, args);\n return r;\n });\n }\n trigger(\"pulse\", prop, args);\n return result;\n };\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n initialState: StateType,\n pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n initialState: StateType,\n pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n console.warn(\n \"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render.\"\n );\n return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n const runEffect = () => {\n try {\n effectStack.push(runEffect);\n effect();\n } finally {\n effectStack.pop();\n }\n };\n runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n let value = initialValue;\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const runSubscribers = () => {\n subscribers.forEach((subscriber) => subscriber());\n };\n\n const read = (): T => {\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n return value;\n };\n\n const write = (newValue: T | ((current: T) => T)): void => {\n const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n if (!hasChanged(value, resolvedValue)) {\n return;\n }\n value = resolvedValue;\n runSubscribers();\n };\n\n return [read, write, runSubscribers];\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["pulses"]
|
|
7
7
|
}
|
package/dist/pulses/index.min.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
(()=>{"use strict";var e=Object.defineProperty,t=Object.getOwnPropertyDescriptor,n=Object.getOwnPropertyNames,r=Object.prototype.hasOwnProperty,o={};((t,n)=>{for(var r in n)e(t,r,{get:n[r],enumerable:!0})})(o,{createEffect:()=>h,createMutableStore:()=>
|
|
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:()=>h,createMutableStore:()=>p,createPulse:()=>y,createPulseStore:()=>d});var s,u=(s=o,((o,s,u,a)=>{if(s&&"object"==typeof s||"function"==typeof s)for(let l of n(s))r.call(o,l)||l===u||e(o,l,{get:()=>s[l],enumerable:!(a=t(s,l))||a.enumerable});return o})(e({},"__esModule",{value:!0}),s)),a=require("valyrian.js"),l=require("valyrian.js/utils"),c=[];function f(e,t){const n=a.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,a.updateVnode)(r.vnode),r.parentElement||(e.delete(o),t.delete(r))};e.add(o),t.add(r)}}function i(e,t,n=!1){const r=new Set,o=new WeakSet,s={};for(const e in t){if("function"!=typeof t[e])throw new Error(`Pulse '${e}' must be a function`);if("state"===e)throw new Error("A pulse cannot be named 'state'");s[e]=m(e)}const u=("function"==typeof e?e():e)||{};function a(){if(n)throw new Error("You need to call a pulse to modify the state")}let i=null,d=0;const p=new Proxy(u,{get:(e,t)=>{if(i)return i[t];const n=c[c.length-1];return n&&!r.has(n)&&r.add(n),f(r,o),e[t]},set:(e,t,n)=>(a(),Reflect.set(e,t,n),!0),deleteProperty:(e,t)=>(a(),Reflect.deleteProperty(e,t),!0)});function h(e){for(const t in e)u[t]=n?(0,l.deepFreeze)(e[t]):e[t];for(const t in u)t in e||Reflect.deleteProperty(u,t)}let y=null;function w(e,t=!1){d--,(0,l.hasChanged)(u,e)&&(d>0&&!t||(h(e),t||(i=null),d=0,y&&clearTimeout(y),y=setTimeout(()=>r.forEach(e=>e()),0)))}function m(e){return function(...n){d++;const r=(null===i&&(i=(0,l.deepCloneUnfreeze)(u)),i),o=Object.create(t);o.$flush=async()=>{i&&w(i,!0)};const s=async()=>{},a=t=>{throw console.error(`Error in pulse '${e}':`,t),o.$flush=s,d--,d<=0&&(i=null,d=0),t};try{const u=t[e].apply(o,[r,...n]);return u instanceof Promise?u.then(e=>(w(r),o.$flush=s,e)).catch(a):(w(r),o.$flush=s,u)}catch(e){a(e)}}}h(u);const P={},g=(e,...t)=>{P[e]&&P[e].forEach(e=>e(...t))};return new Proxy(s,{get:(e,t)=>{if("state"===t)return p;if("on"===t)return(e,t)=>{P[e]||(P[e]=[]),P[e].push(t)};if("off"===t)return(e,t)=>{P[e]&&(P[e]=P[e].filter(e=>e!==t))};if(!(t in e))throw new Error(`Pulse '${t}' does not exist`);const n=e[t];return(...e)=>{const r=n.apply(n,e);return r instanceof Promise?r.then(n=>(g("pulse",t,e),n)):(g("pulse",t,e),r)}}})}function d(e,t){return i(e,t,!0)}function p(e,t){return console.warn("Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render."),i(e,t,!1)}function h(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),f(n,r),t},e=>{const n="function"==typeof e?e(t):e;(0,l.hasChanged)(t,n)&&(t=n,o())},o]}"undefined"!=typeof module?module.exports=u:self.ValyrianPulses=u})();//# sourceMappingURL=index.min.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["index_exports","__export","createEffect","createMutableStore","createPulse","createPulseStore","__EXPORTS__","import_valyrian","require","import_utils","effectStack","registerDomSubscription","subscribers","domWithVnodesToUpdate","currentVnode","current","vnode","has","dom","hasParent","parent","parentElement","subscription","updateVnode","delete","add","createStore","initialState","pulses","immutable","Set","WeakSet","boundPulses","key","Error","getPulseMethod","localState","isMutable","currentState","pulseCallCount","proxyState","Proxy","get","state","prop","currentEffect","length","set","value","Reflect","deleteProperty","syncState","newState","deepFreeze","debounceTimeout","setState","flush","hasChanged","clearTimeout","setTimeout","forEach","subscriber","args","deepCloneUnfreeze","context","Object","create","$flush","async","Promise","resolve","emptyFlush","pulseResult","apply","then","resolvedValue","catch","error","console","listeners","trigger","event","callback","push","filter","cb","pulseMethod","result","r","warn","effect","runEffect","pop","initialValue","runSubscribers","newValue"],"sources":["../../lib/pulses/index.ts"],"sourcesContent":["/* eslint-disable no-console */\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type PulseContext = {\n  $flush: () => Promise<void>;\n};\n\nexport type Pulse<StateType, TReturn = unknown> = (state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType> = {\n  [K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R\n    ? (...args: Args) => R\n    : never;\n};\n\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n  const currentVnode = current.vnode as VnodeWithDom;\n  if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n    return;\n  }\n\n  let hasParent = false;\n  let parent = currentVnode.dom.parentElement as DomElement;\n  while (parent) {\n    if (domWithVnodesToUpdate.has(parent)) {\n      hasParent = true;\n      break;\n    }\n    parent = parent.parentElement as DomElement;\n  }\n\n  if (!hasParent) {\n    const dom = currentVnode.dom;\n    const subscription = () => {\n      updateVnode(dom.vnode);\n      if (!dom.parentElement) {\n        subscribers.delete(subscription);\n        domWithVnodesToUpdate.delete(dom);\n      }\n    };\n    subscribers.add(subscription);\n    domWithVnodesToUpdate.add(dom);\n  }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n  initialState: StateType | (() => StateType) | null,\n  pulses: PulsesType & ThisType<PulsesType & PulseContext>,\n  immutable = false\n): StorePulses<PulsesType> & {\n  state: ProxyState<StateType>;\n  on: (event: string, callback: Function) => void;\n  off: (event: string, callback: Function) => void;\n} {\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const boundPulses: Record<string, Function> = {};\n  for (const key in pulses) {\n    if (typeof pulses[key] !== \"function\") {\n      throw new Error(`Pulse '${key}' must be a function`);\n    }\n    if (key === \"state\") {\n      throw new Error(`A pulse cannot be named 'state'`);\n    }\n    boundPulses[key] = getPulseMethod(key);\n  }\n\n  const localState: StateType =\n    (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n  function isMutable() {\n    if (immutable) {\n      throw new Error(\"You need to call a pulse to modify the state\");\n    }\n  }\n\n  let currentState: StateType | null = null;\n  let pulseCallCount = 0;\n\n  const proxyState = new Proxy(localState, {\n    get: (state, prop: string) => {\n      if (currentState) {\n        return currentState[prop];\n      }\n\n      const currentEffect = effectStack[effectStack.length - 1];\n      if (currentEffect && !subscribers.has(currentEffect)) {\n        subscribers.add(currentEffect);\n      }\n\n      registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n      return state[prop];\n    },\n    set: (state, prop: string, value: any) => {\n      isMutable();\n      Reflect.set(state, prop, value);\n      return true;\n    },\n    deleteProperty: (state, prop: string) => {\n      isMutable();\n      Reflect.deleteProperty(state, prop);\n      return true;\n    }\n  });\n\n  function syncState(newState: StateType) {\n    for (const key in newState) {\n      localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n    }\n    for (const key in localState) {\n      if (!(key in newState)) {\n        Reflect.deleteProperty(localState, key);\n      }\n    }\n  }\n\n  let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n  function debouncedUpdate() {\n    if (debounceTimeout) {\n      clearTimeout(debounceTimeout);\n    }\n    debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n  }\n\n  function setState(newState: StateType, flush = false) {\n    pulseCallCount--;\n    if (!hasChanged(localState, newState)) {\n      return;\n    }\n    if (pulseCallCount > 0 && !flush) {\n      return;\n    }\n    syncState(newState);\n    currentState = null;\n    debouncedUpdate();\n  }\n\n  function getPulseMethod(key: string) {\n    return function (this: any, ...args: any[]) {\n      pulseCallCount++;\n      if (currentState === null) {\n        currentState = deepCloneUnfreeze(localState);\n      }\n\n      // Runtime: Creamos un objeto que hereda de 'pulses' para que 'this' tenga acceso a los hermanos\n      const context = Object.create(pulses);\n\n      context.$flush = async () => {\n        if (currentState) {\n          setState(currentState, true);\n          currentState = deepCloneUnfreeze(localState);\n        }\n        await new Promise((resolve) => setTimeout(resolve, 0));\n      };\n\n      const emptyFlush = async () => {};\n\n      try {\n        const pulseResult = pulses[key].apply(context, [currentState, ...args]);\n        if (pulseResult instanceof Promise) {\n          return pulseResult\n            .then((resolvedValue) => {\n              setState(currentState as StateType);\n              context.$flush = emptyFlush;\n              return resolvedValue;\n            })\n            .catch((error) => {\n              console.error(`Error in pulse '${key}':`, error);\n              context.$flush = emptyFlush;\n              throw error;\n            });\n        } else {\n          setState(currentState);\n          context.$flush = emptyFlush;\n          return pulseResult;\n        }\n      } catch (error) {\n        console.error(`Error in pulse '${key}':`, error);\n        context.$flush = emptyFlush;\n        throw error;\n      }\n    };\n  }\n\n  syncState(localState);\n\n  const listeners: Record<string, Function[]> = {};\n  const trigger = (event: string, ...args: any[]) => {\n    if (listeners[event]) {\n      listeners[event].forEach((callback) => callback(...args));\n    }\n  };\n\n  const pulsesProxy = new Proxy(boundPulses, {\n    get: (pulses, prop: string) => {\n      if (prop === \"state\") {\n        return proxyState;\n      }\n      if (prop === \"on\") {\n        return (event: string, callback: Function) => {\n          if (!listeners[event]) {\n            listeners[event] = [];\n          }\n          listeners[event].push(callback);\n        };\n      }\n      if (prop === \"off\") {\n        return (event: string, callback: Function) => {\n          if (listeners[event]) {\n            listeners[event] = listeners[event].filter((cb) => cb !== callback);\n          }\n        };\n      }\n      if (!(prop in pulses)) {\n        throw new Error(`Pulse '${prop}' does not exist`);\n      }\n      const pulseMethod = pulses[prop];\n\n      return (...args: any[]) => {\n        const result = pulseMethod.apply(pulseMethod, args);\n        if (result instanceof Promise) {\n          return result.then((r) => {\n            trigger(\"pulse\", prop, args);\n            return r;\n          });\n        }\n        trigger(\"pulse\", prop, args);\n        return result;\n      };\n    }\n  });\n\n  return pulsesProxy as StorePulses<PulsesType> & {\n    state: ProxyState<StateType>;\n    on: (event: string, callback: Function) => void;\n    off: (event: string, callback: Function) => void;\n  };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n  initialState: StateType,\n  pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n  return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n  initialState: StateType,\n  pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n  console.warn(\n    \"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render.\"\n  );\n  return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n  const runEffect = () => {\n    try {\n      effectStack.push(runEffect);\n      effect();\n    } finally {\n      effectStack.pop();\n    }\n  };\n  runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n  let value = initialValue;\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const runSubscribers = () => {\n    subscribers.forEach((subscriber) => subscriber());\n  };\n\n  const read = (): T => {\n    const currentEffect = effectStack[effectStack.length - 1];\n    if (currentEffect && !subscribers.has(currentEffect)) {\n      subscribers.add(currentEffect);\n    }\n    registerDomSubscription(subscribers, domWithVnodesToUpdate);\n    return value;\n  };\n\n  const write = (newValue: T | ((current: T) => T)): void => {\n    const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n    if (!hasChanged(value, resolvedValue)) {\n      return;\n    }\n    value = resolvedValue;\n    runSubscribers();\n  };\n\n  return [read, write, runSubscribers];\n}\n"],"mappings":"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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","context","Object","create","$flush","async","emptyFlush","handleError","error","console","pulseResult","apply","Promise","then","resolvedValue","catch","listeners","trigger","event","callback","push","filter","cb","pulseMethod","result","r","warn","effect","runEffect","pop","initialValue","runSubscribers","newValue"],"sources":["../../lib/pulses/index.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-function-type */\n\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type PulseContext = {\n  $flush: () => Promise<void>;\n};\n\nexport type Pulse<StateType, TReturn = unknown> = (state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType> = {\n  [K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R\n    ? (...args: Args) => R\n    : never;\n};\n\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n  const currentVnode = current.vnode as VnodeWithDom;\n  if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n    return;\n  }\n\n  let hasParent = false;\n  let parent = currentVnode.dom.parentElement as DomElement;\n  while (parent) {\n    if (domWithVnodesToUpdate.has(parent)) {\n      hasParent = true;\n      break;\n    }\n    parent = parent.parentElement as DomElement;\n  }\n\n  if (!hasParent) {\n    const dom = currentVnode.dom;\n    const subscription = () => {\n      updateVnode(dom.vnode);\n      if (!dom.parentElement) {\n        subscribers.delete(subscription);\n        domWithVnodesToUpdate.delete(dom);\n      }\n    };\n    subscribers.add(subscription);\n    domWithVnodesToUpdate.add(dom);\n  }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n  initialState: StateType | (() => StateType) | null,\n  pulses: PulsesType & ThisType<PulsesType & PulseContext>,\n  immutable = false\n): StorePulses<PulsesType> & {\n  state: ProxyState<StateType>;\n  on: (event: string, callback: Function) => void;\n  off: (event: string, callback: Function) => void;\n} {\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const boundPulses: Record<string, Function> = {};\n  for (const key in pulses) {\n    if (typeof pulses[key] !== \"function\") {\n      throw new Error(`Pulse '${key}' must be a function`);\n    }\n    if (key === \"state\") {\n      throw new Error(`A pulse cannot be named 'state'`);\n    }\n    boundPulses[key] = getPulseMethod(key);\n  }\n\n  const localState: StateType =\n    (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n  function isMutable() {\n    if (immutable) {\n      throw new Error(\"You need to call a pulse to modify the state\");\n    }\n  }\n\n  let currentState: StateType | null = null;\n  let pulseCallCount = 0;\n\n  const proxyState = new Proxy(localState, {\n    get: (state, prop: string) => {\n      if (currentState) {\n        return currentState[prop];\n      }\n\n      const currentEffect = effectStack[effectStack.length - 1];\n      if (currentEffect && !subscribers.has(currentEffect)) {\n        subscribers.add(currentEffect);\n      }\n\n      registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n      return state[prop];\n    },\n    set: (state, prop: string, value: any) => {\n      isMutable();\n      Reflect.set(state, prop, value);\n      return true;\n    },\n    deleteProperty: (state, prop: string) => {\n      isMutable();\n      Reflect.deleteProperty(state, prop);\n      return true;\n    }\n  });\n\n  function syncState(newState: StateType) {\n    for (const key in newState) {\n      localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n    }\n    for (const key in localState) {\n      if (!(key in newState)) {\n        Reflect.deleteProperty(localState, key);\n      }\n    }\n  }\n\n  let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n  function debouncedUpdate() {\n    if (debounceTimeout) {\n      clearTimeout(debounceTimeout);\n    }\n    debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n  }\n\n  function setState(newState: StateType, flush = false) {\n    pulseCallCount--;\n    if (!hasChanged(localState, newState)) {\n      return;\n    }\n    if (pulseCallCount > 0 && !flush) {\n      return;\n    }\n    syncState(newState);\n    if (!flush) {\n      currentState = null;\n    }\n    pulseCallCount = 0;\n    debouncedUpdate();\n  }\n\n  function unfreezeState() {\n    if (currentState === null) {\n      currentState = deepCloneUnfreeze(localState);\n    }\n    return currentState;\n  }\n\n  function getPulseMethod(key: string) {\n    return function (this: any, ...args: any[]) {\n      pulseCallCount++;\n\n      const state = unfreezeState();\n\n      const context = Object.create(pulses);\n\n      context.$flush = async () => {\n        if (currentState) {\n          setState(currentState, true);\n        }\n      };\n\n      const emptyFlush = async () => {};\n\n      const handleError = (error: any) => {\n        console.error(`Error in pulse '${key}':`, error);\n        context.$flush = emptyFlush;\n\n        pulseCallCount--;\n        if (pulseCallCount <= 0) {\n          currentState = null;\n          pulseCallCount = 0;\n        }\n\n        throw error;\n      };\n\n      try {\n        const pulseResult = pulses[key].apply(context, [state, ...args]);\n        if (pulseResult instanceof Promise) {\n          return pulseResult\n            .then((resolvedValue) => {\n              setState(state as StateType);\n              context.$flush = emptyFlush;\n              return resolvedValue;\n            })\n            .catch(handleError);\n        } else {\n          setState(state);\n          context.$flush = emptyFlush;\n          return pulseResult;\n        }\n      } catch (error) {\n        handleError(error);\n      }\n    };\n  }\n\n  syncState(localState);\n\n  const listeners: Record<string, Function[]> = {};\n  const trigger = (event: string, ...args: any[]) => {\n    if (listeners[event]) {\n      listeners[event].forEach((callback) => callback(...args));\n    }\n  };\n\n  const pulsesProxy = new Proxy(boundPulses, {\n    get: (pulses, prop: string) => {\n      if (prop === \"state\") {\n        return proxyState;\n      }\n      if (prop === \"on\") {\n        return (event: string, callback: Function) => {\n          if (!listeners[event]) {\n            listeners[event] = [];\n          }\n          listeners[event].push(callback);\n        };\n      }\n      if (prop === \"off\") {\n        return (event: string, callback: Function) => {\n          if (listeners[event]) {\n            listeners[event] = listeners[event].filter((cb) => cb !== callback);\n          }\n        };\n      }\n      if (!(prop in pulses)) {\n        throw new Error(`Pulse '${prop}' does not exist`);\n      }\n      const pulseMethod = pulses[prop];\n\n      return (...args: any[]) => {\n        const result = pulseMethod.apply(pulseMethod, args);\n        if (result instanceof Promise) {\n          return result.then((r) => {\n            trigger(\"pulse\", prop, args);\n            return r;\n          });\n        }\n        trigger(\"pulse\", prop, args);\n        return result;\n      };\n    }\n  });\n\n  return pulsesProxy as StorePulses<PulsesType> & {\n    state: ProxyState<StateType>;\n    on: (event: string, callback: Function) => void;\n    off: (event: string, callback: Function) => void;\n  };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n  initialState: StateType,\n  pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n  return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n  initialState: StateType,\n  pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n  console.warn(\n    \"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render.\"\n  );\n  return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n  const runEffect = () => {\n    try {\n      effectStack.push(runEffect);\n      effect();\n    } finally {\n      effectStack.pop();\n    }\n  };\n  runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n  let value = initialValue;\n  const subscribers = new Set<Function>();\n  const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n  const runSubscribers = () => {\n    subscribers.forEach((subscriber) => subscriber());\n  };\n\n  const read = (): T => {\n    const currentEffect = effectStack[effectStack.length - 1];\n    if (currentEffect && !subscribers.has(currentEffect)) {\n      subscribers.add(currentEffect);\n    }\n    registerDomSubscription(subscribers, domWithVnodesToUpdate);\n    return value;\n  };\n\n  const write = (newValue: T | ((current: T) => T)): void => {\n    const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n    if (!hasChanged(value, resolvedValue)) {\n      return;\n    }\n    value = resolvedValue;\n    runSubscribers();\n  };\n\n  return [read, write, runSubscribers];\n}\n"],"mappings":"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"}
|
package/dist/pulses/index.mjs
CHANGED
|
@@ -99,46 +99,55 @@ function createStore(initialState, pulses, immutable = false) {
|
|
|
99
99
|
return;
|
|
100
100
|
}
|
|
101
101
|
syncState(newState);
|
|
102
|
-
|
|
102
|
+
if (!flush) {
|
|
103
|
+
currentState = null;
|
|
104
|
+
}
|
|
105
|
+
pulseCallCount = 0;
|
|
103
106
|
debouncedUpdate();
|
|
104
107
|
}
|
|
108
|
+
function unfreezeState() {
|
|
109
|
+
if (currentState === null) {
|
|
110
|
+
currentState = deepCloneUnfreeze(localState);
|
|
111
|
+
}
|
|
112
|
+
return currentState;
|
|
113
|
+
}
|
|
105
114
|
function getPulseMethod(key) {
|
|
106
115
|
return function(...args) {
|
|
107
116
|
pulseCallCount++;
|
|
108
|
-
|
|
109
|
-
currentState = deepCloneUnfreeze(localState);
|
|
110
|
-
}
|
|
117
|
+
const state = unfreezeState();
|
|
111
118
|
const context = Object.create(pulses);
|
|
112
119
|
context.$flush = async () => {
|
|
113
120
|
if (currentState) {
|
|
114
121
|
setState(currentState, true);
|
|
115
|
-
currentState = deepCloneUnfreeze(localState);
|
|
116
122
|
}
|
|
117
|
-
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
118
123
|
};
|
|
119
124
|
const emptyFlush = async () => {
|
|
120
125
|
};
|
|
126
|
+
const handleError = (error) => {
|
|
127
|
+
console.error(`Error in pulse '${key}':`, error);
|
|
128
|
+
context.$flush = emptyFlush;
|
|
129
|
+
pulseCallCount--;
|
|
130
|
+
if (pulseCallCount <= 0) {
|
|
131
|
+
currentState = null;
|
|
132
|
+
pulseCallCount = 0;
|
|
133
|
+
}
|
|
134
|
+
throw error;
|
|
135
|
+
};
|
|
121
136
|
try {
|
|
122
|
-
const pulseResult = pulses[key].apply(context, [
|
|
137
|
+
const pulseResult = pulses[key].apply(context, [state, ...args]);
|
|
123
138
|
if (pulseResult instanceof Promise) {
|
|
124
139
|
return pulseResult.then((resolvedValue) => {
|
|
125
|
-
setState(
|
|
140
|
+
setState(state);
|
|
126
141
|
context.$flush = emptyFlush;
|
|
127
142
|
return resolvedValue;
|
|
128
|
-
}).catch(
|
|
129
|
-
console.error(`Error in pulse '${key}':`, error);
|
|
130
|
-
context.$flush = emptyFlush;
|
|
131
|
-
throw error;
|
|
132
|
-
});
|
|
143
|
+
}).catch(handleError);
|
|
133
144
|
} else {
|
|
134
|
-
setState(
|
|
145
|
+
setState(state);
|
|
135
146
|
context.$flush = emptyFlush;
|
|
136
147
|
return pulseResult;
|
|
137
148
|
}
|
|
138
149
|
} catch (error) {
|
|
139
|
-
|
|
140
|
-
context.$flush = emptyFlush;
|
|
141
|
-
throw error;
|
|
150
|
+
handleError(error);
|
|
142
151
|
}
|
|
143
152
|
};
|
|
144
153
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../lib/pulses/index.ts"],
|
|
4
|
-
"sourcesContent": ["/* eslint-disable no-
|
|
5
|
-
"mappings": ";
|
|
4
|
+
"sourcesContent": ["/* eslint-disable @typescript-eslint/no-unsafe-function-type */\n\nimport { updateVnode, VnodeWithDom, current, DomElement } from \"valyrian.js\";\nimport { deepCloneUnfreeze, deepFreeze, hasChanged } from \"valyrian.js/utils\";\n\ntype State = Record<string, any>;\n\nexport type PulseContext = {\n $flush: () => Promise<void>;\n};\n\nexport type Pulse<StateType, TReturn = unknown> = (state: StateType, ...args: any[]) => TReturn | Promise<TReturn>;\n\ntype ProxyState<StateType> = StateType & { [key: string]: any };\n\nconst effectStack: Function[] = [];\n\ntype StorePulses<PulsesType> = {\n [K in keyof PulsesType]: PulsesType[K] extends (state: any, ...args: infer Args) => infer R\n ? (...args: Args) => R\n : never;\n};\n\nfunction registerDomSubscription(subscribers: Set<Function>, domWithVnodesToUpdate: WeakSet<DomElement>): void {\n const currentVnode = current.vnode as VnodeWithDom;\n if (!currentVnode || domWithVnodesToUpdate.has(currentVnode.dom)) {\n return;\n }\n\n let hasParent = false;\n let parent = currentVnode.dom.parentElement as DomElement;\n while (parent) {\n if (domWithVnodesToUpdate.has(parent)) {\n hasParent = true;\n break;\n }\n parent = parent.parentElement as DomElement;\n }\n\n if (!hasParent) {\n const dom = currentVnode.dom;\n const subscription = () => {\n updateVnode(dom.vnode);\n if (!dom.parentElement) {\n subscribers.delete(subscription);\n domWithVnodesToUpdate.delete(dom);\n }\n };\n subscribers.add(subscription);\n domWithVnodesToUpdate.add(dom);\n }\n}\n\nfunction createStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n initialState: StateType | (() => StateType) | null,\n pulses: PulsesType & ThisType<PulsesType & PulseContext>,\n immutable = false\n): StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n} {\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const boundPulses: Record<string, Function> = {};\n for (const key in pulses) {\n if (typeof pulses[key] !== \"function\") {\n throw new Error(`Pulse '${key}' must be a function`);\n }\n if (key === \"state\") {\n throw new Error(`A pulse cannot be named 'state'`);\n }\n boundPulses[key] = getPulseMethod(key);\n }\n\n const localState: StateType =\n (typeof initialState === \"function\" ? initialState() : initialState) || ({} as StateType);\n\n function isMutable() {\n if (immutable) {\n throw new Error(\"You need to call a pulse to modify the state\");\n }\n }\n\n let currentState: StateType | null = null;\n let pulseCallCount = 0;\n\n const proxyState = new Proxy(localState, {\n get: (state, prop: string) => {\n if (currentState) {\n return currentState[prop];\n }\n\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n\n return state[prop];\n },\n set: (state, prop: string, value: any) => {\n isMutable();\n Reflect.set(state, prop, value);\n return true;\n },\n deleteProperty: (state, prop: string) => {\n isMutable();\n Reflect.deleteProperty(state, prop);\n return true;\n }\n });\n\n function syncState(newState: StateType) {\n for (const key in newState) {\n localState[key] = immutable ? deepFreeze(newState[key]) : newState[key];\n }\n for (const key in localState) {\n if (!(key in newState)) {\n Reflect.deleteProperty(localState, key);\n }\n }\n }\n\n let debounceTimeout: ReturnType<typeof setTimeout> | null = null;\n function debouncedUpdate() {\n if (debounceTimeout) {\n clearTimeout(debounceTimeout);\n }\n debounceTimeout = setTimeout(() => subscribers.forEach((subscriber) => subscriber()), 0);\n }\n\n function setState(newState: StateType, flush = false) {\n pulseCallCount--;\n if (!hasChanged(localState, newState)) {\n return;\n }\n if (pulseCallCount > 0 && !flush) {\n return;\n }\n syncState(newState);\n if (!flush) {\n currentState = null;\n }\n pulseCallCount = 0;\n debouncedUpdate();\n }\n\n function unfreezeState() {\n if (currentState === null) {\n currentState = deepCloneUnfreeze(localState);\n }\n return currentState;\n }\n\n function getPulseMethod(key: string) {\n return function (this: any, ...args: any[]) {\n pulseCallCount++;\n\n const state = unfreezeState();\n\n const context = Object.create(pulses);\n\n context.$flush = async () => {\n if (currentState) {\n setState(currentState, true);\n }\n };\n\n const emptyFlush = async () => {};\n\n const handleError = (error: any) => {\n console.error(`Error in pulse '${key}':`, error);\n context.$flush = emptyFlush;\n\n pulseCallCount--;\n if (pulseCallCount <= 0) {\n currentState = null;\n pulseCallCount = 0;\n }\n\n throw error;\n };\n\n try {\n const pulseResult = pulses[key].apply(context, [state, ...args]);\n if (pulseResult instanceof Promise) {\n return pulseResult\n .then((resolvedValue) => {\n setState(state as StateType);\n context.$flush = emptyFlush;\n return resolvedValue;\n })\n .catch(handleError);\n } else {\n setState(state);\n context.$flush = emptyFlush;\n return pulseResult;\n }\n } catch (error) {\n handleError(error);\n }\n };\n }\n\n syncState(localState);\n\n const listeners: Record<string, Function[]> = {};\n const trigger = (event: string, ...args: any[]) => {\n if (listeners[event]) {\n listeners[event].forEach((callback) => callback(...args));\n }\n };\n\n const pulsesProxy = new Proxy(boundPulses, {\n get: (pulses, prop: string) => {\n if (prop === \"state\") {\n return proxyState;\n }\n if (prop === \"on\") {\n return (event: string, callback: Function) => {\n if (!listeners[event]) {\n listeners[event] = [];\n }\n listeners[event].push(callback);\n };\n }\n if (prop === \"off\") {\n return (event: string, callback: Function) => {\n if (listeners[event]) {\n listeners[event] = listeners[event].filter((cb) => cb !== callback);\n }\n };\n }\n if (!(prop in pulses)) {\n throw new Error(`Pulse '${prop}' does not exist`);\n }\n const pulseMethod = pulses[prop];\n\n return (...args: any[]) => {\n const result = pulseMethod.apply(pulseMethod, args);\n if (result instanceof Promise) {\n return result.then((r) => {\n trigger(\"pulse\", prop, args);\n return r;\n });\n }\n trigger(\"pulse\", prop, args);\n return result;\n };\n }\n });\n\n return pulsesProxy as StorePulses<PulsesType> & {\n state: ProxyState<StateType>;\n on: (event: string, callback: Function) => void;\n off: (event: string, callback: Function) => void;\n };\n}\n\nexport function createPulseStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n initialState: StateType,\n pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n return createStore(initialState, pulses, true);\n}\n\nexport function createMutableStore<StateType extends State, PulsesType extends Record<string, Pulse<StateType, any>>>(\n initialState: StateType,\n pulses: PulsesType & ThisType<PulsesType & PulseContext>\n) {\n console.warn(\n \"Warning: You are working with a mutable state. All state changes made outside of a pulse will not trigger a re-render.\"\n );\n return createStore(initialState, pulses, false);\n}\n\nexport function createEffect(effect: Function) {\n const runEffect = () => {\n try {\n effectStack.push(runEffect);\n effect();\n } finally {\n effectStack.pop();\n }\n };\n runEffect();\n}\n\nexport function createPulse<T>(initialValue: T): [() => T, (newValue: T | ((current: T) => T)) => void, () => void] {\n let value = initialValue;\n const subscribers = new Set<Function>();\n const domWithVnodesToUpdate = new WeakSet<DomElement>();\n\n const runSubscribers = () => {\n subscribers.forEach((subscriber) => subscriber());\n };\n\n const read = (): T => {\n const currentEffect = effectStack[effectStack.length - 1];\n if (currentEffect && !subscribers.has(currentEffect)) {\n subscribers.add(currentEffect);\n }\n registerDomSubscription(subscribers, domWithVnodesToUpdate);\n return value;\n };\n\n const write = (newValue: T | ((current: T) => T)): void => {\n const resolvedValue = typeof newValue === \"function\" ? (newValue as (current: T) => T)(value) : newValue;\n if (!hasChanged(value, resolvedValue)) {\n return;\n }\n value = resolvedValue;\n runSubscribers();\n };\n\n return [read, write, runSubscribers];\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["pulses"]
|
|
7
7
|
}
|
package/lib/pulses/index.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
/* eslint-disable no-
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-unsafe-function-type */
|
|
2
|
+
|
|
2
3
|
import { updateVnode, VnodeWithDom, current, DomElement } from "valyrian.js";
|
|
3
4
|
import { deepCloneUnfreeze, deepFreeze, hasChanged } from "valyrian.js/utils";
|
|
4
5
|
|
|
@@ -140,53 +141,66 @@ function createStore<StateType extends State, PulsesType extends Record<string,
|
|
|
140
141
|
return;
|
|
141
142
|
}
|
|
142
143
|
syncState(newState);
|
|
143
|
-
|
|
144
|
+
if (!flush) {
|
|
145
|
+
currentState = null;
|
|
146
|
+
}
|
|
147
|
+
pulseCallCount = 0;
|
|
144
148
|
debouncedUpdate();
|
|
145
149
|
}
|
|
146
150
|
|
|
151
|
+
function unfreezeState() {
|
|
152
|
+
if (currentState === null) {
|
|
153
|
+
currentState = deepCloneUnfreeze(localState);
|
|
154
|
+
}
|
|
155
|
+
return currentState;
|
|
156
|
+
}
|
|
157
|
+
|
|
147
158
|
function getPulseMethod(key: string) {
|
|
148
159
|
return function (this: any, ...args: any[]) {
|
|
149
160
|
pulseCallCount++;
|
|
150
|
-
if (currentState === null) {
|
|
151
|
-
currentState = deepCloneUnfreeze(localState);
|
|
152
|
-
}
|
|
153
161
|
|
|
154
|
-
|
|
162
|
+
const state = unfreezeState();
|
|
163
|
+
|
|
155
164
|
const context = Object.create(pulses);
|
|
156
165
|
|
|
157
166
|
context.$flush = async () => {
|
|
158
167
|
if (currentState) {
|
|
159
168
|
setState(currentState, true);
|
|
160
|
-
currentState = deepCloneUnfreeze(localState);
|
|
161
169
|
}
|
|
162
|
-
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
163
170
|
};
|
|
164
171
|
|
|
165
172
|
const emptyFlush = async () => {};
|
|
166
173
|
|
|
174
|
+
const handleError = (error: any) => {
|
|
175
|
+
console.error(`Error in pulse '${key}':`, error);
|
|
176
|
+
context.$flush = emptyFlush;
|
|
177
|
+
|
|
178
|
+
pulseCallCount--;
|
|
179
|
+
if (pulseCallCount <= 0) {
|
|
180
|
+
currentState = null;
|
|
181
|
+
pulseCallCount = 0;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
throw error;
|
|
185
|
+
};
|
|
186
|
+
|
|
167
187
|
try {
|
|
168
|
-
const pulseResult = pulses[key].apply(context, [
|
|
188
|
+
const pulseResult = pulses[key].apply(context, [state, ...args]);
|
|
169
189
|
if (pulseResult instanceof Promise) {
|
|
170
190
|
return pulseResult
|
|
171
191
|
.then((resolvedValue) => {
|
|
172
|
-
setState(
|
|
192
|
+
setState(state as StateType);
|
|
173
193
|
context.$flush = emptyFlush;
|
|
174
194
|
return resolvedValue;
|
|
175
195
|
})
|
|
176
|
-
.catch(
|
|
177
|
-
console.error(`Error in pulse '${key}':`, error);
|
|
178
|
-
context.$flush = emptyFlush;
|
|
179
|
-
throw error;
|
|
180
|
-
});
|
|
196
|
+
.catch(handleError);
|
|
181
197
|
} else {
|
|
182
|
-
setState(
|
|
198
|
+
setState(state);
|
|
183
199
|
context.$flush = emptyFlush;
|
|
184
200
|
return pulseResult;
|
|
185
201
|
}
|
|
186
202
|
} catch (error) {
|
|
187
|
-
|
|
188
|
-
context.$flush = emptyFlush;
|
|
189
|
-
throw error;
|
|
203
|
+
handleError(error);
|
|
190
204
|
}
|
|
191
205
|
};
|
|
192
206
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "valyrian.js",
|
|
3
|
-
"version": "8.1.
|
|
3
|
+
"version": "8.1.9",
|
|
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>",
|