stunk 2.5.0 → 2.7.1

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/index.cjs CHANGED
@@ -1,2 +1 @@
1
- 'use strict';var K=Object.defineProperty;var z=(e,n)=>{for(var r in n)K(e,r,{get:n[r],enumerable:true});};function B(e){return e!==null}function W(e){if(!e||typeof e!="object")return false;let n=e;return ["get","set","update","subscribe","derive","reset","destroy"].every(t=>typeof n[t]=="function")}function J(e){let n=false,r;return ()=>(n||(r=e(),n=true),r)}function U(e){let n=Object.keys(e).reduce((c,s)=>(c[s]=null,c),{}),r={loading:Object.keys(e).length>0,error:null,errors:{},data:n},t=p(r);Object.values(e);return Object.entries(e).forEach(([c,s])=>{s.subscribe(i=>{let f=t.get(),a=false,u=null,T={};Object.entries(e).forEach(([w,h])=>{let l=h.get();l.loading&&(a=true),l.error&&(u||(u=l.error),T[w]=l.error);}),t.set({loading:a,error:u,errors:T,data:{...f.data,[c]:i.data}});});}),t}function j(e,n){if(e===null)throw new Error("Value cannot be null.");let r=e,t=0;for(;t<n.length;){let o=n[t],c=typeof o=="function"?o:o.fn,s=typeof o=="function"?`index ${t}`:o.name||`index ${t}`,i=false,f=null;try{c(r,a=>{i=!0,f=a;});}catch(a){let u=a instanceof Error?a.message:String(a);throw new Error(`Middleware "${s}" threw an error: ${u}`)}if(!i)break;if(f===null)throw new Error(`Middleware at index ${t} returned null value.`);r=f,t++;}return r}function E(e,n){if(e===n)return true;if(!e||!n||typeof e!=typeof n)return false;if(Array.isArray(e)&&Array.isArray(n)){if(e.length!==n.length)return false;for(let r=0;r<e.length;r++)if(e[r]!==n[r])return false;return true}if(typeof e=="object"&&typeof n=="object"){let r=Object.keys(e),t=Object.keys(n);if(r.length!==t.length)return false;for(let o of r)if(!Object.prototype.hasOwnProperty.call(n,o)||e[o]!==n[o])return false;return true}return false}function v(e,n,r=""){if(typeof e=="object"&&e!==null&&typeof n=="object"&&n!==null){if(Array.isArray(e)&&Array.isArray(n)){if(e.length>0&&typeof e[0]=="object")for(let t=0;t<n.length;t++)v(e[0],n[t],`${r}[${t}]`);}else if(!Array.isArray(e)&&!Array.isArray(n)){let t=Object.keys(e),o=Object.keys(n),c=o.filter(s=>!t.includes(s));c.length>0&&(console.error(`\u{1F6A8} Stunk: Unknown properties detected at '${r||"root"}': ${c.join(", ")}. This might cause bugs.`),console.error("Expected keys:",t),console.error("Received keys:",o));for(let s of t)v(e[s],n[s],r?`${r}.${s}`:s);}}}var S=false,P=new Set,V=new Map,L=0;function G(e){let n=S;S=true;try{e();}finally{if(!n){S=false;let r=Array.from(P);P.clear(),r.forEach(t=>{let o=V.get(t);o&&o.notify();});}}}function p(e,n=[]){if(e===null)throw new Error("Initial value cannot be null.");let r=e,t=new Set,o=L++,c=()=>{t.forEach(h=>h(r));};V.set(o,{notify:c});let s=()=>{t.size!==0&&(S?P.add(o):c());},i=()=>r,f=h=>{let l;typeof h=="function"?l=h(r):l=h,v(r,l);let y=j(l,n);y!==r&&(r=y,s());},a=h=>{if(typeof h!="function")throw new Error("Callback must be a function.");return t.add(h),h(r),()=>t.delete(h)};return {get:i,set:f,subscribe:a,derive:h=>{if(typeof h!="function")throw new Error("Derive function must be a function.");let l=h(r),y=p(l),m=a(()=>{let D=h(r);y.set(D);}),g=y.destroy;return y.destroy=()=>{m(),g();},y},reset:()=>{r=e,s();},destroy:()=>{t.clear(),r=e,P.delete(o),V.delete(o);}}}function Q(e,n={}){let {initialData:r=null,onError:t,retryCount:o=0,retryDelay:c=1e3,refresh:s={},enabled:i=true}=n,{staleTime:f=0,cacheTime:a=5*60*1e3,refetchInterval:u}=s,T=e.length>0,w={loading:i&&!T,error:null,data:r,lastFetched:void 0},h=d=>T?d===void 0?false:Array.isArray(d)?d.every(x=>x!=null):true:true,l=p(w),y,m=null,g=null,D=()=>{let d=l.get();return d.lastFetched?Date.now()-d.lastFetched>f:true},N=()=>{l.set({...l.get(),data:r,lastFetched:void 0});},q=()=>{g&&clearTimeout(g),a>0&&(g=setTimeout(N,a));},k=async(d,x=o,b=false)=>{if(i){if(d!==void 0&&(y=d),!h(y)){l.set({...l.get(),loading:false});return}if(!(!b&&!D()&&l.get().data!==null&&f>0)){l.set({...l.get(),loading:true,error:null});try{let C=T?await e(...y):await e(),A=Date.now();l.set({loading:!1,error:null,data:C,lastFetched:A}),q();}catch(C){if(x>0)return await new Promise(A=>setTimeout(A,c)),k(d,x-1,b);l.set({loading:false,error:C,data:l.get().data,lastFetched:l.get().lastFetched}),t&&t(C);}}}};u&&u>0&&i&&(m=setInterval(()=>{i&&k(y,0,false);},u)),i&&!T&&k();let M=()=>{m&&(clearInterval(m),m=null),g&&(clearTimeout(g),g=null);};return {...l,reload:async(...d)=>{await k(d.length>0?d:void 0,o,true);},refresh:async(...d)=>{await k(d.length>0?d:void 0,o,false);},mutate:d=>{let x=l.get(),b=d(x.data);l.set({...x,data:b});},reset:()=>{M(),l.set({...w,loading:i}),i&&(k(),u&&u>0&&(m=setInterval(()=>{i&&k(y,0,false);},u)));},cleanup:M,setParams:(...d)=>{y=d,i&&h(d)&&k(d);}}}function X(e,n){let r=e.map(a=>a.get()),t=n(...r),o=p(t),c=o.set,s=false,i=()=>{let a=false;for(let u=0;u<e.length;u++){let T=e[u].get();T!==r[u]&&(r[u]=T,a=true);}if(a){let u=n(...r);u!==t&&(typeof u!="object"||typeof t!="object"||!E(u,t))&&(t=u,c(u)),s=false;}},f=e.map(a=>a.subscribe(()=>{s=true,i();}));return {...o,get:()=>(s&&i(),t),recompute:i,isDirty:()=>s,set:()=>{throw new Error("Cannot set values directly on computed. Modify the source chunk instead.")},reset:()=>(e.forEach(a=>{typeof a.reset=="function"&&a.reset();}),s=true,i(),t),destroy:()=>{f.forEach(a=>a()),o.destroy?.();}}}function O(e,n,r={}){let{useShallowEqual:t=false}=r,o=e.get(),c=n(o),s=p(c),i=()=>{let a=e.get(),u=n(a);o=a,(t?!E(u,c):u!==c)&&(c=u,s.set(u));},f=e.subscribe(i);return {get:()=>s.get(),set:()=>{throw new Error("Cannot set values directly on a selector. Modify the source chunk instead.")},subscribe:s.subscribe,derive:a=>O(s,a,r),reset:()=>{throw new Error("Cannot reset a selector chunk. Reset the source chunk instead.")},destroy:()=>{f(),s.destroy();}}}var $={};z($,{logger:()=>R,nonNegativeValidator:()=>I,withHistory:()=>F,withPersistence:()=>H});var R=(e,n)=>{console.log("Setting value:",e),n(e);};var I=(e,n)=>{if(e<0)throw new Error("Value must be non-negative!");n(e);};function F(e,n={}){let{maxHistory:r=100}=n,t=[e.get()],o=0,c=false,s={...e,set:i=>{if(c){e.set(i);return}let f;if(typeof i=="function"){let a=e.get();f=i(a);}else f=i;if(t.splice(o+1),t.push(f),t.length>r){console.warn("History limit reached. Removing oldest entries.");let a=t.length-r;t.splice(0,a),o=Math.max(0,o-a);}o=t.length-1,e.set(f);},undo:()=>{s.canUndo()&&(c=true,o--,s.set(t[o]),c=false);},redo:()=>{s.canRedo()&&(c=true,o++,s.set(t[o]),c=false);},canUndo:()=>o>0,canRedo:()=>o<t.length-1,getHistory:()=>[...t],clearHistory:()=>{let i=e.get();t.length=0,t.push(i),o=0;},destroy:()=>{t.length=0,e.destroy();}};return s}function H(e,n){let{key:r,storage:t=localStorage,serialize:o=JSON.stringify,deserialize:c=JSON.parse}=n;try{let s=t.getItem(r);if(s){let i=c(s);e.set(i);}}catch(s){console.error("Failed to load persisted state:",s);}return e.subscribe(s=>{try{let i=o(s);t.setItem(r,i);}catch(i){console.log("Failed to persist chunk",i);}}),e}exports.asyncChunk=Q;exports.batch=G;exports.chunk=p;exports.combineAsyncChunks=U;exports.computed=X;exports.isChunk=W;exports.isValidChunkValue=B;exports.middleware=$;exports.once=J;exports.select=O;//# sourceMappingURL=index.cjs.map
2
- //# sourceMappingURL=index.cjs.map
1
+ 'use strict';var Y=Object.defineProperty;var Z=(e,a)=>{for(var r in a)Y(e,r,{get:a[r],enumerable:true});};function _(e){return e!==null}function ee(e){if(!e||typeof e!="object")return false;let a=e;return ["get","set","subscribe","derive","reset","destroy"].every(t=>typeof a[t]=="function")}function te(e){let a=false,r;return ()=>(a||(r=e(),a=true),r)}function re(e){let a=Object.keys(e).reduce((n,u)=>(n[u]=null,n),{}),r={loading:Object.keys(e).length>0,error:null,errors:{},data:a},t=k(r);return Object.entries(e).forEach(([n,u])=>{u.subscribe(o=>{let c=t.get(),l=false,s=null,d={};Object.entries(e).forEach(([h,x])=>{let p=x.get();p.loading&&(l=true),p.error&&(s||(s=p.error),d[h]=p.error);}),t.set({loading:l,error:s,errors:d,data:{...c.data,[n]:o.data}});});}),t}function N(e,a){if(e===null)throw new Error("Value cannot be null.");let r=e,t=0;for(;t<a.length;){let n=a[t],u=typeof n=="function"?n:n.fn,o=typeof n=="function"?`index ${t}`:n.name||`index ${t}`,c=false,l=null;try{u(r,s=>{c=!0,l=s;});}catch(s){let d=s instanceof Error?s.message:String(s);throw new Error(`Middleware "${o}" threw an error: ${d}`)}if(!c)break;if(l===null)throw new Error(`Middleware at index ${t} returned null value.`);r=l,t++;}return r}function M(e,a){if(e===a)return true;if(!e||!a||typeof e!=typeof a)return false;if(Array.isArray(e)&&Array.isArray(a)){if(e.length!==a.length)return false;for(let r=0;r<e.length;r++)if(e[r]!==a[r])return false;return true}if(typeof e=="object"&&typeof a=="object"){let r=Object.keys(e),t=Object.keys(a);if(r.length!==t.length)return false;for(let n of r)if(!Object.prototype.hasOwnProperty.call(a,n)||e[n]!==a[n])return false;return true}return false}function D(e,a,r=""){if(typeof e=="object"&&e!==null&&typeof a=="object"&&a!==null){if(Array.isArray(e)&&Array.isArray(a)){if(e.length>0&&typeof e[0]=="object")for(let t=0;t<a.length;t++)D(e[0],a[t],`${r}[${t}]`);}else if(!Array.isArray(e)&&!Array.isArray(a)){let t=Object.keys(e),n=Object.keys(a),u=n.filter(o=>!t.includes(o));u.length>0&&(console.error(`\u{1F6A8} Stunk: Unknown properties detected at '${r||"root"}': ${u.join(", ")}. This might cause bugs.`),console.error("Expected keys:",t),console.error("Received keys:",n));for(let o of t)D(e[o],a[o],r?`${r}.${o}`:o);}}}var V=false,R=new Set,O=new Map,ne=0;function ae(e){let a=V;V=true;try{e();}finally{if(!a){V=false;let r=Array.from(R);R.clear(),r.forEach(t=>{let n=O.get(t);n&&n.notify();});}}}function k(e,a=[]){if(e===null)throw new Error("Initial value cannot be null.");let r=e,t=new Set,n=ne++,u=()=>{t.forEach(p=>p(r));};O.set(n,{notify:u});let o=()=>{t.size!==0&&(V?R.add(n):u());},c=()=>r,l=p=>{let y;typeof p=="function"?y=p(r):y=p,D(r,y);let b=N(y,a);b!==r&&(r=b,o());},s=p=>{if(typeof p!="function")throw new Error("Callback must be a function.");return t.add(p),p(r),()=>t.delete(p)};return {get:c,set:l,subscribe:s,derive:p=>{if(typeof p!="function")throw new Error("Derive function must be a function.");let y=p(r),b=k(y),f=s(()=>{let E=p(r);b.set(E);}),g=b.destroy;return b.destroy=()=>{f(),g();},b},reset:()=>{r=e,o();},destroy:()=>{t.clear(),r=e,R.delete(n),O.delete(n);}}}function z(e,a={}){let{initialData:r=null,onError:t,retryCount:n=0,retryDelay:u=1e3,refresh:o={},pagination:c,enabled:l=true}=a,{staleTime:s=0,cacheTime:d=300*1e3,refetchInterval:h}=o,x=!!c,p=c?.mode||"replace",y=e.length>0,b={loading:l&&!y,error:null,data:r,lastFetched:void 0,pagination:x?{page:c.initialPage||1,pageSize:c.pageSize||10}:void 0},f=k(b),g={},E=null,v=null,L=()=>{let i=f.get();return !i.lastFetched||s===0?true:Date.now()-i.lastFetched>s},G=()=>{f.set({...f.get(),data:r,lastFetched:void 0});},Q=()=>{v&&clearTimeout(v),d>0&&(v=setTimeout(G,d));},m=async(i,T=n,A=false)=>{if(!l||(i!==void 0&&(g={...g,...i}),!A&&!L()&&f.get().data!==null))return;let C=f.get();f.set({...C,loading:true,error:null});try{let S={...g};x&&C.pagination&&(S.page=C.pagination.page,S.pageSize=C.pagination.pageSize);let P=y?await e(S):await e(),w,H,$;if(P&&typeof P=="object"&&"data"in P){let j=P;w=j.data,H=j.total,$=j.hasMore;}else w=P;x&&p==="accumulate"&&C.data&&Array.isArray(C.data)&&Array.isArray(w)&&(w=[...C.data,...w]);let X=Date.now();f.set({loading:!1,error:null,data:w,lastFetched:X,pagination:x?{...C.pagination,total:H,hasMore:$}:void 0}),Q();}catch(S){if(T>0)return await new Promise(w=>setTimeout(w,u)),m(i,T-1,A);let P=f.get();f.set({loading:false,error:S,data:P.data,lastFetched:P.lastFetched,pagination:P.pagination}),t&&t(S);}};h&&h>0&&l&&(E=setInterval(()=>{l&&m(void 0,0,false);},h)),l&&!y&&m();let F=()=>{E&&(clearInterval(E),E=null),v&&(clearTimeout(v),v=null);},I={...f,reload:async i=>{await m(i,n,true);},refresh:async i=>{await m(i,n,false);},mutate:i=>{let T=f.get(),A=i(T.data);f.set({...T,data:A});},reset:()=>{F(),g={},f.set({...b,loading:l&&!y}),l&&!y&&(m(),h&&h>0&&(E=setInterval(()=>{l&&m(void 0,0,false);},h)));},cleanup:F,setParams:i=>{g={...g,...i},l&&m(i,n,true);}};return x?{...I,nextPage:async()=>{let i=f.get();i.pagination&&i.pagination.hasMore!==false&&(f.set({...i,pagination:{...i.pagination,page:i.pagination.page+1}}),await m(g,n,true));},prevPage:async()=>{let i=f.get();!i.pagination||i.pagination.page<=1||(f.set({...i,pagination:{...i.pagination,page:i.pagination.page-1}}),await m(g,n,true));},goToPage:async i=>{let T=f.get();!T.pagination||i<1||(f.set({...T,pagination:{...T.pagination,page:i}}),await m(g,n,true));},resetPagination:async()=>{let i=f.get();if(!i.pagination)return;let T=c?.initialPage||1;f.set({...i,data:p==="accumulate"?r:i.data,pagination:{...i.pagination,page:T}}),await m(g,n,true);}}:I}function oe(e,a={}){let{pageSize:r=10,staleTime:t,cacheTime:n,retryCount:u,retryDelay:o,onError:c}=a;return z(e,{pagination:{pageSize:r,mode:"accumulate",initialPage:1},refresh:{staleTime:t,cacheTime:n},retryCount:u,retryDelay:o,onError:c})}function se(e,a){let r=e.map(s=>s.get()),t=a(...r),n=k(t),u=n.set,o=false,c=()=>{let s=false;for(let d=0;d<e.length;d++){let h=e[d].get();h!==r[d]&&(r[d]=h,s=true);}if(s){let d=a(...r);d!==t&&(typeof d!="object"||typeof t!="object"||!M(d,t))&&(t=d,u(d)),o=false;}},l=e.map(s=>s.subscribe(()=>{o=true,c();}));return {...n,get:()=>(o&&c(),t),recompute:c,isDirty:()=>o,set:()=>{throw new Error("Cannot set values directly on computed. Modify the source chunk instead.")},reset:()=>(e.forEach(s=>{typeof s.reset=="function"&&s.reset();}),o=true,c(),t),destroy:()=>{l.forEach(s=>s()),n.destroy?.();}}}function q(e,a,r={}){let{useShallowEqual:t=false}=r,n=e.get(),u=a(n),o=k(u),c=()=>{let s=e.get(),d=a(s);n=s,(t?!M(d,u):d!==u)&&(u=d,o.set(d));},l=e.subscribe(c);return {get:()=>o.get(),set:()=>{throw new Error("Cannot set values directly on a selector. Modify the source chunk instead.")},subscribe:o.subscribe,derive:s=>q(o,s,r),reset:()=>{throw new Error("Cannot reset a selector chunk. Reset the source chunk instead.")},destroy:()=>{l(),o.destroy();}}}var U={};Z(U,{logger:()=>K,nonNegativeValidator:()=>W,withHistory:()=>B,withPersistence:()=>J});var K=(e,a)=>{console.log("Setting value:",e),a(e);};var W=(e,a)=>{if(e<0)throw new Error("Value must be non-negative!");a(e);};function B(e,a={}){let{maxHistory:r=100}=a,t=[e.get()],n=0,u=false,o={...e,set:c=>{if(u){e.set(c);return}let l;if(typeof c=="function"){let s=e.get();l=c(s);}else l=c;if(t.splice(n+1),t.push(l),t.length>r){console.warn("History limit reached. Removing oldest entries.");let s=t.length-r;t.splice(0,s),n=Math.max(0,n-s);}n=t.length-1,e.set(l);},undo:()=>{o.canUndo()&&(u=true,n--,o.set(t[n]),u=false);},redo:()=>{o.canRedo()&&(u=true,n++,o.set(t[n]),u=false);},canUndo:()=>n>0,canRedo:()=>n<t.length-1,getHistory:()=>[...t],clearHistory:()=>{let c=e.get();t.length=0,t.push(c),n=0;},destroy:()=>{t.length=0,e.destroy();}};return o}function J(e,a){let{key:r,storage:t=localStorage,serialize:n=JSON.stringify,deserialize:u=JSON.parse}=a;try{let o=t.getItem(r);if(o){let c=u(o);e.set(c);}}catch(o){console.error("Failed to load persisted state:",o);}return e.subscribe(o=>{try{let c=n(o);t.setItem(r,c);}catch(c){console.log("Failed to persist chunk",c);}}),e}exports.asyncChunk=z;exports.batch=ae;exports.chunk=k;exports.combineAsyncChunks=re;exports.computed=se;exports.infiniteAsyncChunk=oe;exports.isChunk=ee;exports.isValidChunkValue=_;exports.middleware=U;exports.once=te;exports.select=q;
package/dist/index.d.cts CHANGED
@@ -50,39 +50,103 @@ interface AsyncState<T, E extends Error> {
50
50
  data: T | null;
51
51
  lastFetched?: number;
52
52
  }
53
+ interface PaginationState {
54
+ page: number;
55
+ pageSize: number;
56
+ total?: number;
57
+ hasMore?: boolean;
58
+ }
59
+ interface AsyncStateWithPagination<T, E extends Error> extends AsyncState<T, E> {
60
+ pagination?: PaginationState;
61
+ }
53
62
  interface RefreshConfig {
54
- /** Time in milliseconds after which data becomes stale */
63
+ /** Time in ms after which data becomes stale */
55
64
  staleTime?: number;
56
- /** Time in milliseconds to cache data before considering it expired */
65
+ /** Time in ms to cache data */
57
66
  cacheTime?: number;
58
- /** Auto-refresh interval in milliseconds */
67
+ /** Auto-refresh interval in ms */
59
68
  refetchInterval?: number;
60
69
  }
70
+ interface PaginationConfig {
71
+ /** Initial page number (default: 1) */
72
+ initialPage?: number;
73
+ /** Items per page (default: 10) */
74
+ pageSize?: number;
75
+ /** Whether to accumulate pages (infinite scroll) or replace */
76
+ mode?: 'replace' | 'accumulate';
77
+ }
61
78
  interface AsyncChunkOptExtended<T, E extends Error> extends AsyncChunkOpt<T, E> {
62
79
  refresh?: RefreshConfig;
63
- /** Enable/disable the fetcher - useful for conditional fetching */
80
+ pagination?: PaginationConfig;
81
+ /** Enable/disable the fetcher */
64
82
  enabled?: boolean;
65
83
  }
66
- interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncState<T, E>> {
67
- /** Reload the data from the source. */
68
- reload: () => Promise<void>;
84
+ interface FetcherResponse<T> {
85
+ data: T;
86
+ total?: number;
87
+ hasMore?: boolean;
88
+ }
89
+ interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncStateWithPagination<T, E>> {
90
+ /** Force reload data */
91
+ reload: (params?: any) => Promise<void>;
69
92
  /** Smart refresh - respects stale time */
70
- refresh: () => Promise<void>;
71
- /** Mutate the data directly. */
93
+ refresh: (params?: any) => Promise<void>;
94
+ /** Mutate data directly */
72
95
  mutate: (mutator: (currentData: T | null) => T) => void;
73
- /** Reset the state to the initial value. */
96
+ /** Reset to initial state */
74
97
  reset: () => void;
75
- /** Clean up intervals and timeouts */
98
+ /** Clean up intervals */
76
99
  cleanup: () => void;
77
100
  }
78
- declare function asyncChunk<T, E extends Error = Error>(fetcher: () => Promise<T>, options?: AsyncChunkOptExtended<T, E>): AsyncChunk<T, E>;
79
- declare function asyncChunk<T, E extends Error = Error, P extends any[] = []>(fetcher: (...params: P) => Promise<T>, options?: AsyncChunkOptExtended<T, E>): AsyncChunk<T, E> & {
80
- /** Reload with new parameters */
81
- reload: (...params: P) => Promise<void>;
82
- /** Smart refresh with parameters */
83
- refresh: (...params: P) => Promise<void>;
84
- /** Set parameters for future calls */
85
- setParams: (...params: P) => void;
101
+ interface PaginatedAsyncChunk<T, E extends Error = Error> extends AsyncChunk<T, E> {
102
+ /** Load next page */
103
+ nextPage: () => Promise<void>;
104
+ /** Load previous page */
105
+ prevPage: () => Promise<void>;
106
+ /** Go to specific page */
107
+ goToPage: (page: number) => Promise<void>;
108
+ /** Reset pagination to first page */
109
+ resetPagination: () => Promise<void>;
110
+ }
111
+ declare function asyncChunk<T, E extends Error = Error>(fetcher: () => Promise<T | FetcherResponse<T>>, options?: AsyncChunkOptExtended<T, E>): AsyncChunk<T, E> | PaginatedAsyncChunk<T, E>;
112
+ declare function asyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(fetcher: (params: P & {
113
+ page?: number;
114
+ pageSize?: number;
115
+ }) => Promise<T | FetcherResponse<T>>, options?: AsyncChunkOptExtended<T, E>): (AsyncChunk<T, E> | PaginatedAsyncChunk<T, E>) & {
116
+ setParams: (params: Partial<P>) => void;
117
+ reload: (params?: Partial<P>) => Promise<void>;
118
+ refresh: (params?: Partial<P>) => Promise<void>;
119
+ };
120
+
121
+ interface InfiniteAsyncChunkOptions<T, E extends Error> {
122
+ /** Initial page size (default: 10) */
123
+ pageSize?: number;
124
+ /** Time in ms after which data becomes stale */
125
+ staleTime?: number;
126
+ /** Time in ms to cache data */
127
+ cacheTime?: number;
128
+ /** Retry count on error */
129
+ retryCount?: number;
130
+ /** Delay between retries in ms */
131
+ retryDelay?: number;
132
+ /** Error callback */
133
+ onError?: (error: E) => void;
134
+ }
135
+ /**
136
+ * Create an infinite scroll async chunk with accumulate mode.
137
+ * Automatically handles pagination in accumulate mode for infinite scrolling.
138
+ */
139
+ declare function infiniteAsyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(fetcher: (params: P & {
140
+ page: number;
141
+ pageSize: number;
142
+ }) => Promise<{
143
+ data: T[];
144
+ hasMore?: boolean;
145
+ total?: number;
146
+ }>, options?: InfiniteAsyncChunkOptions<T, E>): PaginatedAsyncChunk<T[], E> & {
147
+ setParams: (params: Partial<P>) => void;
148
+ reload: (params?: Partial<P>) => Promise<void>;
149
+ refresh: (params?: Partial<P>) => Promise<void>;
86
150
  };
87
151
 
88
152
  type ChunkValue<T> = T extends Chunk<infer U> ? U : never;
@@ -120,6 +184,10 @@ declare function select<T, S>(sourceChunk: Chunk<T>, selector: (value: T) => S,
120
184
  declare function isValidChunkValue(value: unknown): boolean;
121
185
  declare function isChunk<T>(value: unknown): value is Chunk<T>;
122
186
  declare function once<T>(fn: () => T): () => T;
187
+ /**
188
+ * Combines multiple async chunks into a single chunk.
189
+ * The combined chunk tracks loading, error, and data states from all source chunks.
190
+ */
123
191
  declare function combineAsyncChunks<T extends Record<string, AsyncChunk<any>>>(chunks: T): Chunk<CombinedState<T>>;
124
192
 
125
193
  declare const logger: Middleware<any>;
@@ -172,4 +240,4 @@ declare namespace index {
172
240
  export { index_logger as logger, index_nonNegativeValidator as nonNegativeValidator, index_withHistory as withHistory, index_withPersistence as withPersistence };
173
241
  }
174
242
 
175
- export { type Chunk, type Middleware, asyncChunk, batch, chunk, combineAsyncChunks, computed, isChunk, isValidChunkValue, index as middleware, once, select };
243
+ export { type AsyncChunk, type AsyncState, type AsyncStateWithPagination, type Chunk, type Middleware, type PaginatedAsyncChunk, asyncChunk, batch, chunk, combineAsyncChunks, computed, infiniteAsyncChunk, isChunk, isValidChunkValue, index as middleware, once, select };
package/dist/index.d.ts CHANGED
@@ -50,39 +50,103 @@ interface AsyncState<T, E extends Error> {
50
50
  data: T | null;
51
51
  lastFetched?: number;
52
52
  }
53
+ interface PaginationState {
54
+ page: number;
55
+ pageSize: number;
56
+ total?: number;
57
+ hasMore?: boolean;
58
+ }
59
+ interface AsyncStateWithPagination<T, E extends Error> extends AsyncState<T, E> {
60
+ pagination?: PaginationState;
61
+ }
53
62
  interface RefreshConfig {
54
- /** Time in milliseconds after which data becomes stale */
63
+ /** Time in ms after which data becomes stale */
55
64
  staleTime?: number;
56
- /** Time in milliseconds to cache data before considering it expired */
65
+ /** Time in ms to cache data */
57
66
  cacheTime?: number;
58
- /** Auto-refresh interval in milliseconds */
67
+ /** Auto-refresh interval in ms */
59
68
  refetchInterval?: number;
60
69
  }
70
+ interface PaginationConfig {
71
+ /** Initial page number (default: 1) */
72
+ initialPage?: number;
73
+ /** Items per page (default: 10) */
74
+ pageSize?: number;
75
+ /** Whether to accumulate pages (infinite scroll) or replace */
76
+ mode?: 'replace' | 'accumulate';
77
+ }
61
78
  interface AsyncChunkOptExtended<T, E extends Error> extends AsyncChunkOpt<T, E> {
62
79
  refresh?: RefreshConfig;
63
- /** Enable/disable the fetcher - useful for conditional fetching */
80
+ pagination?: PaginationConfig;
81
+ /** Enable/disable the fetcher */
64
82
  enabled?: boolean;
65
83
  }
66
- interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncState<T, E>> {
67
- /** Reload the data from the source. */
68
- reload: () => Promise<void>;
84
+ interface FetcherResponse<T> {
85
+ data: T;
86
+ total?: number;
87
+ hasMore?: boolean;
88
+ }
89
+ interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncStateWithPagination<T, E>> {
90
+ /** Force reload data */
91
+ reload: (params?: any) => Promise<void>;
69
92
  /** Smart refresh - respects stale time */
70
- refresh: () => Promise<void>;
71
- /** Mutate the data directly. */
93
+ refresh: (params?: any) => Promise<void>;
94
+ /** Mutate data directly */
72
95
  mutate: (mutator: (currentData: T | null) => T) => void;
73
- /** Reset the state to the initial value. */
96
+ /** Reset to initial state */
74
97
  reset: () => void;
75
- /** Clean up intervals and timeouts */
98
+ /** Clean up intervals */
76
99
  cleanup: () => void;
77
100
  }
78
- declare function asyncChunk<T, E extends Error = Error>(fetcher: () => Promise<T>, options?: AsyncChunkOptExtended<T, E>): AsyncChunk<T, E>;
79
- declare function asyncChunk<T, E extends Error = Error, P extends any[] = []>(fetcher: (...params: P) => Promise<T>, options?: AsyncChunkOptExtended<T, E>): AsyncChunk<T, E> & {
80
- /** Reload with new parameters */
81
- reload: (...params: P) => Promise<void>;
82
- /** Smart refresh with parameters */
83
- refresh: (...params: P) => Promise<void>;
84
- /** Set parameters for future calls */
85
- setParams: (...params: P) => void;
101
+ interface PaginatedAsyncChunk<T, E extends Error = Error> extends AsyncChunk<T, E> {
102
+ /** Load next page */
103
+ nextPage: () => Promise<void>;
104
+ /** Load previous page */
105
+ prevPage: () => Promise<void>;
106
+ /** Go to specific page */
107
+ goToPage: (page: number) => Promise<void>;
108
+ /** Reset pagination to first page */
109
+ resetPagination: () => Promise<void>;
110
+ }
111
+ declare function asyncChunk<T, E extends Error = Error>(fetcher: () => Promise<T | FetcherResponse<T>>, options?: AsyncChunkOptExtended<T, E>): AsyncChunk<T, E> | PaginatedAsyncChunk<T, E>;
112
+ declare function asyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(fetcher: (params: P & {
113
+ page?: number;
114
+ pageSize?: number;
115
+ }) => Promise<T | FetcherResponse<T>>, options?: AsyncChunkOptExtended<T, E>): (AsyncChunk<T, E> | PaginatedAsyncChunk<T, E>) & {
116
+ setParams: (params: Partial<P>) => void;
117
+ reload: (params?: Partial<P>) => Promise<void>;
118
+ refresh: (params?: Partial<P>) => Promise<void>;
119
+ };
120
+
121
+ interface InfiniteAsyncChunkOptions<T, E extends Error> {
122
+ /** Initial page size (default: 10) */
123
+ pageSize?: number;
124
+ /** Time in ms after which data becomes stale */
125
+ staleTime?: number;
126
+ /** Time in ms to cache data */
127
+ cacheTime?: number;
128
+ /** Retry count on error */
129
+ retryCount?: number;
130
+ /** Delay between retries in ms */
131
+ retryDelay?: number;
132
+ /** Error callback */
133
+ onError?: (error: E) => void;
134
+ }
135
+ /**
136
+ * Create an infinite scroll async chunk with accumulate mode.
137
+ * Automatically handles pagination in accumulate mode for infinite scrolling.
138
+ */
139
+ declare function infiniteAsyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(fetcher: (params: P & {
140
+ page: number;
141
+ pageSize: number;
142
+ }) => Promise<{
143
+ data: T[];
144
+ hasMore?: boolean;
145
+ total?: number;
146
+ }>, options?: InfiniteAsyncChunkOptions<T, E>): PaginatedAsyncChunk<T[], E> & {
147
+ setParams: (params: Partial<P>) => void;
148
+ reload: (params?: Partial<P>) => Promise<void>;
149
+ refresh: (params?: Partial<P>) => Promise<void>;
86
150
  };
87
151
 
88
152
  type ChunkValue<T> = T extends Chunk<infer U> ? U : never;
@@ -120,6 +184,10 @@ declare function select<T, S>(sourceChunk: Chunk<T>, selector: (value: T) => S,
120
184
  declare function isValidChunkValue(value: unknown): boolean;
121
185
  declare function isChunk<T>(value: unknown): value is Chunk<T>;
122
186
  declare function once<T>(fn: () => T): () => T;
187
+ /**
188
+ * Combines multiple async chunks into a single chunk.
189
+ * The combined chunk tracks loading, error, and data states from all source chunks.
190
+ */
123
191
  declare function combineAsyncChunks<T extends Record<string, AsyncChunk<any>>>(chunks: T): Chunk<CombinedState<T>>;
124
192
 
125
193
  declare const logger: Middleware<any>;
@@ -172,4 +240,4 @@ declare namespace index {
172
240
  export { index_logger as logger, index_nonNegativeValidator as nonNegativeValidator, index_withHistory as withHistory, index_withPersistence as withPersistence };
173
241
  }
174
242
 
175
- export { type Chunk, type Middleware, asyncChunk, batch, chunk, combineAsyncChunks, computed, isChunk, isValidChunkValue, index as middleware, once, select };
243
+ export { type AsyncChunk, type AsyncState, type AsyncStateWithPagination, type Chunk, type Middleware, type PaginatedAsyncChunk, asyncChunk, batch, chunk, combineAsyncChunks, computed, infiniteAsyncChunk, isChunk, isValidChunkValue, index as middleware, once, select };
package/dist/index.js CHANGED
@@ -1,2 +1 @@
1
- var K=Object.defineProperty;var z=(e,n)=>{for(var r in n)K(e,r,{get:n[r],enumerable:true});};function B(e){return e!==null}function W(e){if(!e||typeof e!="object")return false;let n=e;return ["get","set","update","subscribe","derive","reset","destroy"].every(t=>typeof n[t]=="function")}function J(e){let n=false,r;return ()=>(n||(r=e(),n=true),r)}function U(e){let n=Object.keys(e).reduce((c,s)=>(c[s]=null,c),{}),r={loading:Object.keys(e).length>0,error:null,errors:{},data:n},t=p(r);Object.values(e);return Object.entries(e).forEach(([c,s])=>{s.subscribe(i=>{let f=t.get(),a=false,u=null,T={};Object.entries(e).forEach(([w,h])=>{let l=h.get();l.loading&&(a=true),l.error&&(u||(u=l.error),T[w]=l.error);}),t.set({loading:a,error:u,errors:T,data:{...f.data,[c]:i.data}});});}),t}function j(e,n){if(e===null)throw new Error("Value cannot be null.");let r=e,t=0;for(;t<n.length;){let o=n[t],c=typeof o=="function"?o:o.fn,s=typeof o=="function"?`index ${t}`:o.name||`index ${t}`,i=false,f=null;try{c(r,a=>{i=!0,f=a;});}catch(a){let u=a instanceof Error?a.message:String(a);throw new Error(`Middleware "${s}" threw an error: ${u}`)}if(!i)break;if(f===null)throw new Error(`Middleware at index ${t} returned null value.`);r=f,t++;}return r}function E(e,n){if(e===n)return true;if(!e||!n||typeof e!=typeof n)return false;if(Array.isArray(e)&&Array.isArray(n)){if(e.length!==n.length)return false;for(let r=0;r<e.length;r++)if(e[r]!==n[r])return false;return true}if(typeof e=="object"&&typeof n=="object"){let r=Object.keys(e),t=Object.keys(n);if(r.length!==t.length)return false;for(let o of r)if(!Object.prototype.hasOwnProperty.call(n,o)||e[o]!==n[o])return false;return true}return false}function v(e,n,r=""){if(typeof e=="object"&&e!==null&&typeof n=="object"&&n!==null){if(Array.isArray(e)&&Array.isArray(n)){if(e.length>0&&typeof e[0]=="object")for(let t=0;t<n.length;t++)v(e[0],n[t],`${r}[${t}]`);}else if(!Array.isArray(e)&&!Array.isArray(n)){let t=Object.keys(e),o=Object.keys(n),c=o.filter(s=>!t.includes(s));c.length>0&&(console.error(`\u{1F6A8} Stunk: Unknown properties detected at '${r||"root"}': ${c.join(", ")}. This might cause bugs.`),console.error("Expected keys:",t),console.error("Received keys:",o));for(let s of t)v(e[s],n[s],r?`${r}.${s}`:s);}}}var S=false,P=new Set,V=new Map,L=0;function G(e){let n=S;S=true;try{e();}finally{if(!n){S=false;let r=Array.from(P);P.clear(),r.forEach(t=>{let o=V.get(t);o&&o.notify();});}}}function p(e,n=[]){if(e===null)throw new Error("Initial value cannot be null.");let r=e,t=new Set,o=L++,c=()=>{t.forEach(h=>h(r));};V.set(o,{notify:c});let s=()=>{t.size!==0&&(S?P.add(o):c());},i=()=>r,f=h=>{let l;typeof h=="function"?l=h(r):l=h,v(r,l);let y=j(l,n);y!==r&&(r=y,s());},a=h=>{if(typeof h!="function")throw new Error("Callback must be a function.");return t.add(h),h(r),()=>t.delete(h)};return {get:i,set:f,subscribe:a,derive:h=>{if(typeof h!="function")throw new Error("Derive function must be a function.");let l=h(r),y=p(l),m=a(()=>{let D=h(r);y.set(D);}),g=y.destroy;return y.destroy=()=>{m(),g();},y},reset:()=>{r=e,s();},destroy:()=>{t.clear(),r=e,P.delete(o),V.delete(o);}}}function Q(e,n={}){let {initialData:r=null,onError:t,retryCount:o=0,retryDelay:c=1e3,refresh:s={},enabled:i=true}=n,{staleTime:f=0,cacheTime:a=5*60*1e3,refetchInterval:u}=s,T=e.length>0,w={loading:i&&!T,error:null,data:r,lastFetched:void 0},h=d=>T?d===void 0?false:Array.isArray(d)?d.every(x=>x!=null):true:true,l=p(w),y,m=null,g=null,D=()=>{let d=l.get();return d.lastFetched?Date.now()-d.lastFetched>f:true},N=()=>{l.set({...l.get(),data:r,lastFetched:void 0});},q=()=>{g&&clearTimeout(g),a>0&&(g=setTimeout(N,a));},k=async(d,x=o,b=false)=>{if(i){if(d!==void 0&&(y=d),!h(y)){l.set({...l.get(),loading:false});return}if(!(!b&&!D()&&l.get().data!==null&&f>0)){l.set({...l.get(),loading:true,error:null});try{let C=T?await e(...y):await e(),A=Date.now();l.set({loading:!1,error:null,data:C,lastFetched:A}),q();}catch(C){if(x>0)return await new Promise(A=>setTimeout(A,c)),k(d,x-1,b);l.set({loading:false,error:C,data:l.get().data,lastFetched:l.get().lastFetched}),t&&t(C);}}}};u&&u>0&&i&&(m=setInterval(()=>{i&&k(y,0,false);},u)),i&&!T&&k();let M=()=>{m&&(clearInterval(m),m=null),g&&(clearTimeout(g),g=null);};return {...l,reload:async(...d)=>{await k(d.length>0?d:void 0,o,true);},refresh:async(...d)=>{await k(d.length>0?d:void 0,o,false);},mutate:d=>{let x=l.get(),b=d(x.data);l.set({...x,data:b});},reset:()=>{M(),l.set({...w,loading:i}),i&&(k(),u&&u>0&&(m=setInterval(()=>{i&&k(y,0,false);},u)));},cleanup:M,setParams:(...d)=>{y=d,i&&h(d)&&k(d);}}}function X(e,n){let r=e.map(a=>a.get()),t=n(...r),o=p(t),c=o.set,s=false,i=()=>{let a=false;for(let u=0;u<e.length;u++){let T=e[u].get();T!==r[u]&&(r[u]=T,a=true);}if(a){let u=n(...r);u!==t&&(typeof u!="object"||typeof t!="object"||!E(u,t))&&(t=u,c(u)),s=false;}},f=e.map(a=>a.subscribe(()=>{s=true,i();}));return {...o,get:()=>(s&&i(),t),recompute:i,isDirty:()=>s,set:()=>{throw new Error("Cannot set values directly on computed. Modify the source chunk instead.")},reset:()=>(e.forEach(a=>{typeof a.reset=="function"&&a.reset();}),s=true,i(),t),destroy:()=>{f.forEach(a=>a()),o.destroy?.();}}}function O(e,n,r={}){let{useShallowEqual:t=false}=r,o=e.get(),c=n(o),s=p(c),i=()=>{let a=e.get(),u=n(a);o=a,(t?!E(u,c):u!==c)&&(c=u,s.set(u));},f=e.subscribe(i);return {get:()=>s.get(),set:()=>{throw new Error("Cannot set values directly on a selector. Modify the source chunk instead.")},subscribe:s.subscribe,derive:a=>O(s,a,r),reset:()=>{throw new Error("Cannot reset a selector chunk. Reset the source chunk instead.")},destroy:()=>{f(),s.destroy();}}}var $={};z($,{logger:()=>R,nonNegativeValidator:()=>I,withHistory:()=>F,withPersistence:()=>H});var R=(e,n)=>{console.log("Setting value:",e),n(e);};var I=(e,n)=>{if(e<0)throw new Error("Value must be non-negative!");n(e);};function F(e,n={}){let{maxHistory:r=100}=n,t=[e.get()],o=0,c=false,s={...e,set:i=>{if(c){e.set(i);return}let f;if(typeof i=="function"){let a=e.get();f=i(a);}else f=i;if(t.splice(o+1),t.push(f),t.length>r){console.warn("History limit reached. Removing oldest entries.");let a=t.length-r;t.splice(0,a),o=Math.max(0,o-a);}o=t.length-1,e.set(f);},undo:()=>{s.canUndo()&&(c=true,o--,s.set(t[o]),c=false);},redo:()=>{s.canRedo()&&(c=true,o++,s.set(t[o]),c=false);},canUndo:()=>o>0,canRedo:()=>o<t.length-1,getHistory:()=>[...t],clearHistory:()=>{let i=e.get();t.length=0,t.push(i),o=0;},destroy:()=>{t.length=0,e.destroy();}};return s}function H(e,n){let{key:r,storage:t=localStorage,serialize:o=JSON.stringify,deserialize:c=JSON.parse}=n;try{let s=t.getItem(r);if(s){let i=c(s);e.set(i);}}catch(s){console.error("Failed to load persisted state:",s);}return e.subscribe(s=>{try{let i=o(s);t.setItem(r,i);}catch(i){console.log("Failed to persist chunk",i);}}),e}export{Q as asyncChunk,G as batch,p as chunk,U as combineAsyncChunks,X as computed,W as isChunk,B as isValidChunkValue,$ as middleware,J as once,O as select};//# sourceMappingURL=index.js.map
2
- //# sourceMappingURL=index.js.map
1
+ var Y=Object.defineProperty;var Z=(e,a)=>{for(var r in a)Y(e,r,{get:a[r],enumerable:true});};function _(e){return e!==null}function ee(e){if(!e||typeof e!="object")return false;let a=e;return ["get","set","subscribe","derive","reset","destroy"].every(t=>typeof a[t]=="function")}function te(e){let a=false,r;return ()=>(a||(r=e(),a=true),r)}function re(e){let a=Object.keys(e).reduce((n,u)=>(n[u]=null,n),{}),r={loading:Object.keys(e).length>0,error:null,errors:{},data:a},t=k(r);return Object.entries(e).forEach(([n,u])=>{u.subscribe(o=>{let c=t.get(),l=false,s=null,d={};Object.entries(e).forEach(([h,x])=>{let p=x.get();p.loading&&(l=true),p.error&&(s||(s=p.error),d[h]=p.error);}),t.set({loading:l,error:s,errors:d,data:{...c.data,[n]:o.data}});});}),t}function N(e,a){if(e===null)throw new Error("Value cannot be null.");let r=e,t=0;for(;t<a.length;){let n=a[t],u=typeof n=="function"?n:n.fn,o=typeof n=="function"?`index ${t}`:n.name||`index ${t}`,c=false,l=null;try{u(r,s=>{c=!0,l=s;});}catch(s){let d=s instanceof Error?s.message:String(s);throw new Error(`Middleware "${o}" threw an error: ${d}`)}if(!c)break;if(l===null)throw new Error(`Middleware at index ${t} returned null value.`);r=l,t++;}return r}function M(e,a){if(e===a)return true;if(!e||!a||typeof e!=typeof a)return false;if(Array.isArray(e)&&Array.isArray(a)){if(e.length!==a.length)return false;for(let r=0;r<e.length;r++)if(e[r]!==a[r])return false;return true}if(typeof e=="object"&&typeof a=="object"){let r=Object.keys(e),t=Object.keys(a);if(r.length!==t.length)return false;for(let n of r)if(!Object.prototype.hasOwnProperty.call(a,n)||e[n]!==a[n])return false;return true}return false}function D(e,a,r=""){if(typeof e=="object"&&e!==null&&typeof a=="object"&&a!==null){if(Array.isArray(e)&&Array.isArray(a)){if(e.length>0&&typeof e[0]=="object")for(let t=0;t<a.length;t++)D(e[0],a[t],`${r}[${t}]`);}else if(!Array.isArray(e)&&!Array.isArray(a)){let t=Object.keys(e),n=Object.keys(a),u=n.filter(o=>!t.includes(o));u.length>0&&(console.error(`\u{1F6A8} Stunk: Unknown properties detected at '${r||"root"}': ${u.join(", ")}. This might cause bugs.`),console.error("Expected keys:",t),console.error("Received keys:",n));for(let o of t)D(e[o],a[o],r?`${r}.${o}`:o);}}}var V=false,R=new Set,O=new Map,ne=0;function ae(e){let a=V;V=true;try{e();}finally{if(!a){V=false;let r=Array.from(R);R.clear(),r.forEach(t=>{let n=O.get(t);n&&n.notify();});}}}function k(e,a=[]){if(e===null)throw new Error("Initial value cannot be null.");let r=e,t=new Set,n=ne++,u=()=>{t.forEach(p=>p(r));};O.set(n,{notify:u});let o=()=>{t.size!==0&&(V?R.add(n):u());},c=()=>r,l=p=>{let y;typeof p=="function"?y=p(r):y=p,D(r,y);let b=N(y,a);b!==r&&(r=b,o());},s=p=>{if(typeof p!="function")throw new Error("Callback must be a function.");return t.add(p),p(r),()=>t.delete(p)};return {get:c,set:l,subscribe:s,derive:p=>{if(typeof p!="function")throw new Error("Derive function must be a function.");let y=p(r),b=k(y),f=s(()=>{let E=p(r);b.set(E);}),g=b.destroy;return b.destroy=()=>{f(),g();},b},reset:()=>{r=e,o();},destroy:()=>{t.clear(),r=e,R.delete(n),O.delete(n);}}}function z(e,a={}){let{initialData:r=null,onError:t,retryCount:n=0,retryDelay:u=1e3,refresh:o={},pagination:c,enabled:l=true}=a,{staleTime:s=0,cacheTime:d=300*1e3,refetchInterval:h}=o,x=!!c,p=c?.mode||"replace",y=e.length>0,b={loading:l&&!y,error:null,data:r,lastFetched:void 0,pagination:x?{page:c.initialPage||1,pageSize:c.pageSize||10}:void 0},f=k(b),g={},E=null,v=null,L=()=>{let i=f.get();return !i.lastFetched||s===0?true:Date.now()-i.lastFetched>s},G=()=>{f.set({...f.get(),data:r,lastFetched:void 0});},Q=()=>{v&&clearTimeout(v),d>0&&(v=setTimeout(G,d));},m=async(i,T=n,A=false)=>{if(!l||(i!==void 0&&(g={...g,...i}),!A&&!L()&&f.get().data!==null))return;let C=f.get();f.set({...C,loading:true,error:null});try{let S={...g};x&&C.pagination&&(S.page=C.pagination.page,S.pageSize=C.pagination.pageSize);let P=y?await e(S):await e(),w,H,$;if(P&&typeof P=="object"&&"data"in P){let j=P;w=j.data,H=j.total,$=j.hasMore;}else w=P;x&&p==="accumulate"&&C.data&&Array.isArray(C.data)&&Array.isArray(w)&&(w=[...C.data,...w]);let X=Date.now();f.set({loading:!1,error:null,data:w,lastFetched:X,pagination:x?{...C.pagination,total:H,hasMore:$}:void 0}),Q();}catch(S){if(T>0)return await new Promise(w=>setTimeout(w,u)),m(i,T-1,A);let P=f.get();f.set({loading:false,error:S,data:P.data,lastFetched:P.lastFetched,pagination:P.pagination}),t&&t(S);}};h&&h>0&&l&&(E=setInterval(()=>{l&&m(void 0,0,false);},h)),l&&!y&&m();let F=()=>{E&&(clearInterval(E),E=null),v&&(clearTimeout(v),v=null);},I={...f,reload:async i=>{await m(i,n,true);},refresh:async i=>{await m(i,n,false);},mutate:i=>{let T=f.get(),A=i(T.data);f.set({...T,data:A});},reset:()=>{F(),g={},f.set({...b,loading:l&&!y}),l&&!y&&(m(),h&&h>0&&(E=setInterval(()=>{l&&m(void 0,0,false);},h)));},cleanup:F,setParams:i=>{g={...g,...i},l&&m(i,n,true);}};return x?{...I,nextPage:async()=>{let i=f.get();i.pagination&&i.pagination.hasMore!==false&&(f.set({...i,pagination:{...i.pagination,page:i.pagination.page+1}}),await m(g,n,true));},prevPage:async()=>{let i=f.get();!i.pagination||i.pagination.page<=1||(f.set({...i,pagination:{...i.pagination,page:i.pagination.page-1}}),await m(g,n,true));},goToPage:async i=>{let T=f.get();!T.pagination||i<1||(f.set({...T,pagination:{...T.pagination,page:i}}),await m(g,n,true));},resetPagination:async()=>{let i=f.get();if(!i.pagination)return;let T=c?.initialPage||1;f.set({...i,data:p==="accumulate"?r:i.data,pagination:{...i.pagination,page:T}}),await m(g,n,true);}}:I}function oe(e,a={}){let{pageSize:r=10,staleTime:t,cacheTime:n,retryCount:u,retryDelay:o,onError:c}=a;return z(e,{pagination:{pageSize:r,mode:"accumulate",initialPage:1},refresh:{staleTime:t,cacheTime:n},retryCount:u,retryDelay:o,onError:c})}function se(e,a){let r=e.map(s=>s.get()),t=a(...r),n=k(t),u=n.set,o=false,c=()=>{let s=false;for(let d=0;d<e.length;d++){let h=e[d].get();h!==r[d]&&(r[d]=h,s=true);}if(s){let d=a(...r);d!==t&&(typeof d!="object"||typeof t!="object"||!M(d,t))&&(t=d,u(d)),o=false;}},l=e.map(s=>s.subscribe(()=>{o=true,c();}));return {...n,get:()=>(o&&c(),t),recompute:c,isDirty:()=>o,set:()=>{throw new Error("Cannot set values directly on computed. Modify the source chunk instead.")},reset:()=>(e.forEach(s=>{typeof s.reset=="function"&&s.reset();}),o=true,c(),t),destroy:()=>{l.forEach(s=>s()),n.destroy?.();}}}function q(e,a,r={}){let{useShallowEqual:t=false}=r,n=e.get(),u=a(n),o=k(u),c=()=>{let s=e.get(),d=a(s);n=s,(t?!M(d,u):d!==u)&&(u=d,o.set(d));},l=e.subscribe(c);return {get:()=>o.get(),set:()=>{throw new Error("Cannot set values directly on a selector. Modify the source chunk instead.")},subscribe:o.subscribe,derive:s=>q(o,s,r),reset:()=>{throw new Error("Cannot reset a selector chunk. Reset the source chunk instead.")},destroy:()=>{l(),o.destroy();}}}var U={};Z(U,{logger:()=>K,nonNegativeValidator:()=>W,withHistory:()=>B,withPersistence:()=>J});var K=(e,a)=>{console.log("Setting value:",e),a(e);};var W=(e,a)=>{if(e<0)throw new Error("Value must be non-negative!");a(e);};function B(e,a={}){let{maxHistory:r=100}=a,t=[e.get()],n=0,u=false,o={...e,set:c=>{if(u){e.set(c);return}let l;if(typeof c=="function"){let s=e.get();l=c(s);}else l=c;if(t.splice(n+1),t.push(l),t.length>r){console.warn("History limit reached. Removing oldest entries.");let s=t.length-r;t.splice(0,s),n=Math.max(0,n-s);}n=t.length-1,e.set(l);},undo:()=>{o.canUndo()&&(u=true,n--,o.set(t[n]),u=false);},redo:()=>{o.canRedo()&&(u=true,n++,o.set(t[n]),u=false);},canUndo:()=>n>0,canRedo:()=>n<t.length-1,getHistory:()=>[...t],clearHistory:()=>{let c=e.get();t.length=0,t.push(c),n=0;},destroy:()=>{t.length=0,e.destroy();}};return o}function J(e,a){let{key:r,storage:t=localStorage,serialize:n=JSON.stringify,deserialize:u=JSON.parse}=a;try{let o=t.getItem(r);if(o){let c=u(o);e.set(c);}}catch(o){console.error("Failed to load persisted state:",o);}return e.subscribe(o=>{try{let c=n(o);t.setItem(r,c);}catch(c){console.log("Failed to persist chunk",c);}}),e}export{z as asyncChunk,ae as batch,k as chunk,re as combineAsyncChunks,se as computed,oe as infiniteAsyncChunk,ee as isChunk,_ as isValidChunkValue,U as middleware,te as once,q as select};
@@ -1,2 +1 @@
1
- var d=(e,n)=>{console.log("Setting value:",e),n(e);};var u=(e,n)=>{if(e<0)throw new Error("Value must be non-negative!");n(e);};function g(e,n={}){let{maxHistory:l=100}=n,t=[e.get()],r=0,s=false,o={...e,set:i=>{if(s){e.set(i);return}let c;if(typeof i=="function"){let a=e.get();c=i(a);}else c=i;if(t.splice(r+1),t.push(c),t.length>l){console.warn("History limit reached. Removing oldest entries.");let a=t.length-l;t.splice(0,a),r=Math.max(0,r-a);}r=t.length-1,e.set(c);},undo:()=>{o.canUndo()&&(s=true,r--,o.set(t[r]),s=false);},redo:()=>{o.canRedo()&&(s=true,r++,o.set(t[r]),s=false);},canUndo:()=>r>0,canRedo:()=>r<t.length-1,getHistory:()=>[...t],clearHistory:()=>{let i=e.get();t.length=0,t.push(i),r=0;},destroy:()=>{t.length=0,e.destroy();}};return o}function h(e,n){let{key:l,storage:t=localStorage,serialize:r=JSON.stringify,deserialize:s=JSON.parse}=n;try{let o=t.getItem(l);if(o){let i=s(o);e.set(i);}}catch(o){console.error("Failed to load persisted state:",o);}return e.subscribe(o=>{try{let i=r(o);t.setItem(l,i);}catch(i){console.log("Failed to persist chunk",i);}}),e}export{d as logger,u as nonNegativeValidator,g as withHistory,h as withPersistence};//# sourceMappingURL=index.js.map
2
- //# sourceMappingURL=index.js.map
1
+ var d=(e,n)=>{console.log("Setting value:",e),n(e);};var u=(e,n)=>{if(e<0)throw new Error("Value must be non-negative!");n(e);};function g(e,n={}){let{maxHistory:l=100}=n,t=[e.get()],r=0,s=false,o={...e,set:i=>{if(s){e.set(i);return}let c;if(typeof i=="function"){let a=e.get();c=i(a);}else c=i;if(t.splice(r+1),t.push(c),t.length>l){console.warn("History limit reached. Removing oldest entries.");let a=t.length-l;t.splice(0,a),r=Math.max(0,r-a);}r=t.length-1,e.set(c);},undo:()=>{o.canUndo()&&(s=true,r--,o.set(t[r]),s=false);},redo:()=>{o.canRedo()&&(s=true,r++,o.set(t[r]),s=false);},canUndo:()=>r>0,canRedo:()=>r<t.length-1,getHistory:()=>[...t],clearHistory:()=>{let i=e.get();t.length=0,t.push(i),r=0;},destroy:()=>{t.length=0,e.destroy();}};return o}function h(e,n){let{key:l,storage:t=localStorage,serialize:r=JSON.stringify,deserialize:s=JSON.parse}=n;try{let o=t.getItem(l);if(o){let i=s(o);e.set(i);}}catch(o){console.error("Failed to load persisted state:",o);}return e.subscribe(o=>{try{let i=r(o);t.setItem(l,i);}catch(i){console.log("Failed to persist chunk",i);}}),e}export{d as logger,u as nonNegativeValidator,g as withHistory,h as withPersistence};
@@ -49,34 +49,93 @@ interface AsyncState<T, E extends Error> {
49
49
  data: T | null;
50
50
  lastFetched?: number;
51
51
  }
52
- interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncState<T, E>> {
53
- /** Reload the data from the source. */
54
- reload: () => Promise<void>;
52
+ interface PaginationState {
53
+ page: number;
54
+ pageSize: number;
55
+ total?: number;
56
+ hasMore?: boolean;
57
+ }
58
+ interface AsyncStateWithPagination<T, E extends Error> extends AsyncState<T, E> {
59
+ pagination?: PaginationState;
60
+ }
61
+ interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncStateWithPagination<T, E>> {
62
+ /** Force reload data */
63
+ reload: (params?: any) => Promise<void>;
55
64
  /** Smart refresh - respects stale time */
56
- refresh: () => Promise<void>;
57
- /** Mutate the data directly. */
65
+ refresh: (params?: any) => Promise<void>;
66
+ /** Mutate data directly */
58
67
  mutate: (mutator: (currentData: T | null) => T) => void;
59
- /** Reset the state to the initial value. */
68
+ /** Reset to initial state */
60
69
  reset: () => void;
61
- /** Clean up intervals and timeouts */
70
+ /** Clean up intervals */
62
71
  cleanup: () => void;
63
72
  }
73
+ interface PaginatedAsyncChunk<T, E extends Error = Error> extends AsyncChunk<T, E> {
74
+ /** Load next page */
75
+ nextPage: () => Promise<void>;
76
+ /** Load previous page */
77
+ prevPage: () => Promise<void>;
78
+ /** Go to specific page */
79
+ goToPage: (page: number) => Promise<void>;
80
+ /** Reset pagination to first page */
81
+ resetPagination: () => Promise<void>;
82
+ }
64
83
 
65
- /**
66
- * A hook that handles asynchronous state with built-in reactivity.
67
- * Provides loading, error, and data states with full asyncChunk functionality.
68
- */
69
- declare function useAsyncChunk<T, E extends Error = Error, P extends any[] = []>(asyncChunk: AsyncChunk<T, E> | (AsyncChunk<T, E> & {
70
- setParams: (...params: P) => void;
71
- }), params?: P): {
84
+ interface UseAsyncChunkResult<T, E extends Error, P extends Record<string, any>> {
72
85
  data: T | null;
73
86
  loading: boolean;
74
87
  error: E | null;
75
- lastFetched: number | undefined;
76
- reload: (...params: any[]) => any;
77
- refresh: (...params: any[]) => any;
88
+ lastFetched?: number;
89
+ reload: (params?: Partial<P>) => Promise<void>;
90
+ refresh: (params?: Partial<P>) => Promise<void>;
78
91
  mutate: (mutator: (currentData: T | null) => T) => void;
79
92
  reset: () => void;
80
- };
93
+ }
94
+ interface UseAsyncChunkResultWithParams<T, E extends Error, P extends Record<string, any>> extends UseAsyncChunkResult<T, E, P> {
95
+ setParams: (params: Partial<P>) => void;
96
+ }
97
+ interface UseAsyncChunkResultWithPagination<T, E extends Error, P extends Record<string, any>> extends UseAsyncChunkResult<T, E, P> {
98
+ pagination?: PaginationState;
99
+ nextPage: () => Promise<void>;
100
+ prevPage: () => Promise<void>;
101
+ goToPage: (page: number) => Promise<void>;
102
+ resetPagination: () => Promise<void>;
103
+ }
104
+ interface UseAsyncChunkResultWithParamsAndPagination<T, E extends Error, P extends Record<string, any>> extends UseAsyncChunkResultWithParams<T, E, P>, Omit<UseAsyncChunkResultWithPagination<T, E, P>, keyof UseAsyncChunkResult<T, E, P>> {
105
+ }
106
+ /**
107
+ * A hook that handles asynchronous state with built-in reactivity.
108
+ * Provides loading, error, and data states with full asyncChunk functionality.
109
+ */
110
+ interface UseAsyncChunkOptions<P extends Record<string, any> = {}> {
111
+ /** Initial parameters to pass to the fetcher */
112
+ initialParams?: Partial<P>;
113
+ /** Force fetch on mount, even without params (default: false) */
114
+ fetchOnMount?: boolean;
115
+ }
116
+ declare function useAsyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(asyncChunk: PaginatedAsyncChunk<T, E> & {
117
+ setParams: (params: Partial<P>) => void;
118
+ }, options?: UseAsyncChunkOptions<P> | Partial<P>): UseAsyncChunkResultWithParamsAndPagination<T, E, P>;
119
+ declare function useAsyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(asyncChunk: PaginatedAsyncChunk<T, E>, options?: UseAsyncChunkOptions<P> | Partial<P>): UseAsyncChunkResultWithPagination<T, E, P>;
120
+ declare function useAsyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(asyncChunk: AsyncChunk<T, E> & {
121
+ setParams: (params: Partial<P>) => void;
122
+ }, options?: UseAsyncChunkOptions<P> | Partial<P>): UseAsyncChunkResultWithParams<T, E, P>;
123
+ declare function useAsyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(asyncChunk: AsyncChunk<T, E>, options?: UseAsyncChunkOptions<P> | Partial<P>): UseAsyncChunkResult<T, E, P>;
124
+
125
+ interface UseInfiniteAsyncChunkOptions<P extends Record<string, any>> extends Omit<UseAsyncChunkOptions<P>, 'initialParams'> {
126
+ /** Initial parameters (page and pageSize added automatically) */
127
+ initialParams?: Omit<Partial<P>, 'page' | 'pageSize'>;
128
+ /** Enable auto-loading on scroll (default: true) */
129
+ autoLoad?: boolean;
130
+ /** Intersection observer threshold (default: 1.0) */
131
+ threshold?: number;
132
+ }
133
+ /**
134
+ * Hook for infinite scroll functionality.
135
+ * Automatically loads more data when scrolling to the bottom.
136
+ */
137
+ declare function useInfiniteAsyncChunk<T, E extends Error = Error, P extends Record<string, any> = {}>(asyncChunk: PaginatedAsyncChunk<T[], E> & {
138
+ setParams: (params: Partial<P>) => void;
139
+ }, options?: UseInfiniteAsyncChunkOptions<P>): any;
81
140
 
82
- export { useAsyncChunk, useChunk, useChunkProperty, useChunkValue, useChunkValues, useComputed, useDerive };
141
+ export { useAsyncChunk, useChunk, useChunkProperty, useChunkValue, useChunkValues, useComputed, useDerive, useInfiniteAsyncChunk };
@@ -1,2 +1 @@
1
- import {useState,useEffect,useCallback,useRef,useMemo}from'react';var E=new Set,D=new Map,P=0;function y(e,r=[]){if(e===null)throw new Error("Initial value cannot be null.");let t=e,n=new Set,o=P++,a=()=>{n.forEach(d=>d(t));};D.set(o,{notify:a});let s=()=>{n.size!==0&&(a());},f=()=>t,c=d=>{let l;typeof d=="function"?l=d(t):l=d,b(t,l);let p=V(l,r);p!==t&&(t=p,s());},u=d=>{if(typeof d!="function")throw new Error("Callback must be a function.");return n.add(d),d(t),()=>n.delete(d)};return {get:f,set:c,subscribe:u,derive:d=>{if(typeof d!="function")throw new Error("Derive function must be a function.");let l=d(t),p=y(l),S=u(()=>{let M=d(t);p.set(M);}),j=p.destroy;return p.destroy=()=>{S(),j();},p},reset:()=>{t=e,s();},destroy:()=>{n.clear(),t=e,E.delete(o),D.delete(o);}}}function V(e,r){if(e===null)throw new Error("Value cannot be null.");let t=e,n=0;for(;n<r.length;){let o=r[n],a=typeof o=="function"?o:o.fn,s=typeof o=="function"?`index ${n}`:o.name||`index ${n}`,f=false,c=null;try{a(t,u=>{f=!0,c=u;});}catch(u){let i=u instanceof Error?u.message:String(u);throw new Error(`Middleware "${s}" threw an error: ${i}`)}if(!f)break;if(c===null)throw new Error(`Middleware at index ${n} returned null value.`);t=c,n++;}return t}function k(e,r){if(e===r)return true;if(!e||!r||typeof e!=typeof r)return false;if(Array.isArray(e)&&Array.isArray(r)){if(e.length!==r.length)return false;for(let t=0;t<e.length;t++)if(e[t]!==r[t])return false;return true}if(typeof e=="object"&&typeof r=="object"){let t=Object.keys(e),n=Object.keys(r);if(t.length!==n.length)return false;for(let o of t)if(!Object.prototype.hasOwnProperty.call(r,o)||e[o]!==r[o])return false;return true}return false}function b(e,r,t=""){if(typeof e=="object"&&e!==null&&typeof r=="object"&&r!==null){if(Array.isArray(e)&&Array.isArray(r)){if(e.length>0&&typeof e[0]=="object")for(let n=0;n<r.length;n++)b(e[0],r[n],`${t}[${n}]`);}else if(!Array.isArray(e)&&!Array.isArray(r)){let n=Object.keys(e),o=Object.keys(r),a=o.filter(s=>!n.includes(s));a.length>0&&(console.error(`\u{1F6A8} Stunk: Unknown properties detected at '${t||"root"}': ${a.join(", ")}. This might cause bugs.`),console.error("Expected keys:",n),console.error("Received keys:",o));for(let s of n)b(e[s],r[s],t?`${t}.${s}`:s);}}}function w(e,r,t={}){let{useShallowEqual:n=false}=t,o=e.get(),a=r(o),s=y(a),f=()=>{let u=e.get(),i=r(u);o=u,(n?!k(i,a):i!==a)&&(a=i,s.set(i));},c=e.subscribe(f);return {get:()=>s.get(),set:()=>{throw new Error("Cannot set values directly on a selector. Modify the source chunk instead.")},subscribe:s.subscribe,derive:u=>w(s,u,t),reset:()=>{throw new Error("Cannot reset a selector chunk. Reset the source chunk instead.")},destroy:()=>{c(),s.destroy();}}}function m(e,r){let t=r?w(e,r):e,[n,o]=useState(()=>t.get());useEffect(()=>{let c=t.subscribe(u=>{o(()=>u);});return ()=>c()},[t]);let a=useCallback(c=>{e.set(c);},[e]),s=useCallback(()=>{e.reset();},[e]),f=useCallback(()=>{e.destroy();},[e]);return [n,a,s,f]}function N(e,r){let t=useRef(r);useEffect(()=>{t.current=r;},[r]);let n=useMemo(()=>e.derive(a=>t.current(a)),[e]),[o]=m(n);return o}function A(e,r){let t=e.map(u=>u.get()),n=r(...t),o=y(n),a=o.set,s=false,f=()=>{let u=false;for(let i=0;i<e.length;i++){let T=e[i].get();T!==t[i]&&(t[i]=T,u=true);}if(u){let i=r(...t);i!==n&&(typeof i!="object"||typeof n!="object"||!k(i,n))&&(n=i,a(i)),s=false;}},c=e.map(u=>u.subscribe(()=>{s=true,f();}));return {...o,get:()=>(s&&f(),n),recompute:f,isDirty:()=>s,set:()=>{throw new Error("Cannot set values directly on computed. Modify the source chunk instead.")},reset:()=>(e.forEach(u=>{typeof u.reset=="function"&&u.reset();}),s=true,f(),n),destroy:()=>{c.forEach(u=>u()),o.destroy?.();}}}function z(e,r){let t=useMemo(()=>A(e,r),[...e]),[n,o]=useState(()=>t.get());return useEffect(()=>{let a=t.subscribe(s=>{o(s);});return ()=>{a();}},[t]),n}function C(e,r){let[t]=m(e,r);return t}function G(e,r){let t=useMemo(()=>n=>n[r],[r]);return C(e,t)}function Q(e){let[r,t]=useState(()=>e.map(n=>n.get()));return useEffect(()=>{let n=e.map((o,a)=>o.subscribe(s=>{t(f=>{let c=[...f];return c[a]=s,c});}));return ()=>{n.forEach(o=>o());}},[e]),r}function X(e,r){let[t,n]=useState(()=>e.get());useEffect(()=>{let l=e.subscribe(p=>{n(p);});return ()=>{l();}},[e]),useEffect(()=>{r&&"setParams"in e&&e.setParams(...r);},[e,...r||[]]),useEffect(()=>()=>{e.cleanup();},[e]);let o=useCallback((...l)=>"setParams"in e&&l.length>0?e.reload(...l):e.reload(),[e]),a=useCallback((...l)=>"setParams"in e&&l.length>0?e.refresh(...l):e.refresh(),[e]),s=useCallback(l=>e.mutate(l),[e]),f=useCallback(()=>e.reset(),[e]),c=useCallback((...l)=>{"setParams"in e&&e.setParams(...l);},[e]),{data:u,loading:i,error:T,lastFetched:g}=t,d={data:u,loading:i,error:T,lastFetched:g,reload:o,refresh:a,mutate:s,reset:f};return "setParams"in e&&(d.setParams=c),d}export{X as useAsyncChunk,m as useChunk,G as useChunkProperty,C as useChunkValue,Q as useChunkValues,z as useComputed,N as useDerive};//# sourceMappingURL=index.js.map
2
- //# sourceMappingURL=index.js.map
1
+ import {useState,useEffect,useCallback,useRef,useMemo}from'react';var S=new Set,D=new Map,j=0;function P(e,r=[]){if(e===null)throw new Error("Initial value cannot be null.");let t=e,n=new Set,s=j++,i=()=>{n.forEach(f=>f(t));};D.set(s,{notify:i});let o=()=>{n.size!==0&&(i());},l=()=>t,c=f=>{let h;typeof f=="function"?h=f(t):h=f,b(t,h);let p=M(h,r);p!==t&&(t=p,o());},a=f=>{if(typeof f!="function")throw new Error("Callback must be a function.");return n.add(f),f(t),()=>n.delete(f)};return {get:l,set:c,subscribe:a,derive:f=>{if(typeof f!="function")throw new Error("Derive function must be a function.");let h=f(t),p=P(h),m=a(()=>{let k=f(t);p.set(k);}),d=p.destroy;return p.destroy=()=>{m(),d();},p},reset:()=>{t=e,o();},destroy:()=>{n.clear(),t=e,S.delete(s),D.delete(s);}}}function M(e,r){if(e===null)throw new Error("Value cannot be null.");let t=e,n=0;for(;n<r.length;){let s=r[n],i=typeof s=="function"?s:s.fn,o=typeof s=="function"?`index ${n}`:s.name||`index ${n}`,l=false,c=null;try{i(t,a=>{l=!0,c=a;});}catch(a){let u=a instanceof Error?a.message:String(a);throw new Error(`Middleware "${o}" threw an error: ${u}`)}if(!l)break;if(c===null)throw new Error(`Middleware at index ${n} returned null value.`);t=c,n++;}return t}function x(e,r){if(e===r)return true;if(!e||!r||typeof e!=typeof r)return false;if(Array.isArray(e)&&Array.isArray(r)){if(e.length!==r.length)return false;for(let t=0;t<e.length;t++)if(e[t]!==r[t])return false;return true}if(typeof e=="object"&&typeof r=="object"){let t=Object.keys(e),n=Object.keys(r);if(t.length!==n.length)return false;for(let s of t)if(!Object.prototype.hasOwnProperty.call(r,s)||e[s]!==r[s])return false;return true}return false}function b(e,r,t=""){if(typeof e=="object"&&e!==null&&typeof r=="object"&&r!==null){if(Array.isArray(e)&&Array.isArray(r)){if(e.length>0&&typeof e[0]=="object")for(let n=0;n<r.length;n++)b(e[0],r[n],`${t}[${n}]`);}else if(!Array.isArray(e)&&!Array.isArray(r)){let n=Object.keys(e),s=Object.keys(r),i=s.filter(o=>!n.includes(o));i.length>0&&(console.error(`\u{1F6A8} Stunk: Unknown properties detected at '${t||"root"}': ${i.join(", ")}. This might cause bugs.`),console.error("Expected keys:",n),console.error("Received keys:",s));for(let o of n)b(e[o],r[o],t?`${t}.${o}`:o);}}}function E(e,r,t={}){let{useShallowEqual:n=false}=t,s=e.get(),i=r(s),o=P(i),l=()=>{let a=e.get(),u=r(a);s=a,(n?!x(u,i):u!==i)&&(i=u,o.set(u));},c=e.subscribe(l);return {get:()=>o.get(),set:()=>{throw new Error("Cannot set values directly on a selector. Modify the source chunk instead.")},subscribe:o.subscribe,derive:a=>E(o,a,t),reset:()=>{throw new Error("Cannot reset a selector chunk. Reset the source chunk instead.")},destroy:()=>{c(),o.destroy();}}}function C(e,r){let t=r?E(e,r):e,[n,s]=useState(()=>t.get());useEffect(()=>{let c=t.subscribe(a=>{s(()=>a);});return ()=>c()},[t]);let i=useCallback(c=>{e.set(c);},[e]),o=useCallback(()=>{e.reset();},[e]),l=useCallback(()=>{e.destroy();},[e]);return [n,i,o,l]}function F(e,r){let t=useRef(r);useEffect(()=>{t.current=r;},[r]);let n=useMemo(()=>e.derive(i=>t.current(i)),[e]),[s]=C(n);return s}function V(e,r){let t=e.map(a=>a.get()),n=r(...t),s=P(n),i=s.set,o=false,l=()=>{let a=false;for(let u=0;u<e.length;u++){let T=e[u].get();T!==t[u]&&(t[u]=T,a=true);}if(a){let u=r(...t);u!==n&&(typeof u!="object"||typeof n!="object"||!x(u,n))&&(n=u,i(u)),o=false;}},c=e.map(a=>a.subscribe(()=>{o=true,l();}));return {...s,get:()=>(o&&l(),n),recompute:l,isDirty:()=>o,set:()=>{throw new Error("Cannot set values directly on computed. Modify the source chunk instead.")},reset:()=>(e.forEach(a=>{typeof a.reset=="function"&&a.reset();}),o=true,l(),n),destroy:()=>{c.forEach(a=>a()),s.destroy?.();}}}function L(e,r){let t=useMemo(()=>V(e,r),[...e]),[n,s]=useState(()=>t.get());return useEffect(()=>{let i=t.subscribe(o=>{s(o);});return ()=>{i();}},[t]),n}function v(e,r){let[t]=C(e,r);return t}function G(e,r){let t=useMemo(()=>n=>n[r],[r]);return v(e,t)}function X(e){let[r,t]=useState(()=>e.map(n=>n.get()));return useEffect(()=>{let n=e.map((s,i)=>s.subscribe(o=>{t(l=>{let c=[...l];return c[i]=o,c});}));return ()=>{n.forEach(s=>s());}},[e]),r}function Z(e){return "nextPage"in e}function O(e){return "setParams"in e}function R(e,r){let{initialParams:t,fetchOnMount:n}=typeof r=="object"&&("initialParams"in r||"fetchOnMount"in r)?r:{initialParams:r,fetchOnMount:false},[s,i]=useState(()=>e.get());useEffect(()=>e.subscribe(k=>{i(k);}),[e]),useEffect(()=>{t&&O(e)?e.setParams(t):n&&!t&&e.reload();},[e]),useEffect(()=>()=>{e.cleanup();},[e]);let o=useCallback(d=>e.reload(d),[e]),l=useCallback(d=>e.refresh(d),[e]),c=useCallback(d=>e.mutate(d),[e]),a=useCallback(()=>e.reset(),[e]),u=useCallback(d=>{"setParams"in e&&e.setParams(d);},[e]),{data:T,loading:g,error:f,lastFetched:h,pagination:p}=s,m={data:T,loading:g,error:f,lastFetched:h,reload:o,refresh:l,mutate:c,reset:a};if(O(e)&&(m.setParams=u),Z(e)){let d=m;d.pagination=p,d.nextPage=useCallback(()=>e.nextPage(),[e]),d.prevPage=useCallback(()=>e.prevPage(),[e]),d.goToPage=useCallback(k=>e.goToPage(k),[e]),d.resetPagination=useCallback(()=>e.resetPagination(),[e]);}return m}function re(e,r={}){let{initialParams:t,autoLoad:n=true,threshold:s=1,...i}=r,o=useRef(null),l=R(e,{initialParams:{...t,page:1,pageSize:e.get().pagination?.pageSize||10},...i}),{loading:c,pagination:a,nextPage:u}=l;useEffect(()=>{if(!n)return;let g=new IntersectionObserver(h=>{h[0].isIntersecting&&!c&&a?.hasMore&&u();},{threshold:s}),f=o.current;return f&&g.observe(f),()=>{f&&g.unobserve(f);}},[c,a?.hasMore,u,n,s]);let T=useCallback(()=>{!c&&a?.hasMore&&u();},[c,a?.hasMore,u]);return {...l,loadMore:T,observerTarget:o,hasMore:a?.hasMore??false,isFetchingMore:c&&(l.data?.length??0)>0}}export{R as useAsyncChunk,C as useChunk,G as useChunkProperty,v as useChunkValue,X as useChunkValues,L as useComputed,F as useDerive,re as useInfiniteAsyncChunk};
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "stunk",
3
- "version": "2.5.0",
4
- "description": "Stunk is a lightweight, framework-agnostic state management library for JavaScript and TypeScript. It uses chunk-based state units for efficient updates, reactivity, and performance optimization in React, Vue, Svelte, and Vanilla JS/TS applications.",
3
+ "version": "2.7.1",
4
+ "description": "Stunk is a lightweight, framework-agnostic state management library for JavaScript and TypeScript. It uses chunk-based state units for efficient updates, reactivity, and performance optimization in React, Vue(WIP), Svelte(Coming soon), and Vanilla JS/TS applications.",
5
5
  "type": "module",
6
6
  "repository": {
7
7
  "type": "git",
@@ -67,8 +67,12 @@
67
67
  "url": "https://github.com/I-am-abdulazeez"
68
68
  },
69
69
  {
70
- "name": "AbdulAzeez",
71
- "url": "https://github.com/I-am-abdulazeez"
70
+ "name": "Chiboy",
71
+ "url": "https://github.com/chibx"
72
+ },
73
+ {
74
+ "name": "Idris",
75
+ "url": "https://github.com/Dreezy305"
72
76
  }
73
77
  ],
74
78
  "license": "MIT",
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/utils.ts","../src/core/core.ts","../src/core/asyncChunk.ts","../src/core/computed.ts","../src/core/selector.ts","../src/middleware/index.ts","../src/middleware/logger.ts","../src/middleware/validator.ts","../src/middleware/history.ts","../src/middleware/persistence.ts"],"names":["isValidChunkValue","value","isChunk","chunk","method","once","fn","called","result","combineAsyncChunks","chunks","initialData","acc","key","initialState","combined","asyncChunk","state","currentState","hasLoading","firstError","allErrors","chunkKey","chunkState","processMiddleware","initialValue","middleware","currentValue","index","current","middlewareFn","middlewareName","nextCalled","nextValue","val","error","errorMessage","shallowEqual","a","b","i","keysA","keysB","validateObjectShape","original","updated","path","originalKeys","updatedKeys","extraKeys","isBatching","dirtyChunks","chunkRegistry","chunkIdCounter","batch","callback","wasBatchingBefore","id","subscribers","chunkId","notify","subscriber","notifySubscribers","get","set","newValueOrUpdater","newValue","processedValue","subscribe","initialDerivedValue","derivedChunk","unsubscribe","newDerivedValue","originalDestroy","fetcher","options","onError","retryCount","retryDelay","refreshConfig","enabled","staleTime","cacheTime","refetchInterval","expectsParams","hasValidParams","params","p","baseChunk","currentParams","intervalId","cacheTimeoutId","isStale","clearCache","setCacheTimeout","fetchData","retries","force","data","now","resolve","cleanup","mutator","newData","computed","dependencies","computeFn","dependencyValues","dep","cachedValue","computedChunk","originalSet","isDirty","recompute","hasChanges","unsubs","unsub","select","sourceChunk","selector","useShallowEqual","prevSourceValue","currentResult","update","newSourceValue","newResult","middleware_exports","__export","logger","nonNegativeValidator","withHistory","withPersistence","next","maxHistory","history","currentIndex","isHistoryAction","historyChunk","removeCount","storage","serialize","deserialize","savedChunk","parsed","serialized"],"mappings":"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","file":"index.cjs","sourcesContent":["import { chunk, Chunk, Middleware, NamedMiddleware } from \"./core/core\";\n\nimport { AsyncChunk } from \"./core/asyncChunk\";\nimport { CombinedData, CombinedState } from \"./core/types\";\n\nexport function isValidChunkValue(value: unknown): boolean {\n return value !== null;\n}\n\nexport function isValidChunk<T>(value: unknown, validateBehavior = false): value is Chunk<T> {\n if (!isChunk(value)) {\n return false;\n }\n\n if (!validateBehavior) {\n return true;\n }\n\n try {\n const currentValue = value.get();\n value.set(currentValue);\n return true;\n } catch {\n return false;\n }\n}\n\nexport function isChunk<T>(value: unknown): value is Chunk<T> {\n if (!value || typeof value !== 'object') {\n return false;\n }\n\n const chunk = value as Record<string, unknown>;\n const requiredMethods = [\n 'get',\n 'set',\n 'update',\n 'subscribe',\n 'derive',\n 'reset',\n 'destroy'\n ] as const;\n\n return requiredMethods.every(method =>\n typeof chunk[method] === 'function'\n );\n}\n\nexport function once<T>(fn: () => T): () => T {\n let called = false;\n let result: T;\n return () => {\n if (!called) {\n result = fn();\n called = true;\n }\n return result;\n };\n};\n\nexport function combineAsyncChunks<T extends Record<string, AsyncChunk<any>>>(\n chunks: T\n): Chunk<CombinedState<T>> {\n const initialData = Object.keys(chunks).reduce((acc, key) => {\n acc[key as keyof T] = null;\n return acc;\n }, {} as CombinedData<T>);\n\n const initialState: CombinedState<T> = {\n loading: Object.keys(chunks).length > 0,\n error: null,\n errors: {},\n data: initialData\n };\n\n const combined = chunk(initialState);\n\n const chunkValues = Object.values(chunks);\n\n Object.entries(chunks).forEach(([key, asyncChunk]) => {\n asyncChunk.subscribe((state) => {\n const currentState = combined.get();\n\n let hasLoading = false;\n let firstError: Error | null = null;\n const allErrors: Partial<{ [K in keyof T]: Error }> = {};\n\n Object.entries(chunks).forEach(([chunkKey, chunk]) => {\n const chunkState = chunk.get();\n if (chunkState.loading) hasLoading = true;\n if (chunkState.error) {\n if (!firstError) firstError = chunkState.error;\n allErrors[chunkKey as keyof T] = chunkState.error;\n }\n });\n\n combined.set({\n loading: hasLoading,\n error: firstError,\n errors: allErrors,\n data: {\n ...currentState.data,\n [key]: state.data\n },\n });\n });\n });\n\n return combined;\n}\n\nexport function processMiddleware<T>(initialValue: T, middleware: (Middleware<T> | NamedMiddleware<T>)[]): T {\n if (initialValue === null) {\n throw new Error(\"Value cannot be null.\");\n }\n\n let currentValue = initialValue;\n let index = 0;\n\n while (index < middleware.length) {\n const current = middleware[index];\n\n const middlewareFn = typeof current === 'function' ? current : current.fn;\n const middlewareName = typeof current === 'function' ? `index ${index}` : (current.name || `index ${index}`);\n let nextCalled = false;\n let nextValue: T | null = null;\n\n try {\n middlewareFn(currentValue, (val) => {\n nextCalled = true;\n nextValue = val;\n });\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n throw new Error(`Middleware \"${middlewareName}\" threw an error: ${errorMessage}`);\n }\n\n if (!nextCalled) break;\n\n if (nextValue === null) {\n throw new Error(`Middleware at index ${index} returned null value.`);\n }\n\n currentValue = nextValue;\n index++;\n }\n\n return currentValue;\n}\n\nexport function shallowEqual<T>(a: T, b: T): boolean {\n if (a === b) {\n return true;\n }\n\n if (!a || !b || typeof a !== typeof b) {\n return false;\n }\n\n if (Array.isArray(a) && Array.isArray(b)) {\n if (a.length !== b.length) {\n return false;\n }\n for (let i = 0; i < a.length; i++) {\n if (a[i] !== b[i]) {\n return false;\n }\n }\n return true;\n }\n\n if (typeof a === 'object' && typeof b === 'object') {\n const keysA = Object.keys(a);\n const keysB = Object.keys(b);\n\n if (keysA.length !== keysB.length) {\n return false;\n }\n\n for (const key of keysA) {\n if (!Object.prototype.hasOwnProperty.call(b, key) || (a as any)[key] !== (b as any)[key]) {\n return false;\n }\n }\n return true;\n }\n\n // For primitive types, return false. Strict equality already handled by initial check\n return false;\n}\n\nexport function validateObjectShape<T>(original: T, updated: T, path = ''): void {\n if (typeof original === 'object' && original !== null && typeof updated === 'object' && updated !== null) {\n if (Array.isArray(original) && Array.isArray(updated)) {\n if (original.length > 0 && typeof original[0] === 'object') {\n for (let i = 0; i < updated.length; i++) {\n validateObjectShape(original[0], updated[i], `${path}[${i}]`);\n }\n }\n } else if (!Array.isArray(original) && !Array.isArray(updated)) {\n const originalKeys = Object.keys(original as object);\n const updatedKeys = Object.keys(updated as object);\n const extraKeys = updatedKeys.filter(key => !originalKeys.includes(key));\n\n if (extraKeys.length > 0) {\n const fullPath = path || 'root';\n console.error(`🚨 Stunk: Unknown properties detected at '${fullPath}': ${extraKeys.join(', ')}. This might cause bugs.`);\n console.error('Expected keys:', originalKeys);\n console.error('Received keys:', updatedKeys);\n }\n\n // Recurse into common keys\n for (const key of originalKeys) {\n validateObjectShape((original as any)[key], (updated as any)[key], path ? `${path}.${key}` : key);\n }\n }\n }\n}\n","import { processMiddleware, validateObjectShape } from \"../utils\";\n\nexport type Subscriber<T> = (newValue: T) => void;\nexport type Middleware<T> = (value: T, next: (newValue: T) => void) => void;\nexport type NamedMiddleware<T> = {\n name?: string;\n fn: Middleware<T>;\n};\n\nexport interface Chunk<T> {\n /** Get the current value of the chunk. */\n get: () => T;\n /** Set a new value for the chunk & Update existing value efficiently. */\n set: (newValueOrUpdater: T | ((currentValue: T) => T)) => void;\n /** Subscribe to changes in the chunk. Returns an unsubscribe function. */\n subscribe: (callback: Subscriber<T>) => () => void;\n /** Create a derived chunk based on this chunk's value. */\n derive: <D>(fn: (value: T) => D) => Chunk<D>;\n /** Reset the chunk to its initial value. */\n reset: () => void;\n /** Destroy the chunk and all its subscribers. */\n destroy: () => void;\n}\n\nlet isBatching = false;\nconst dirtyChunks = new Set<number>();\nconst chunkRegistry = new Map<number, { notify: () => void }>();\nlet chunkIdCounter = 0;\n\n/**\n * Batch multiple chunk updates into a single re-render.\n * Useful for updating multiple chunks at once without causing multiple re-renders.\n */\nexport function batch(callback: () => void) {\n const wasBatchingBefore = isBatching;\n isBatching = true;\n try {\n callback();\n } finally {\n if (!wasBatchingBefore) {\n isBatching = false;\n const chunks = Array.from(dirtyChunks); // Snapshot to avoid mutation issues\n dirtyChunks.clear(); // Clear early to prevent re-adds\n chunks.forEach(id => {\n const chunk = chunkRegistry.get(id);\n if (chunk) chunk.notify();\n });\n }\n }\n}\n\nexport function chunk<T>(initialValue: T, middleware: (Middleware<T> | NamedMiddleware<T>)[] = []): Chunk<T> {\n if (initialValue === null) {\n throw new Error(\"Initial value cannot be null.\");\n }\n\n let value = initialValue;\n const subscribers = new Set<Subscriber<T>>();\n const chunkId = chunkIdCounter++;\n\n const notify = () => {\n subscribers.forEach(subscriber => subscriber(value));\n };\n\n chunkRegistry.set(chunkId, { notify });\n\n const notifySubscribers = () => {\n if (subscribers.size === 0) return; // Skip if no subscribers\n if (isBatching) {\n dirtyChunks.add(chunkId);\n } else {\n notify();\n }\n };\n\n const get = () => value;\n\n const set = (newValueOrUpdater: T | ((currentValue: T) => T)) => {\n let newValue: T;\n\n if (typeof newValueOrUpdater === 'function') {\n const updaterFn = newValueOrUpdater as (currentValue: T) => T;\n newValue = updaterFn(value);\n } else {\n newValue = newValueOrUpdater;\n }\n\n validateObjectShape(value, newValue);\n\n const processedValue = processMiddleware(newValue, middleware);\n\n if (processedValue !== value) {\n value = processedValue as T & {};\n notifySubscribers();\n }\n };\n\n const subscribe = (callback: Subscriber<T>) => {\n if (typeof callback !== \"function\") {\n throw new Error(\"Callback must be a function.\");\n }\n subscribers.add(callback);\n callback(value);\n return () => subscribers.delete(callback);\n };\n\n const reset = () => {\n value = initialValue;\n notifySubscribers();\n };\n\n const destroy = () => {\n subscribers.clear();\n value = initialValue;\n dirtyChunks.delete(chunkId);\n chunkRegistry.delete(chunkId);\n };\n\n const derive = <D>(fn: (value: T) => D) => {\n if (typeof fn !== \"function\") {\n throw new Error(\"Derive function must be a function.\");\n }\n\n const initialDerivedValue = fn(value);\n const derivedChunk = chunk(initialDerivedValue);\n\n const unsubscribe = subscribe(() => {\n const newDerivedValue = fn(value);\n derivedChunk.set(newDerivedValue);\n });\n\n // Add a cleanup method to the derived chunk\n const originalDestroy = derivedChunk.destroy;\n derivedChunk.destroy = () => {\n unsubscribe();\n originalDestroy();\n };\n\n return derivedChunk;\n };\n\n return { get, set, subscribe, derive, reset, destroy };\n}\n","import { chunk, Chunk } from \"./core\";\nimport { AsyncChunkOpt } from \"./types\";\n\nexport interface AsyncState<T, E extends Error> {\n loading: boolean;\n error: E | null;\n data: T | null;\n lastFetched?: number;\n}\n\nexport interface RefreshConfig {\n /** Time in milliseconds after which data becomes stale */\n staleTime?: number;\n /** Time in milliseconds to cache data before considering it expired */\n cacheTime?: number;\n /** Auto-refresh interval in milliseconds */\n refetchInterval?: number;\n}\n\n// Extend existing options without breaking changes\nexport interface AsyncChunkOptExtended<T, E extends Error> extends AsyncChunkOpt<T, E> {\n refresh?: RefreshConfig;\n /** Enable/disable the fetcher - useful for conditional fetching */\n enabled?: boolean;\n}\n\nexport interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncState<T, E>> {\n /** Reload the data from the source. */\n reload: () => Promise<void>;\n\n /** Smart refresh - respects stale time */\n refresh: () => Promise<void>;\n\n /** Mutate the data directly. */\n mutate: (mutator: (currentData: T | null) => T) => void;\n\n /** Reset the state to the initial value. */\n reset: () => void;\n\n /** Clean up intervals and timeouts */\n cleanup: () => void;\n}\n\n// Overloaded function signatures for backward compatibility\nexport function asyncChunk<T, E extends Error = Error>(\n fetcher: () => Promise<T>,\n options?: AsyncChunkOptExtended<T, E>\n): AsyncChunk<T, E>;\n\nexport function asyncChunk<T, E extends Error = Error, P extends any[] = []>(\n fetcher: (...params: P) => Promise<T>,\n options?: AsyncChunkOptExtended<T, E>\n): AsyncChunk<T, E> & {\n /** Reload with new parameters */\n reload: (...params: P) => Promise<void>;\n /** Smart refresh with parameters */\n refresh: (...params: P) => Promise<void>;\n /** Set parameters for future calls */\n setParams: (...params: P) => void;\n};\n\nexport function asyncChunk<T, E extends Error = Error, P extends any[] = []>(\n fetcher: (...params: P) => Promise<T>,\n options: AsyncChunkOptExtended<T, E> = {}\n) {\n const {\n initialData = null,\n onError,\n retryCount = 0,\n retryDelay = 1000,\n refresh: refreshConfig = {},\n enabled = true,\n } = options;\n\n const { staleTime = 0, cacheTime = 5 * 60 * 1000, refetchInterval } = refreshConfig;\n\n // let's expect params\n const expectsParams = fetcher.length > 0;\n\n const initialState: AsyncState<T, E> = {\n loading: enabled && !expectsParams,\n error: null,\n data: initialData,\n lastFetched: undefined,\n };\n\n const hasValidParams = (params?: P): boolean => {\n if (!expectsParams) return true;\n\n if (params === undefined) return false;\n if (Array.isArray(params)) {\n return params.every(p => p !== undefined && p !== null);\n }\n return true;\n };\n\n const baseChunk = chunk(initialState);\n let currentParams: P | undefined;\n let intervalId: number | null = null;\n let cacheTimeoutId: number | null = null;\n\n const isStale = () => {\n const state = baseChunk.get();\n if (!state.lastFetched) return true;\n return Date.now() - state.lastFetched > staleTime;\n };\n\n const isCacheExpired = () => {\n const state = baseChunk.get();\n if (!state.lastFetched) return true;\n return Date.now() - state.lastFetched > cacheTime;\n };\n\n const clearCache = () => {\n baseChunk.set({\n ...baseChunk.get(),\n data: initialData,\n lastFetched: undefined,\n });\n };\n\n const setCacheTimeout = () => {\n if (cacheTimeoutId) {\n clearTimeout(cacheTimeoutId);\n }\n if (cacheTime > 0) {\n cacheTimeoutId = setTimeout(clearCache, cacheTime);\n }\n };\n\n const fetchData = async (params?: P, retries = retryCount, force = false): Promise<void> => {\n if (!enabled) return;\n\n // Store params for reuse\n if (params !== undefined) {\n currentParams = params;\n }\n\n // Don't fetch if we don't have valid parameters (only matters if params expected)\n if (!hasValidParams(currentParams)) {\n baseChunk.set({ ...baseChunk.get(), loading: false });\n return;\n }\n\n // Don't fetch if data is fresh and not forcing\n if (!force && !isStale() && baseChunk.get().data !== null && staleTime > 0) {\n return;\n }\n\n baseChunk.set({ ...baseChunk.get(), loading: true, error: null });\n\n try {\n const data = expectsParams\n ? await fetcher(...currentParams!)\n : await fetcher(...([] as unknown as P));\n const now = Date.now();\n\n baseChunk.set({\n loading: false,\n error: null,\n data,\n lastFetched: now\n });\n\n setCacheTimeout();\n } catch (error) {\n if (retries > 0) {\n await new Promise(resolve => setTimeout(resolve, retryDelay));\n return fetchData(params, retries - 1, force);\n }\n\n baseChunk.set({\n loading: false,\n error: error as E,\n data: baseChunk.get().data,\n lastFetched: baseChunk.get().lastFetched\n });\n\n if (onError) {\n onError(error as E);\n }\n }\n };\n\n // Setup auto-refresh\n if (refetchInterval && refetchInterval > 0 && enabled) {\n intervalId = setInterval(() => {\n if (enabled) {\n fetchData(currentParams, 0, false);\n }\n }, refetchInterval);\n }\n\n // Initial fetch\n if (enabled && !expectsParams) {\n fetchData();\n }\n\n const cleanup = () => {\n if (intervalId) {\n clearInterval(intervalId);\n intervalId = null;\n }\n if (cacheTimeoutId) {\n clearTimeout(cacheTimeoutId);\n cacheTimeoutId = null;\n }\n };\n\n const asyncChunkInstance = {\n ...baseChunk,\n\n reload: async (...params: P) => {\n await fetchData(params.length > 0 ? params : undefined, retryCount, true);\n },\n\n refresh: async (...params: P) => {\n await fetchData(params.length > 0 ? params : undefined, retryCount, false);\n },\n\n mutate: (mutator: (currentData: T | null) => T) => {\n const currentState = baseChunk.get();\n const newData = mutator(currentState.data);\n baseChunk.set({ ...currentState, data: newData });\n },\n\n reset: () => {\n cleanup();\n baseChunk.set({\n ...initialState,\n loading: enabled\n });\n if (enabled) {\n fetchData();\n\n if (refetchInterval && refetchInterval > 0) {\n intervalId = setInterval(() => {\n if (enabled) {\n fetchData(currentParams, 0, false);\n }\n }, refetchInterval);\n }\n }\n },\n\n cleanup,\n\n // Only add setParams if parameters were used\n setParams: (...params: P) => {\n currentParams = params;\n if (enabled && hasValidParams(params)) {\n fetchData(params);\n }\n },\n\n };\n\n return asyncChunkInstance;\n}\n","import { Chunk, chunk } from \"./core\";\n\nimport { shallowEqual } from \"../utils\";\n\n// Helper type to extract the value type from a Chunk\nexport type ChunkValue<T> = T extends Chunk<infer U> ? U : never;\n\n// Helper type to transform an array of Chunks into an array of their value types\nexport type DependencyValues<T extends Chunk<any>[]> = {\n [K in keyof T]: T[K] extends Chunk<any> ? ChunkValue<T[K]> : never;\n};\n\nexport interface Computed<T> extends Chunk<T> {\n /**\n * Checks if the computed value needs to be recalculated due to dependency changes.\n * @returns True if the computed value is dirty, false otherwise.\n */\n isDirty: () => boolean;\n /** Manually forces recalculation of the computed value from its dependencies. */\n recompute: () => void;\n}\n\nexport function computed<TDeps extends Chunk<any>[], TResult>(\n dependencies: [...TDeps],\n computeFn: (...args: DependencyValues<TDeps>) => TResult\n): Computed<TResult> {\n const dependencyValues = dependencies.map(dep => dep.get());\n let cachedValue = computeFn(...dependencyValues as DependencyValues<TDeps>);\n\n const computedChunk = chunk(cachedValue);\n const originalSet = computedChunk.set;\n\n let isDirty = false;\n\n // Direct synchronous recomputation\n const recompute = () => {\n let hasChanges = false;\n\n for (let i = 0; i < dependencies.length; i++) {\n const newValue = dependencies[i].get();\n if (newValue !== dependencyValues[i]) {\n dependencyValues[i] = newValue;\n hasChanges = true;\n }\n }\n\n if (hasChanges) {\n const newValue = computeFn(...dependencyValues as DependencyValues<TDeps>);\n // Fast path for primitives only. Avoids shallowEqual for performance.\n if (newValue !== cachedValue) {\n // Only use shallowEqual for objects when needed\n if (typeof newValue !== 'object' || typeof cachedValue !== 'object' || !shallowEqual(newValue, cachedValue)) {\n cachedValue = newValue;\n originalSet(newValue);\n }\n }\n isDirty = false;\n }\n };\n\n const unsubs = dependencies.map(dep =>\n dep.subscribe(() => {\n isDirty = true;\n recompute();\n })\n );\n\n return {\n ...computedChunk,\n get: () => {\n if (isDirty) recompute();\n return cachedValue;\n },\n recompute,\n isDirty: () => isDirty,\n set: () => { throw new Error('Cannot set values directly on computed. Modify the source chunk instead.'); },\n reset: () => {\n dependencies.forEach(dep => {\n if (typeof dep.reset === 'function') {\n dep.reset();\n }\n });\n isDirty = true;\n recompute();\n return cachedValue;\n },\n destroy: () => { unsubs.forEach(unsub => unsub()); computedChunk.destroy?.(); }\n };\n}\n","import { shallowEqual } from \"../utils\";\nimport { Chunk, chunk } from \"./core\";\n\nexport interface SelectOptions {\n /**\n * Configuration options for selector functions.\n * @property {boolean} [useShallowEqual] - When true, performs a shallow equality check\n * on the derived selector results to prevent unnecessary updates.\n */\n useShallowEqual?: boolean;\n}\n\n/**\n * Creates a derived read-only chunk based on a selector function.\n * @param sourceChunk The source chunk to derive from.\n * @param selector A function that extracts part of the source value.\n * @param options Optional settings for shallow equality comparison.\n * @returns A read-only derived chunk.\n */\n\nexport function select<T, S>(\n sourceChunk: Chunk<T>,\n selector: (value: T) => S,\n options: SelectOptions = {}\n): Chunk<S> {\n const { useShallowEqual = false } = options;\n\n let prevSourceValue = sourceChunk.get();\n let currentResult = selector(prevSourceValue);\n\n const derivedChunk = chunk(currentResult);\n\n const update = () => {\n const newSourceValue = sourceChunk.get();\n const newResult = selector(newSourceValue);\n\n // Always update the reference to source value\n prevSourceValue = newSourceValue;\n\n // Check if the result has changed\n const resultChanged = useShallowEqual\n ? !shallowEqual(newResult, currentResult)\n : newResult !== currentResult;\n\n if (resultChanged) {\n currentResult = newResult;\n derivedChunk.set(newResult);\n }\n };\n\n const unsubscribe = sourceChunk.subscribe(update);\n\n return {\n get: () => derivedChunk.get(),\n set: () => {\n throw new Error('Cannot set values directly on a selector. Modify the source chunk instead.');\n },\n subscribe: derivedChunk.subscribe,\n derive: <D>(fn: (value: S) => D) => select(derivedChunk, fn, options), // Pass options to nested selectors\n reset: () => {\n throw new Error('Cannot reset a selector chunk. Reset the source chunk instead.');\n },\n destroy: () => {\n unsubscribe();\n derivedChunk.destroy();\n }\n };\n}\n","// Middleware passed to chunks\nexport { logger } from \"./logger\";\nexport { nonNegativeValidator } from \"./validator\";\n\n// Middleware used with chunks\nexport { withHistory } from \"./history\";\nexport { withPersistence } from './persistence';\n","import { Middleware } from \"../core/core\";\n\nexport const logger: Middleware<any> = (value, next) => {\n console.log(\"Setting value:\", value);\n next(value);\n};\n","import { chunk, Middleware } from \"../core/core\";\n\nexport const nonNegativeValidator: Middleware<number> = (value, next) => {\n if (value < 0) {\n throw new Error(\"Value must be non-negative!\");\n }\n next(value); // If validation passes, proceed with the update\n};\n","import { Chunk } from \"../core/core\";\n\nexport interface ChunkWithHistory<T> extends Chunk<T> {\n /**\n * Reverts to the previous state (if available).\n */\n undo: () => void;\n /**\n * Moves to the next state (if available).\n */\n redo: () => void;\n /**\n * Returns true if there is a previous state to revert to.\n */\n canUndo: () => boolean;\n /**\n * Returns true if there is a next state to move to.\n */\n canRedo: () => boolean;\n /**\n * Returns an array of all the values in the history.\n */\n getHistory: () => T[];\n /**\n * Clears the history, keeping only the current value.\n */\n clearHistory: () => void;\n}\n\nexport function withHistory<T>(\n baseChunk: Chunk<T>,\n options: { maxHistory?: number } = {}\n): ChunkWithHistory<T> {\n const { maxHistory = 100 } = options;\n const history: T[] = [baseChunk.get()];\n let currentIndex = 0;\n let isHistoryAction = false;\n\n const historyChunk: ChunkWithHistory<T> = {\n ...baseChunk,\n\n set: (newValueOrUpdater: T | ((currentValue: T) => T)) => {\n if (isHistoryAction) {\n baseChunk.set(newValueOrUpdater);\n return;\n }\n\n // Process the value or updater function\n let newValue: T;\n if (typeof newValueOrUpdater === 'function') {\n // Get current value and apply the updater function\n const currentValue = baseChunk.get();\n newValue = (newValueOrUpdater as ((currentValue: T) => T))(currentValue);\n } else {\n // Use directly as the new value\n newValue = newValueOrUpdater;\n }\n history.splice(currentIndex + 1);\n history.push(newValue);\n\n // Limit history size\n if (history.length > maxHistory) {\n console.warn(\"History limit reached. Removing oldest entries.\");\n const removeCount = history.length - maxHistory;\n history.splice(0, removeCount);\n currentIndex = Math.max(0, currentIndex - removeCount);\n }\n\n currentIndex = history.length - 1;\n baseChunk.set(newValue);\n },\n\n undo: () => {\n if (!historyChunk.canUndo()) return;\n\n isHistoryAction = true;\n currentIndex--;\n historyChunk.set(history[currentIndex]);\n isHistoryAction = false;\n },\n\n redo: () => {\n if (!historyChunk.canRedo()) return;\n\n isHistoryAction = true;\n currentIndex++;\n historyChunk.set(history[currentIndex]);\n isHistoryAction = false;\n },\n\n canUndo: () => currentIndex > 0,\n\n canRedo: () => currentIndex < history.length - 1,\n\n getHistory: () => [...history],\n\n clearHistory: () => {\n const currentValue = baseChunk.get();\n history.length = 0;\n history.push(currentValue);\n currentIndex = 0;\n },\n\n // Override destroy to clean up history\n destroy: () => {\n history.length = 0;\n baseChunk.destroy();\n }\n }\n\n return historyChunk;\n\n}\n","import { Chunk } from \"../core/core\";\n\nexport interface PersistOptions<T> {\n key: string;\n storage?: Storage;\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n}\n\nexport function withPersistence<T>(\n baseChunk: Chunk<T>,\n options: PersistOptions<T>\n): Chunk<T> {\n const {\n key,\n storage = localStorage,\n serialize = JSON.stringify,\n deserialize = JSON.parse,\n } = options;\n\n // Try to load initial state from storage\n try {\n const savedChunk = storage.getItem(key);\n if (savedChunk) {\n const parsed = deserialize(savedChunk);\n baseChunk.set(parsed)\n }\n } catch (error) {\n console.error('Failed to load persisted state:', error);\n }\n\n // Save to storage\n baseChunk.subscribe((newValue) => {\n try {\n const serialized = serialize(newValue);\n storage.setItem(key, serialized);\n\n } catch (error) {\n console.log('Failed to persist chunk', error)\n }\n })\n\n return baseChunk\n\n}\n"]}
package/dist/index.js.map DELETED
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/utils.ts","../src/core/core.ts","../src/core/asyncChunk.ts","../src/core/computed.ts","../src/core/selector.ts","../src/middleware/index.ts","../src/middleware/logger.ts","../src/middleware/validator.ts","../src/middleware/history.ts","../src/middleware/persistence.ts"],"names":["isValidChunkValue","value","isChunk","chunk","method","once","fn","called","result","combineAsyncChunks","chunks","initialData","acc","key","initialState","combined","asyncChunk","state","currentState","hasLoading","firstError","allErrors","chunkKey","chunkState","processMiddleware","initialValue","middleware","currentValue","index","current","middlewareFn","middlewareName","nextCalled","nextValue","val","error","errorMessage","shallowEqual","a","b","i","keysA","keysB","validateObjectShape","original","updated","path","originalKeys","updatedKeys","extraKeys","isBatching","dirtyChunks","chunkRegistry","chunkIdCounter","batch","callback","wasBatchingBefore","id","subscribers","chunkId","notify","subscriber","notifySubscribers","get","set","newValueOrUpdater","newValue","processedValue","subscribe","initialDerivedValue","derivedChunk","unsubscribe","newDerivedValue","originalDestroy","fetcher","options","onError","retryCount","retryDelay","refreshConfig","enabled","staleTime","cacheTime","refetchInterval","expectsParams","hasValidParams","params","p","baseChunk","currentParams","intervalId","cacheTimeoutId","isStale","clearCache","setCacheTimeout","fetchData","retries","force","data","now","resolve","cleanup","mutator","newData","computed","dependencies","computeFn","dependencyValues","dep","cachedValue","computedChunk","originalSet","isDirty","recompute","hasChanges","unsubs","unsub","select","sourceChunk","selector","useShallowEqual","prevSourceValue","currentResult","update","newSourceValue","newResult","middleware_exports","__export","logger","nonNegativeValidator","withHistory","withPersistence","next","maxHistory","history","currentIndex","isHistoryAction","historyChunk","removeCount","storage","serialize","deserialize","savedChunk","parsed","serialized"],"mappings":"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","file":"index.js","sourcesContent":["import { chunk, Chunk, Middleware, NamedMiddleware } from \"./core/core\";\n\nimport { AsyncChunk } from \"./core/asyncChunk\";\nimport { CombinedData, CombinedState } from \"./core/types\";\n\nexport function isValidChunkValue(value: unknown): boolean {\n return value !== null;\n}\n\nexport function isValidChunk<T>(value: unknown, validateBehavior = false): value is Chunk<T> {\n if (!isChunk(value)) {\n return false;\n }\n\n if (!validateBehavior) {\n return true;\n }\n\n try {\n const currentValue = value.get();\n value.set(currentValue);\n return true;\n } catch {\n return false;\n }\n}\n\nexport function isChunk<T>(value: unknown): value is Chunk<T> {\n if (!value || typeof value !== 'object') {\n return false;\n }\n\n const chunk = value as Record<string, unknown>;\n const requiredMethods = [\n 'get',\n 'set',\n 'update',\n 'subscribe',\n 'derive',\n 'reset',\n 'destroy'\n ] as const;\n\n return requiredMethods.every(method =>\n typeof chunk[method] === 'function'\n );\n}\n\nexport function once<T>(fn: () => T): () => T {\n let called = false;\n let result: T;\n return () => {\n if (!called) {\n result = fn();\n called = true;\n }\n return result;\n };\n};\n\nexport function combineAsyncChunks<T extends Record<string, AsyncChunk<any>>>(\n chunks: T\n): Chunk<CombinedState<T>> {\n const initialData = Object.keys(chunks).reduce((acc, key) => {\n acc[key as keyof T] = null;\n return acc;\n }, {} as CombinedData<T>);\n\n const initialState: CombinedState<T> = {\n loading: Object.keys(chunks).length > 0,\n error: null,\n errors: {},\n data: initialData\n };\n\n const combined = chunk(initialState);\n\n const chunkValues = Object.values(chunks);\n\n Object.entries(chunks).forEach(([key, asyncChunk]) => {\n asyncChunk.subscribe((state) => {\n const currentState = combined.get();\n\n let hasLoading = false;\n let firstError: Error | null = null;\n const allErrors: Partial<{ [K in keyof T]: Error }> = {};\n\n Object.entries(chunks).forEach(([chunkKey, chunk]) => {\n const chunkState = chunk.get();\n if (chunkState.loading) hasLoading = true;\n if (chunkState.error) {\n if (!firstError) firstError = chunkState.error;\n allErrors[chunkKey as keyof T] = chunkState.error;\n }\n });\n\n combined.set({\n loading: hasLoading,\n error: firstError,\n errors: allErrors,\n data: {\n ...currentState.data,\n [key]: state.data\n },\n });\n });\n });\n\n return combined;\n}\n\nexport function processMiddleware<T>(initialValue: T, middleware: (Middleware<T> | NamedMiddleware<T>)[]): T {\n if (initialValue === null) {\n throw new Error(\"Value cannot be null.\");\n }\n\n let currentValue = initialValue;\n let index = 0;\n\n while (index < middleware.length) {\n const current = middleware[index];\n\n const middlewareFn = typeof current === 'function' ? current : current.fn;\n const middlewareName = typeof current === 'function' ? `index ${index}` : (current.name || `index ${index}`);\n let nextCalled = false;\n let nextValue: T | null = null;\n\n try {\n middlewareFn(currentValue, (val) => {\n nextCalled = true;\n nextValue = val;\n });\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n throw new Error(`Middleware \"${middlewareName}\" threw an error: ${errorMessage}`);\n }\n\n if (!nextCalled) break;\n\n if (nextValue === null) {\n throw new Error(`Middleware at index ${index} returned null value.`);\n }\n\n currentValue = nextValue;\n index++;\n }\n\n return currentValue;\n}\n\nexport function shallowEqual<T>(a: T, b: T): boolean {\n if (a === b) {\n return true;\n }\n\n if (!a || !b || typeof a !== typeof b) {\n return false;\n }\n\n if (Array.isArray(a) && Array.isArray(b)) {\n if (a.length !== b.length) {\n return false;\n }\n for (let i = 0; i < a.length; i++) {\n if (a[i] !== b[i]) {\n return false;\n }\n }\n return true;\n }\n\n if (typeof a === 'object' && typeof b === 'object') {\n const keysA = Object.keys(a);\n const keysB = Object.keys(b);\n\n if (keysA.length !== keysB.length) {\n return false;\n }\n\n for (const key of keysA) {\n if (!Object.prototype.hasOwnProperty.call(b, key) || (a as any)[key] !== (b as any)[key]) {\n return false;\n }\n }\n return true;\n }\n\n // For primitive types, return false. Strict equality already handled by initial check\n return false;\n}\n\nexport function validateObjectShape<T>(original: T, updated: T, path = ''): void {\n if (typeof original === 'object' && original !== null && typeof updated === 'object' && updated !== null) {\n if (Array.isArray(original) && Array.isArray(updated)) {\n if (original.length > 0 && typeof original[0] === 'object') {\n for (let i = 0; i < updated.length; i++) {\n validateObjectShape(original[0], updated[i], `${path}[${i}]`);\n }\n }\n } else if (!Array.isArray(original) && !Array.isArray(updated)) {\n const originalKeys = Object.keys(original as object);\n const updatedKeys = Object.keys(updated as object);\n const extraKeys = updatedKeys.filter(key => !originalKeys.includes(key));\n\n if (extraKeys.length > 0) {\n const fullPath = path || 'root';\n console.error(`🚨 Stunk: Unknown properties detected at '${fullPath}': ${extraKeys.join(', ')}. This might cause bugs.`);\n console.error('Expected keys:', originalKeys);\n console.error('Received keys:', updatedKeys);\n }\n\n // Recurse into common keys\n for (const key of originalKeys) {\n validateObjectShape((original as any)[key], (updated as any)[key], path ? `${path}.${key}` : key);\n }\n }\n }\n}\n","import { processMiddleware, validateObjectShape } from \"../utils\";\n\nexport type Subscriber<T> = (newValue: T) => void;\nexport type Middleware<T> = (value: T, next: (newValue: T) => void) => void;\nexport type NamedMiddleware<T> = {\n name?: string;\n fn: Middleware<T>;\n};\n\nexport interface Chunk<T> {\n /** Get the current value of the chunk. */\n get: () => T;\n /** Set a new value for the chunk & Update existing value efficiently. */\n set: (newValueOrUpdater: T | ((currentValue: T) => T)) => void;\n /** Subscribe to changes in the chunk. Returns an unsubscribe function. */\n subscribe: (callback: Subscriber<T>) => () => void;\n /** Create a derived chunk based on this chunk's value. */\n derive: <D>(fn: (value: T) => D) => Chunk<D>;\n /** Reset the chunk to its initial value. */\n reset: () => void;\n /** Destroy the chunk and all its subscribers. */\n destroy: () => void;\n}\n\nlet isBatching = false;\nconst dirtyChunks = new Set<number>();\nconst chunkRegistry = new Map<number, { notify: () => void }>();\nlet chunkIdCounter = 0;\n\n/**\n * Batch multiple chunk updates into a single re-render.\n * Useful for updating multiple chunks at once without causing multiple re-renders.\n */\nexport function batch(callback: () => void) {\n const wasBatchingBefore = isBatching;\n isBatching = true;\n try {\n callback();\n } finally {\n if (!wasBatchingBefore) {\n isBatching = false;\n const chunks = Array.from(dirtyChunks); // Snapshot to avoid mutation issues\n dirtyChunks.clear(); // Clear early to prevent re-adds\n chunks.forEach(id => {\n const chunk = chunkRegistry.get(id);\n if (chunk) chunk.notify();\n });\n }\n }\n}\n\nexport function chunk<T>(initialValue: T, middleware: (Middleware<T> | NamedMiddleware<T>)[] = []): Chunk<T> {\n if (initialValue === null) {\n throw new Error(\"Initial value cannot be null.\");\n }\n\n let value = initialValue;\n const subscribers = new Set<Subscriber<T>>();\n const chunkId = chunkIdCounter++;\n\n const notify = () => {\n subscribers.forEach(subscriber => subscriber(value));\n };\n\n chunkRegistry.set(chunkId, { notify });\n\n const notifySubscribers = () => {\n if (subscribers.size === 0) return; // Skip if no subscribers\n if (isBatching) {\n dirtyChunks.add(chunkId);\n } else {\n notify();\n }\n };\n\n const get = () => value;\n\n const set = (newValueOrUpdater: T | ((currentValue: T) => T)) => {\n let newValue: T;\n\n if (typeof newValueOrUpdater === 'function') {\n const updaterFn = newValueOrUpdater as (currentValue: T) => T;\n newValue = updaterFn(value);\n } else {\n newValue = newValueOrUpdater;\n }\n\n validateObjectShape(value, newValue);\n\n const processedValue = processMiddleware(newValue, middleware);\n\n if (processedValue !== value) {\n value = processedValue as T & {};\n notifySubscribers();\n }\n };\n\n const subscribe = (callback: Subscriber<T>) => {\n if (typeof callback !== \"function\") {\n throw new Error(\"Callback must be a function.\");\n }\n subscribers.add(callback);\n callback(value);\n return () => subscribers.delete(callback);\n };\n\n const reset = () => {\n value = initialValue;\n notifySubscribers();\n };\n\n const destroy = () => {\n subscribers.clear();\n value = initialValue;\n dirtyChunks.delete(chunkId);\n chunkRegistry.delete(chunkId);\n };\n\n const derive = <D>(fn: (value: T) => D) => {\n if (typeof fn !== \"function\") {\n throw new Error(\"Derive function must be a function.\");\n }\n\n const initialDerivedValue = fn(value);\n const derivedChunk = chunk(initialDerivedValue);\n\n const unsubscribe = subscribe(() => {\n const newDerivedValue = fn(value);\n derivedChunk.set(newDerivedValue);\n });\n\n // Add a cleanup method to the derived chunk\n const originalDestroy = derivedChunk.destroy;\n derivedChunk.destroy = () => {\n unsubscribe();\n originalDestroy();\n };\n\n return derivedChunk;\n };\n\n return { get, set, subscribe, derive, reset, destroy };\n}\n","import { chunk, Chunk } from \"./core\";\nimport { AsyncChunkOpt } from \"./types\";\n\nexport interface AsyncState<T, E extends Error> {\n loading: boolean;\n error: E | null;\n data: T | null;\n lastFetched?: number;\n}\n\nexport interface RefreshConfig {\n /** Time in milliseconds after which data becomes stale */\n staleTime?: number;\n /** Time in milliseconds to cache data before considering it expired */\n cacheTime?: number;\n /** Auto-refresh interval in milliseconds */\n refetchInterval?: number;\n}\n\n// Extend existing options without breaking changes\nexport interface AsyncChunkOptExtended<T, E extends Error> extends AsyncChunkOpt<T, E> {\n refresh?: RefreshConfig;\n /** Enable/disable the fetcher - useful for conditional fetching */\n enabled?: boolean;\n}\n\nexport interface AsyncChunk<T, E extends Error = Error> extends Chunk<AsyncState<T, E>> {\n /** Reload the data from the source. */\n reload: () => Promise<void>;\n\n /** Smart refresh - respects stale time */\n refresh: () => Promise<void>;\n\n /** Mutate the data directly. */\n mutate: (mutator: (currentData: T | null) => T) => void;\n\n /** Reset the state to the initial value. */\n reset: () => void;\n\n /** Clean up intervals and timeouts */\n cleanup: () => void;\n}\n\n// Overloaded function signatures for backward compatibility\nexport function asyncChunk<T, E extends Error = Error>(\n fetcher: () => Promise<T>,\n options?: AsyncChunkOptExtended<T, E>\n): AsyncChunk<T, E>;\n\nexport function asyncChunk<T, E extends Error = Error, P extends any[] = []>(\n fetcher: (...params: P) => Promise<T>,\n options?: AsyncChunkOptExtended<T, E>\n): AsyncChunk<T, E> & {\n /** Reload with new parameters */\n reload: (...params: P) => Promise<void>;\n /** Smart refresh with parameters */\n refresh: (...params: P) => Promise<void>;\n /** Set parameters for future calls */\n setParams: (...params: P) => void;\n};\n\nexport function asyncChunk<T, E extends Error = Error, P extends any[] = []>(\n fetcher: (...params: P) => Promise<T>,\n options: AsyncChunkOptExtended<T, E> = {}\n) {\n const {\n initialData = null,\n onError,\n retryCount = 0,\n retryDelay = 1000,\n refresh: refreshConfig = {},\n enabled = true,\n } = options;\n\n const { staleTime = 0, cacheTime = 5 * 60 * 1000, refetchInterval } = refreshConfig;\n\n // let's expect params\n const expectsParams = fetcher.length > 0;\n\n const initialState: AsyncState<T, E> = {\n loading: enabled && !expectsParams,\n error: null,\n data: initialData,\n lastFetched: undefined,\n };\n\n const hasValidParams = (params?: P): boolean => {\n if (!expectsParams) return true;\n\n if (params === undefined) return false;\n if (Array.isArray(params)) {\n return params.every(p => p !== undefined && p !== null);\n }\n return true;\n };\n\n const baseChunk = chunk(initialState);\n let currentParams: P | undefined;\n let intervalId: number | null = null;\n let cacheTimeoutId: number | null = null;\n\n const isStale = () => {\n const state = baseChunk.get();\n if (!state.lastFetched) return true;\n return Date.now() - state.lastFetched > staleTime;\n };\n\n const isCacheExpired = () => {\n const state = baseChunk.get();\n if (!state.lastFetched) return true;\n return Date.now() - state.lastFetched > cacheTime;\n };\n\n const clearCache = () => {\n baseChunk.set({\n ...baseChunk.get(),\n data: initialData,\n lastFetched: undefined,\n });\n };\n\n const setCacheTimeout = () => {\n if (cacheTimeoutId) {\n clearTimeout(cacheTimeoutId);\n }\n if (cacheTime > 0) {\n cacheTimeoutId = setTimeout(clearCache, cacheTime);\n }\n };\n\n const fetchData = async (params?: P, retries = retryCount, force = false): Promise<void> => {\n if (!enabled) return;\n\n // Store params for reuse\n if (params !== undefined) {\n currentParams = params;\n }\n\n // Don't fetch if we don't have valid parameters (only matters if params expected)\n if (!hasValidParams(currentParams)) {\n baseChunk.set({ ...baseChunk.get(), loading: false });\n return;\n }\n\n // Don't fetch if data is fresh and not forcing\n if (!force && !isStale() && baseChunk.get().data !== null && staleTime > 0) {\n return;\n }\n\n baseChunk.set({ ...baseChunk.get(), loading: true, error: null });\n\n try {\n const data = expectsParams\n ? await fetcher(...currentParams!)\n : await fetcher(...([] as unknown as P));\n const now = Date.now();\n\n baseChunk.set({\n loading: false,\n error: null,\n data,\n lastFetched: now\n });\n\n setCacheTimeout();\n } catch (error) {\n if (retries > 0) {\n await new Promise(resolve => setTimeout(resolve, retryDelay));\n return fetchData(params, retries - 1, force);\n }\n\n baseChunk.set({\n loading: false,\n error: error as E,\n data: baseChunk.get().data,\n lastFetched: baseChunk.get().lastFetched\n });\n\n if (onError) {\n onError(error as E);\n }\n }\n };\n\n // Setup auto-refresh\n if (refetchInterval && refetchInterval > 0 && enabled) {\n intervalId = setInterval(() => {\n if (enabled) {\n fetchData(currentParams, 0, false);\n }\n }, refetchInterval);\n }\n\n // Initial fetch\n if (enabled && !expectsParams) {\n fetchData();\n }\n\n const cleanup = () => {\n if (intervalId) {\n clearInterval(intervalId);\n intervalId = null;\n }\n if (cacheTimeoutId) {\n clearTimeout(cacheTimeoutId);\n cacheTimeoutId = null;\n }\n };\n\n const asyncChunkInstance = {\n ...baseChunk,\n\n reload: async (...params: P) => {\n await fetchData(params.length > 0 ? params : undefined, retryCount, true);\n },\n\n refresh: async (...params: P) => {\n await fetchData(params.length > 0 ? params : undefined, retryCount, false);\n },\n\n mutate: (mutator: (currentData: T | null) => T) => {\n const currentState = baseChunk.get();\n const newData = mutator(currentState.data);\n baseChunk.set({ ...currentState, data: newData });\n },\n\n reset: () => {\n cleanup();\n baseChunk.set({\n ...initialState,\n loading: enabled\n });\n if (enabled) {\n fetchData();\n\n if (refetchInterval && refetchInterval > 0) {\n intervalId = setInterval(() => {\n if (enabled) {\n fetchData(currentParams, 0, false);\n }\n }, refetchInterval);\n }\n }\n },\n\n cleanup,\n\n // Only add setParams if parameters were used\n setParams: (...params: P) => {\n currentParams = params;\n if (enabled && hasValidParams(params)) {\n fetchData(params);\n }\n },\n\n };\n\n return asyncChunkInstance;\n}\n","import { Chunk, chunk } from \"./core\";\n\nimport { shallowEqual } from \"../utils\";\n\n// Helper type to extract the value type from a Chunk\nexport type ChunkValue<T> = T extends Chunk<infer U> ? U : never;\n\n// Helper type to transform an array of Chunks into an array of their value types\nexport type DependencyValues<T extends Chunk<any>[]> = {\n [K in keyof T]: T[K] extends Chunk<any> ? ChunkValue<T[K]> : never;\n};\n\nexport interface Computed<T> extends Chunk<T> {\n /**\n * Checks if the computed value needs to be recalculated due to dependency changes.\n * @returns True if the computed value is dirty, false otherwise.\n */\n isDirty: () => boolean;\n /** Manually forces recalculation of the computed value from its dependencies. */\n recompute: () => void;\n}\n\nexport function computed<TDeps extends Chunk<any>[], TResult>(\n dependencies: [...TDeps],\n computeFn: (...args: DependencyValues<TDeps>) => TResult\n): Computed<TResult> {\n const dependencyValues = dependencies.map(dep => dep.get());\n let cachedValue = computeFn(...dependencyValues as DependencyValues<TDeps>);\n\n const computedChunk = chunk(cachedValue);\n const originalSet = computedChunk.set;\n\n let isDirty = false;\n\n // Direct synchronous recomputation\n const recompute = () => {\n let hasChanges = false;\n\n for (let i = 0; i < dependencies.length; i++) {\n const newValue = dependencies[i].get();\n if (newValue !== dependencyValues[i]) {\n dependencyValues[i] = newValue;\n hasChanges = true;\n }\n }\n\n if (hasChanges) {\n const newValue = computeFn(...dependencyValues as DependencyValues<TDeps>);\n // Fast path for primitives only. Avoids shallowEqual for performance.\n if (newValue !== cachedValue) {\n // Only use shallowEqual for objects when needed\n if (typeof newValue !== 'object' || typeof cachedValue !== 'object' || !shallowEqual(newValue, cachedValue)) {\n cachedValue = newValue;\n originalSet(newValue);\n }\n }\n isDirty = false;\n }\n };\n\n const unsubs = dependencies.map(dep =>\n dep.subscribe(() => {\n isDirty = true;\n recompute();\n })\n );\n\n return {\n ...computedChunk,\n get: () => {\n if (isDirty) recompute();\n return cachedValue;\n },\n recompute,\n isDirty: () => isDirty,\n set: () => { throw new Error('Cannot set values directly on computed. Modify the source chunk instead.'); },\n reset: () => {\n dependencies.forEach(dep => {\n if (typeof dep.reset === 'function') {\n dep.reset();\n }\n });\n isDirty = true;\n recompute();\n return cachedValue;\n },\n destroy: () => { unsubs.forEach(unsub => unsub()); computedChunk.destroy?.(); }\n };\n}\n","import { shallowEqual } from \"../utils\";\nimport { Chunk, chunk } from \"./core\";\n\nexport interface SelectOptions {\n /**\n * Configuration options for selector functions.\n * @property {boolean} [useShallowEqual] - When true, performs a shallow equality check\n * on the derived selector results to prevent unnecessary updates.\n */\n useShallowEqual?: boolean;\n}\n\n/**\n * Creates a derived read-only chunk based on a selector function.\n * @param sourceChunk The source chunk to derive from.\n * @param selector A function that extracts part of the source value.\n * @param options Optional settings for shallow equality comparison.\n * @returns A read-only derived chunk.\n */\n\nexport function select<T, S>(\n sourceChunk: Chunk<T>,\n selector: (value: T) => S,\n options: SelectOptions = {}\n): Chunk<S> {\n const { useShallowEqual = false } = options;\n\n let prevSourceValue = sourceChunk.get();\n let currentResult = selector(prevSourceValue);\n\n const derivedChunk = chunk(currentResult);\n\n const update = () => {\n const newSourceValue = sourceChunk.get();\n const newResult = selector(newSourceValue);\n\n // Always update the reference to source value\n prevSourceValue = newSourceValue;\n\n // Check if the result has changed\n const resultChanged = useShallowEqual\n ? !shallowEqual(newResult, currentResult)\n : newResult !== currentResult;\n\n if (resultChanged) {\n currentResult = newResult;\n derivedChunk.set(newResult);\n }\n };\n\n const unsubscribe = sourceChunk.subscribe(update);\n\n return {\n get: () => derivedChunk.get(),\n set: () => {\n throw new Error('Cannot set values directly on a selector. Modify the source chunk instead.');\n },\n subscribe: derivedChunk.subscribe,\n derive: <D>(fn: (value: S) => D) => select(derivedChunk, fn, options), // Pass options to nested selectors\n reset: () => {\n throw new Error('Cannot reset a selector chunk. Reset the source chunk instead.');\n },\n destroy: () => {\n unsubscribe();\n derivedChunk.destroy();\n }\n };\n}\n","// Middleware passed to chunks\nexport { logger } from \"./logger\";\nexport { nonNegativeValidator } from \"./validator\";\n\n// Middleware used with chunks\nexport { withHistory } from \"./history\";\nexport { withPersistence } from './persistence';\n","import { Middleware } from \"../core/core\";\n\nexport const logger: Middleware<any> = (value, next) => {\n console.log(\"Setting value:\", value);\n next(value);\n};\n","import { chunk, Middleware } from \"../core/core\";\n\nexport const nonNegativeValidator: Middleware<number> = (value, next) => {\n if (value < 0) {\n throw new Error(\"Value must be non-negative!\");\n }\n next(value); // If validation passes, proceed with the update\n};\n","import { Chunk } from \"../core/core\";\n\nexport interface ChunkWithHistory<T> extends Chunk<T> {\n /**\n * Reverts to the previous state (if available).\n */\n undo: () => void;\n /**\n * Moves to the next state (if available).\n */\n redo: () => void;\n /**\n * Returns true if there is a previous state to revert to.\n */\n canUndo: () => boolean;\n /**\n * Returns true if there is a next state to move to.\n */\n canRedo: () => boolean;\n /**\n * Returns an array of all the values in the history.\n */\n getHistory: () => T[];\n /**\n * Clears the history, keeping only the current value.\n */\n clearHistory: () => void;\n}\n\nexport function withHistory<T>(\n baseChunk: Chunk<T>,\n options: { maxHistory?: number } = {}\n): ChunkWithHistory<T> {\n const { maxHistory = 100 } = options;\n const history: T[] = [baseChunk.get()];\n let currentIndex = 0;\n let isHistoryAction = false;\n\n const historyChunk: ChunkWithHistory<T> = {\n ...baseChunk,\n\n set: (newValueOrUpdater: T | ((currentValue: T) => T)) => {\n if (isHistoryAction) {\n baseChunk.set(newValueOrUpdater);\n return;\n }\n\n // Process the value or updater function\n let newValue: T;\n if (typeof newValueOrUpdater === 'function') {\n // Get current value and apply the updater function\n const currentValue = baseChunk.get();\n newValue = (newValueOrUpdater as ((currentValue: T) => T))(currentValue);\n } else {\n // Use directly as the new value\n newValue = newValueOrUpdater;\n }\n history.splice(currentIndex + 1);\n history.push(newValue);\n\n // Limit history size\n if (history.length > maxHistory) {\n console.warn(\"History limit reached. Removing oldest entries.\");\n const removeCount = history.length - maxHistory;\n history.splice(0, removeCount);\n currentIndex = Math.max(0, currentIndex - removeCount);\n }\n\n currentIndex = history.length - 1;\n baseChunk.set(newValue);\n },\n\n undo: () => {\n if (!historyChunk.canUndo()) return;\n\n isHistoryAction = true;\n currentIndex--;\n historyChunk.set(history[currentIndex]);\n isHistoryAction = false;\n },\n\n redo: () => {\n if (!historyChunk.canRedo()) return;\n\n isHistoryAction = true;\n currentIndex++;\n historyChunk.set(history[currentIndex]);\n isHistoryAction = false;\n },\n\n canUndo: () => currentIndex > 0,\n\n canRedo: () => currentIndex < history.length - 1,\n\n getHistory: () => [...history],\n\n clearHistory: () => {\n const currentValue = baseChunk.get();\n history.length = 0;\n history.push(currentValue);\n currentIndex = 0;\n },\n\n // Override destroy to clean up history\n destroy: () => {\n history.length = 0;\n baseChunk.destroy();\n }\n }\n\n return historyChunk;\n\n}\n","import { Chunk } from \"../core/core\";\n\nexport interface PersistOptions<T> {\n key: string;\n storage?: Storage;\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n}\n\nexport function withPersistence<T>(\n baseChunk: Chunk<T>,\n options: PersistOptions<T>\n): Chunk<T> {\n const {\n key,\n storage = localStorage,\n serialize = JSON.stringify,\n deserialize = JSON.parse,\n } = options;\n\n // Try to load initial state from storage\n try {\n const savedChunk = storage.getItem(key);\n if (savedChunk) {\n const parsed = deserialize(savedChunk);\n baseChunk.set(parsed)\n }\n } catch (error) {\n console.error('Failed to load persisted state:', error);\n }\n\n // Save to storage\n baseChunk.subscribe((newValue) => {\n try {\n const serialized = serialize(newValue);\n storage.setItem(key, serialized);\n\n } catch (error) {\n console.log('Failed to persist chunk', error)\n }\n })\n\n return baseChunk\n\n}\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../../src/middleware/logger.ts","../../src/middleware/validator.ts","../../src/middleware/history.ts","../../src/middleware/persistence.ts"],"names":["logger","value","next","nonNegativeValidator","withHistory","baseChunk","options","maxHistory","history","currentIndex","isHistoryAction","historyChunk","newValueOrUpdater","newValue","currentValue","removeCount","withPersistence","key","storage","serialize","deserialize","savedChunk","parsed","error","serialized"],"mappings":"AAEO,IAAMA,CAA0B,CAAA,CAACC,CAAOC,CAAAA,CAAAA,GAAS,CACtD,OAAQ,CAAA,GAAA,CAAI,gBAAkBD,CAAAA,CAAK,EACnCC,CAAKD,CAAAA,CAAK,EACZ,MCHaE,CAA2C,CAAA,CAACF,CAAOC,CAAAA,CAAAA,GAAS,CACvE,GAAID,CAAQ,CAAA,CAAA,CACV,MAAM,IAAI,KAAA,CAAM,6BAA6B,CAAA,CAE/CC,EAAKD,CAAK,EACZ,ECsBO,SAASG,EACdC,CACAC,CAAAA,CAAAA,CAAmC,EAAC,CACf,CACrB,GAAM,CAAE,UAAA,CAAAC,EAAa,GAAI,CAAA,CAAID,CACvBE,CAAAA,CAAAA,CAAe,CAACH,CAAU,CAAA,GAAA,EAAK,CAAA,CACjCI,EAAe,CACfC,CAAAA,CAAAA,CAAkB,KAEhBC,CAAAA,CAAAA,CAAoC,CACxC,GAAGN,CAEH,CAAA,GAAA,CAAMO,GAAoD,CACxD,GAAIF,CAAiB,CAAA,CACnBL,EAAU,GAAIO,CAAAA,CAAiB,CAC/B,CAAA,MACF,CAGA,IAAIC,CAAAA,CACJ,GAAI,OAAOD,CAAsB,EAAA,UAAA,CAAY,CAE3C,IAAME,EAAeT,CAAU,CAAA,GAAA,EAC/BQ,CAAAA,CAAAA,CAAYD,EAA+CE,CAAY,EACzE,CAEED,KAAAA,CAAAA,CAAWD,EAMb,GAJAJ,CAAAA,CAAQ,MAAOC,CAAAA,CAAAA,CAAe,CAAC,CAAA,CAC/BD,CAAQ,CAAA,IAAA,CAAKK,CAAQ,CAGjBL,CAAAA,CAAAA,CAAQ,MAASD,CAAAA,CAAAA,CAAY,CAC/B,OAAQ,CAAA,IAAA,CAAK,iDAAiD,CAAA,CAC9D,IAAMQ,CAAcP,CAAAA,CAAAA,CAAQ,MAASD,CAAAA,CAAAA,CACrCC,CAAQ,CAAA,MAAA,CAAO,CAAGO,CAAAA,CAAW,EAC7BN,CAAe,CAAA,IAAA,CAAK,GAAI,CAAA,CAAA,CAAGA,EAAeM,CAAW,EACvD,CAEAN,CAAAA,CAAeD,EAAQ,MAAS,CAAA,CAAA,CAChCH,CAAU,CAAA,GAAA,CAAIQ,CAAQ,EACxB,CAEA,CAAA,IAAA,CAAM,IAAM,CACLF,CAAAA,CAAa,OAAQ,EAAA,GAE1BD,EAAkB,IAClBD,CAAAA,CAAAA,EAAAA,CACAE,CAAa,CAAA,GAAA,CAAIH,EAAQC,CAAY,CAAC,CACtCC,CAAAA,CAAAA,CAAkB,KACpB,EAAA,CAAA,CAEA,IAAM,CAAA,IAAM,CACLC,CAAa,CAAA,OAAA,EAElBD,GAAAA,CAAAA,CAAkB,KAClBD,CACAE,EAAAA,CAAAA,CAAAA,CAAa,GAAIH,CAAAA,CAAAA,CAAQC,CAAY,CAAC,CAAA,CACtCC,CAAkB,CAAA,KAAA,EACpB,CAEA,CAAA,OAAA,CAAS,IAAMD,CAAAA,CAAe,EAE9B,OAAS,CAAA,IAAMA,CAAeD,CAAAA,CAAAA,CAAQ,OAAS,CAE/C,CAAA,UAAA,CAAY,IAAM,CAAC,GAAGA,CAAO,CAAA,CAE7B,YAAc,CAAA,IAAM,CAClB,IAAMM,CAAeT,CAAAA,CAAAA,CAAU,KAC/BG,CAAAA,CAAAA,CAAQ,MAAS,CAAA,CAAA,CACjBA,EAAQ,IAAKM,CAAAA,CAAY,CACzBL,CAAAA,CAAAA,CAAe,EACjB,CAGA,CAAA,OAAA,CAAS,IAAM,CACbD,CAAQ,CAAA,MAAA,CAAS,CACjBH,CAAAA,CAAAA,CAAU,UACZ,CACF,CAEA,CAAA,OAAOM,CAET,CCvGO,SAASK,CACdX,CAAAA,CAAAA,CACAC,EACU,CACV,GAAM,CACJ,GAAA,CAAAW,CACA,CAAA,OAAA,CAAAC,CAAU,CAAA,YAAA,CACV,UAAAC,CAAY,CAAA,IAAA,CAAK,SACjB,CAAA,WAAA,CAAAC,EAAc,IAAK,CAAA,KACrB,CAAId,CAAAA,CAAAA,CAGJ,GAAI,CACF,IAAMe,CAAaH,CAAAA,CAAAA,CAAQ,OAAQD,CAAAA,CAAG,CACtC,CAAA,GAAII,EAAY,CACd,IAAMC,CAASF,CAAAA,CAAAA,CAAYC,CAAU,CACrChB,CAAAA,CAAAA,CAAU,GAAIiB,CAAAA,CAAM,EACtB,CACF,CAAA,MAASC,CAAO,CAAA,CACd,OAAQ,CAAA,KAAA,CAAM,iCAAmCA,CAAAA,CAAK,EACxD,CAGA,OAAAlB,CAAU,CAAA,SAAA,CAAWQ,GAAa,CAChC,GAAI,CACF,IAAMW,EAAaL,CAAUN,CAAAA,CAAQ,CACrCK,CAAAA,CAAAA,CAAQ,OAAQD,CAAAA,CAAAA,CAAKO,CAAU,EAEjC,OAASD,CAAO,CAAA,CACd,OAAQ,CAAA,GAAA,CAAI,0BAA2BA,CAAK,EAC9C,CACF,CAAC,EAEMlB,CAET","file":"index.js","sourcesContent":["import { Middleware } from \"../core/core\";\n\nexport const logger: Middleware<any> = (value, next) => {\n console.log(\"Setting value:\", value);\n next(value);\n};\n","import { chunk, Middleware } from \"../core/core\";\n\nexport const nonNegativeValidator: Middleware<number> = (value, next) => {\n if (value < 0) {\n throw new Error(\"Value must be non-negative!\");\n }\n next(value); // If validation passes, proceed with the update\n};\n","import { Chunk } from \"../core/core\";\n\nexport interface ChunkWithHistory<T> extends Chunk<T> {\n /**\n * Reverts to the previous state (if available).\n */\n undo: () => void;\n /**\n * Moves to the next state (if available).\n */\n redo: () => void;\n /**\n * Returns true if there is a previous state to revert to.\n */\n canUndo: () => boolean;\n /**\n * Returns true if there is a next state to move to.\n */\n canRedo: () => boolean;\n /**\n * Returns an array of all the values in the history.\n */\n getHistory: () => T[];\n /**\n * Clears the history, keeping only the current value.\n */\n clearHistory: () => void;\n}\n\nexport function withHistory<T>(\n baseChunk: Chunk<T>,\n options: { maxHistory?: number } = {}\n): ChunkWithHistory<T> {\n const { maxHistory = 100 } = options;\n const history: T[] = [baseChunk.get()];\n let currentIndex = 0;\n let isHistoryAction = false;\n\n const historyChunk: ChunkWithHistory<T> = {\n ...baseChunk,\n\n set: (newValueOrUpdater: T | ((currentValue: T) => T)) => {\n if (isHistoryAction) {\n baseChunk.set(newValueOrUpdater);\n return;\n }\n\n // Process the value or updater function\n let newValue: T;\n if (typeof newValueOrUpdater === 'function') {\n // Get current value and apply the updater function\n const currentValue = baseChunk.get();\n newValue = (newValueOrUpdater as ((currentValue: T) => T))(currentValue);\n } else {\n // Use directly as the new value\n newValue = newValueOrUpdater;\n }\n history.splice(currentIndex + 1);\n history.push(newValue);\n\n // Limit history size\n if (history.length > maxHistory) {\n console.warn(\"History limit reached. Removing oldest entries.\");\n const removeCount = history.length - maxHistory;\n history.splice(0, removeCount);\n currentIndex = Math.max(0, currentIndex - removeCount);\n }\n\n currentIndex = history.length - 1;\n baseChunk.set(newValue);\n },\n\n undo: () => {\n if (!historyChunk.canUndo()) return;\n\n isHistoryAction = true;\n currentIndex--;\n historyChunk.set(history[currentIndex]);\n isHistoryAction = false;\n },\n\n redo: () => {\n if (!historyChunk.canRedo()) return;\n\n isHistoryAction = true;\n currentIndex++;\n historyChunk.set(history[currentIndex]);\n isHistoryAction = false;\n },\n\n canUndo: () => currentIndex > 0,\n\n canRedo: () => currentIndex < history.length - 1,\n\n getHistory: () => [...history],\n\n clearHistory: () => {\n const currentValue = baseChunk.get();\n history.length = 0;\n history.push(currentValue);\n currentIndex = 0;\n },\n\n // Override destroy to clean up history\n destroy: () => {\n history.length = 0;\n baseChunk.destroy();\n }\n }\n\n return historyChunk;\n\n}\n","import { Chunk } from \"../core/core\";\n\nexport interface PersistOptions<T> {\n key: string;\n storage?: Storage;\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n}\n\nexport function withPersistence<T>(\n baseChunk: Chunk<T>,\n options: PersistOptions<T>\n): Chunk<T> {\n const {\n key,\n storage = localStorage,\n serialize = JSON.stringify,\n deserialize = JSON.parse,\n } = options;\n\n // Try to load initial state from storage\n try {\n const savedChunk = storage.getItem(key);\n if (savedChunk) {\n const parsed = deserialize(savedChunk);\n baseChunk.set(parsed)\n }\n } catch (error) {\n console.error('Failed to load persisted state:', error);\n }\n\n // Save to storage\n baseChunk.subscribe((newValue) => {\n try {\n const serialized = serialize(newValue);\n storage.setItem(key, serialized);\n\n } catch (error) {\n console.log('Failed to persist chunk', error)\n }\n })\n\n return baseChunk\n\n}\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../../src/core/core.ts","../../src/utils.ts","../../src/core/selector.ts","../../src/use-react/hooks/useChunk.ts","../../src/use-react/hooks/useDerive.ts","../../src/core/computed.ts","../../src/use-react/hooks/useComputed.ts","../../src/use-react/hooks/useChunkValue.ts","../../src/use-react/hooks/useChunkProperty.ts","../../src/use-react/hooks/useChunkValues.ts","../../src/use-react/hooks/useAsyncChunk.ts"],"names":["dirtyChunks","chunkRegistry","chunkIdCounter","chunk","initialValue","middleware","value","subscribers","chunkId","notify","subscriber","notifySubscribers","get","set","newValueOrUpdater","newValue","validateObjectShape","processedValue","processMiddleware","subscribe","callback","fn","initialDerivedValue","derivedChunk","unsubscribe","newDerivedValue","originalDestroy","currentValue","index","current","middlewareFn","middlewareName","nextCalled","nextValue","val","error","errorMessage","shallowEqual","a","b","i","keysA","keysB","key","original","updated","path","originalKeys","updatedKeys","extraKeys","select","sourceChunk","selector","options","useShallowEqual","prevSourceValue","currentResult","update","newSourceValue","newResult","useChunk","selectedChunk","state","setState","useState","useEffect","useCallback","valueOrUpdater","reset","destroy","useDerive","fnRef","useRef","useMemo","derivedValue","computed","dependencies","computeFn","dependencyValues","dep","cachedValue","computedChunk","originalSet","isDirty","recompute","hasChanges","unsubs","unsub","useComputed","computedValue","useChunkValue","useChunkProperty","property","useChunkValues","chunks","values","setValues","unsubscribes","prev","newValues","useAsyncChunk","asyncChunk","params","newState","reload","refresh","mutate","mutator","setParams","data","loading","lastFetched","result"],"mappings":"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","file":"index.js","sourcesContent":["import { processMiddleware, validateObjectShape } from \"../utils\";\n\nexport type Subscriber<T> = (newValue: T) => void;\nexport type Middleware<T> = (value: T, next: (newValue: T) => void) => void;\nexport type NamedMiddleware<T> = {\n name?: string;\n fn: Middleware<T>;\n};\n\nexport interface Chunk<T> {\n /** Get the current value of the chunk. */\n get: () => T;\n /** Set a new value for the chunk & Update existing value efficiently. */\n set: (newValueOrUpdater: T | ((currentValue: T) => T)) => void;\n /** Subscribe to changes in the chunk. Returns an unsubscribe function. */\n subscribe: (callback: Subscriber<T>) => () => void;\n /** Create a derived chunk based on this chunk's value. */\n derive: <D>(fn: (value: T) => D) => Chunk<D>;\n /** Reset the chunk to its initial value. */\n reset: () => void;\n /** Destroy the chunk and all its subscribers. */\n destroy: () => void;\n}\n\nlet isBatching = false;\nconst dirtyChunks = new Set<number>();\nconst chunkRegistry = new Map<number, { notify: () => void }>();\nlet chunkIdCounter = 0;\n\n/**\n * Batch multiple chunk updates into a single re-render.\n * Useful for updating multiple chunks at once without causing multiple re-renders.\n */\nexport function batch(callback: () => void) {\n const wasBatchingBefore = isBatching;\n isBatching = true;\n try {\n callback();\n } finally {\n if (!wasBatchingBefore) {\n isBatching = false;\n const chunks = Array.from(dirtyChunks); // Snapshot to avoid mutation issues\n dirtyChunks.clear(); // Clear early to prevent re-adds\n chunks.forEach(id => {\n const chunk = chunkRegistry.get(id);\n if (chunk) chunk.notify();\n });\n }\n }\n}\n\nexport function chunk<T>(initialValue: T, middleware: (Middleware<T> | NamedMiddleware<T>)[] = []): Chunk<T> {\n if (initialValue === null) {\n throw new Error(\"Initial value cannot be null.\");\n }\n\n let value = initialValue;\n const subscribers = new Set<Subscriber<T>>();\n const chunkId = chunkIdCounter++;\n\n const notify = () => {\n subscribers.forEach(subscriber => subscriber(value));\n };\n\n chunkRegistry.set(chunkId, { notify });\n\n const notifySubscribers = () => {\n if (subscribers.size === 0) return; // Skip if no subscribers\n if (isBatching) {\n dirtyChunks.add(chunkId);\n } else {\n notify();\n }\n };\n\n const get = () => value;\n\n const set = (newValueOrUpdater: T | ((currentValue: T) => T)) => {\n let newValue: T;\n\n if (typeof newValueOrUpdater === 'function') {\n const updaterFn = newValueOrUpdater as (currentValue: T) => T;\n newValue = updaterFn(value);\n } else {\n newValue = newValueOrUpdater;\n }\n\n validateObjectShape(value, newValue);\n\n const processedValue = processMiddleware(newValue, middleware);\n\n if (processedValue !== value) {\n value = processedValue as T & {};\n notifySubscribers();\n }\n };\n\n const subscribe = (callback: Subscriber<T>) => {\n if (typeof callback !== \"function\") {\n throw new Error(\"Callback must be a function.\");\n }\n subscribers.add(callback);\n callback(value);\n return () => subscribers.delete(callback);\n };\n\n const reset = () => {\n value = initialValue;\n notifySubscribers();\n };\n\n const destroy = () => {\n subscribers.clear();\n value = initialValue;\n dirtyChunks.delete(chunkId);\n chunkRegistry.delete(chunkId);\n };\n\n const derive = <D>(fn: (value: T) => D) => {\n if (typeof fn !== \"function\") {\n throw new Error(\"Derive function must be a function.\");\n }\n\n const initialDerivedValue = fn(value);\n const derivedChunk = chunk(initialDerivedValue);\n\n const unsubscribe = subscribe(() => {\n const newDerivedValue = fn(value);\n derivedChunk.set(newDerivedValue);\n });\n\n // Add a cleanup method to the derived chunk\n const originalDestroy = derivedChunk.destroy;\n derivedChunk.destroy = () => {\n unsubscribe();\n originalDestroy();\n };\n\n return derivedChunk;\n };\n\n return { get, set, subscribe, derive, reset, destroy };\n}\n","import { chunk, Chunk, Middleware, NamedMiddleware } from \"./core/core\";\n\nimport { AsyncChunk } from \"./core/asyncChunk\";\nimport { CombinedData, CombinedState } from \"./core/types\";\n\nexport function isValidChunkValue(value: unknown): boolean {\n return value !== null;\n}\n\nexport function isValidChunk<T>(value: unknown, validateBehavior = false): value is Chunk<T> {\n if (!isChunk(value)) {\n return false;\n }\n\n if (!validateBehavior) {\n return true;\n }\n\n try {\n const currentValue = value.get();\n value.set(currentValue);\n return true;\n } catch {\n return false;\n }\n}\n\nexport function isChunk<T>(value: unknown): value is Chunk<T> {\n if (!value || typeof value !== 'object') {\n return false;\n }\n\n const chunk = value as Record<string, unknown>;\n const requiredMethods = [\n 'get',\n 'set',\n 'update',\n 'subscribe',\n 'derive',\n 'reset',\n 'destroy'\n ] as const;\n\n return requiredMethods.every(method =>\n typeof chunk[method] === 'function'\n );\n}\n\nexport function once<T>(fn: () => T): () => T {\n let called = false;\n let result: T;\n return () => {\n if (!called) {\n result = fn();\n called = true;\n }\n return result;\n };\n};\n\nexport function combineAsyncChunks<T extends Record<string, AsyncChunk<any>>>(\n chunks: T\n): Chunk<CombinedState<T>> {\n const initialData = Object.keys(chunks).reduce((acc, key) => {\n acc[key as keyof T] = null;\n return acc;\n }, {} as CombinedData<T>);\n\n const initialState: CombinedState<T> = {\n loading: Object.keys(chunks).length > 0,\n error: null,\n errors: {},\n data: initialData\n };\n\n const combined = chunk(initialState);\n\n const chunkValues = Object.values(chunks);\n\n Object.entries(chunks).forEach(([key, asyncChunk]) => {\n asyncChunk.subscribe((state) => {\n const currentState = combined.get();\n\n let hasLoading = false;\n let firstError: Error | null = null;\n const allErrors: Partial<{ [K in keyof T]: Error }> = {};\n\n Object.entries(chunks).forEach(([chunkKey, chunk]) => {\n const chunkState = chunk.get();\n if (chunkState.loading) hasLoading = true;\n if (chunkState.error) {\n if (!firstError) firstError = chunkState.error;\n allErrors[chunkKey as keyof T] = chunkState.error;\n }\n });\n\n combined.set({\n loading: hasLoading,\n error: firstError,\n errors: allErrors,\n data: {\n ...currentState.data,\n [key]: state.data\n },\n });\n });\n });\n\n return combined;\n}\n\nexport function processMiddleware<T>(initialValue: T, middleware: (Middleware<T> | NamedMiddleware<T>)[]): T {\n if (initialValue === null) {\n throw new Error(\"Value cannot be null.\");\n }\n\n let currentValue = initialValue;\n let index = 0;\n\n while (index < middleware.length) {\n const current = middleware[index];\n\n const middlewareFn = typeof current === 'function' ? current : current.fn;\n const middlewareName = typeof current === 'function' ? `index ${index}` : (current.name || `index ${index}`);\n let nextCalled = false;\n let nextValue: T | null = null;\n\n try {\n middlewareFn(currentValue, (val) => {\n nextCalled = true;\n nextValue = val;\n });\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n throw new Error(`Middleware \"${middlewareName}\" threw an error: ${errorMessage}`);\n }\n\n if (!nextCalled) break;\n\n if (nextValue === null) {\n throw new Error(`Middleware at index ${index} returned null value.`);\n }\n\n currentValue = nextValue;\n index++;\n }\n\n return currentValue;\n}\n\nexport function shallowEqual<T>(a: T, b: T): boolean {\n if (a === b) {\n return true;\n }\n\n if (!a || !b || typeof a !== typeof b) {\n return false;\n }\n\n if (Array.isArray(a) && Array.isArray(b)) {\n if (a.length !== b.length) {\n return false;\n }\n for (let i = 0; i < a.length; i++) {\n if (a[i] !== b[i]) {\n return false;\n }\n }\n return true;\n }\n\n if (typeof a === 'object' && typeof b === 'object') {\n const keysA = Object.keys(a);\n const keysB = Object.keys(b);\n\n if (keysA.length !== keysB.length) {\n return false;\n }\n\n for (const key of keysA) {\n if (!Object.prototype.hasOwnProperty.call(b, key) || (a as any)[key] !== (b as any)[key]) {\n return false;\n }\n }\n return true;\n }\n\n // For primitive types, return false. Strict equality already handled by initial check\n return false;\n}\n\nexport function validateObjectShape<T>(original: T, updated: T, path = ''): void {\n if (typeof original === 'object' && original !== null && typeof updated === 'object' && updated !== null) {\n if (Array.isArray(original) && Array.isArray(updated)) {\n if (original.length > 0 && typeof original[0] === 'object') {\n for (let i = 0; i < updated.length; i++) {\n validateObjectShape(original[0], updated[i], `${path}[${i}]`);\n }\n }\n } else if (!Array.isArray(original) && !Array.isArray(updated)) {\n const originalKeys = Object.keys(original as object);\n const updatedKeys = Object.keys(updated as object);\n const extraKeys = updatedKeys.filter(key => !originalKeys.includes(key));\n\n if (extraKeys.length > 0) {\n const fullPath = path || 'root';\n console.error(`🚨 Stunk: Unknown properties detected at '${fullPath}': ${extraKeys.join(', ')}. This might cause bugs.`);\n console.error('Expected keys:', originalKeys);\n console.error('Received keys:', updatedKeys);\n }\n\n // Recurse into common keys\n for (const key of originalKeys) {\n validateObjectShape((original as any)[key], (updated as any)[key], path ? `${path}.${key}` : key);\n }\n }\n }\n}\n","import { shallowEqual } from \"../utils\";\nimport { Chunk, chunk } from \"./core\";\n\nexport interface SelectOptions {\n /**\n * Configuration options for selector functions.\n * @property {boolean} [useShallowEqual] - When true, performs a shallow equality check\n * on the derived selector results to prevent unnecessary updates.\n */\n useShallowEqual?: boolean;\n}\n\n/**\n * Creates a derived read-only chunk based on a selector function.\n * @param sourceChunk The source chunk to derive from.\n * @param selector A function that extracts part of the source value.\n * @param options Optional settings for shallow equality comparison.\n * @returns A read-only derived chunk.\n */\n\nexport function select<T, S>(\n sourceChunk: Chunk<T>,\n selector: (value: T) => S,\n options: SelectOptions = {}\n): Chunk<S> {\n const { useShallowEqual = false } = options;\n\n let prevSourceValue = sourceChunk.get();\n let currentResult = selector(prevSourceValue);\n\n const derivedChunk = chunk(currentResult);\n\n const update = () => {\n const newSourceValue = sourceChunk.get();\n const newResult = selector(newSourceValue);\n\n // Always update the reference to source value\n prevSourceValue = newSourceValue;\n\n // Check if the result has changed\n const resultChanged = useShallowEqual\n ? !shallowEqual(newResult, currentResult)\n : newResult !== currentResult;\n\n if (resultChanged) {\n currentResult = newResult;\n derivedChunk.set(newResult);\n }\n };\n\n const unsubscribe = sourceChunk.subscribe(update);\n\n return {\n get: () => derivedChunk.get(),\n set: () => {\n throw new Error('Cannot set values directly on a selector. Modify the source chunk instead.');\n },\n subscribe: derivedChunk.subscribe,\n derive: <D>(fn: (value: S) => D) => select(derivedChunk, fn, options), // Pass options to nested selectors\n reset: () => {\n throw new Error('Cannot reset a selector chunk. Reset the source chunk instead.');\n },\n destroy: () => {\n unsubscribe();\n derivedChunk.destroy();\n }\n };\n}\n","import { useState, useEffect, useCallback } from \"react\";\n\nimport { select } from \"../../core/selector\";\n\nimport type { Chunk } from \"../../core/core\";\n\n/**\n * A lightweight hook that subscribes to a chunk and returns its current value, along with setters, selector, reset and destroy.\n * Ensures reactivity and prevents unnecessary re-renders.\n */\n\nexport function useChunk<T, S = T>(\n chunk: Chunk<T>,\n selector?: (value: T) => S\n) {\n const selectedChunk = selector ? select(chunk, selector) : chunk;\n\n const [state, setState] = useState<S>(() => selectedChunk.get() as S);\n\n useEffect(() => {\n const unsubscribe = selectedChunk.subscribe((newValue) => {\n setState(() => newValue as S);\n });\n return () => unsubscribe();\n }, [selectedChunk]);\n\n const set = useCallback((valueOrUpdater: T | ((currentValue: T) => T)) => {\n chunk.set(valueOrUpdater);\n }, [chunk]);\n\n const reset = useCallback(() => {\n chunk.reset();\n }, [chunk]);\n\n const destroy = useCallback(() => {\n chunk.destroy();\n }, [chunk]);\n\n return [state, set, reset, destroy] as const;\n}\n","import { useEffect, useMemo, useRef } from \"react\";\n\nimport { useChunk } from \"./useChunk\";\nimport type { Chunk } from \"../../core/core\";\n\n/**\n * A hook for creating a read-only derived value from a chunk.\n * Ensures reactivity and updates when the source chunk changes.\n */\nexport function useDerive<T, D>(chunk: Chunk<T>, fn: (value: T) => D): D {\n const fnRef = useRef(fn);\n\n useEffect(() => {\n fnRef.current = fn;\n }, [fn]);\n\n const derivedChunk = useMemo(() => {\n return chunk.derive((value) => fnRef.current(value));\n }, [chunk]);\n\n const [derivedValue] = useChunk(derivedChunk);\n return derivedValue;\n}\n","import { Chunk, chunk } from \"./core\";\n\nimport { shallowEqual } from \"../utils\";\n\n// Helper type to extract the value type from a Chunk\nexport type ChunkValue<T> = T extends Chunk<infer U> ? U : never;\n\n// Helper type to transform an array of Chunks into an array of their value types\nexport type DependencyValues<T extends Chunk<any>[]> = {\n [K in keyof T]: T[K] extends Chunk<any> ? ChunkValue<T[K]> : never;\n};\n\nexport interface Computed<T> extends Chunk<T> {\n /**\n * Checks if the computed value needs to be recalculated due to dependency changes.\n * @returns True if the computed value is dirty, false otherwise.\n */\n isDirty: () => boolean;\n /** Manually forces recalculation of the computed value from its dependencies. */\n recompute: () => void;\n}\n\nexport function computed<TDeps extends Chunk<any>[], TResult>(\n dependencies: [...TDeps],\n computeFn: (...args: DependencyValues<TDeps>) => TResult\n): Computed<TResult> {\n const dependencyValues = dependencies.map(dep => dep.get());\n let cachedValue = computeFn(...dependencyValues as DependencyValues<TDeps>);\n\n const computedChunk = chunk(cachedValue);\n const originalSet = computedChunk.set;\n\n let isDirty = false;\n\n // Direct synchronous recomputation\n const recompute = () => {\n let hasChanges = false;\n\n for (let i = 0; i < dependencies.length; i++) {\n const newValue = dependencies[i].get();\n if (newValue !== dependencyValues[i]) {\n dependencyValues[i] = newValue;\n hasChanges = true;\n }\n }\n\n if (hasChanges) {\n const newValue = computeFn(...dependencyValues as DependencyValues<TDeps>);\n // Fast path for primitives only. Avoids shallowEqual for performance.\n if (newValue !== cachedValue) {\n // Only use shallowEqual for objects when needed\n if (typeof newValue !== 'object' || typeof cachedValue !== 'object' || !shallowEqual(newValue, cachedValue)) {\n cachedValue = newValue;\n originalSet(newValue);\n }\n }\n isDirty = false;\n }\n };\n\n const unsubs = dependencies.map(dep =>\n dep.subscribe(() => {\n isDirty = true;\n recompute();\n })\n );\n\n return {\n ...computedChunk,\n get: () => {\n if (isDirty) recompute();\n return cachedValue;\n },\n recompute,\n isDirty: () => isDirty,\n set: () => { throw new Error('Cannot set values directly on computed. Modify the source chunk instead.'); },\n reset: () => {\n dependencies.forEach(dep => {\n if (typeof dep.reset === 'function') {\n dep.reset();\n }\n });\n isDirty = true;\n recompute();\n return cachedValue;\n },\n destroy: () => { unsubs.forEach(unsub => unsub()); computedChunk.destroy?.(); }\n };\n}\n","import { useState, useEffect, useMemo } from \"react\";\n\nimport { computed, DependencyValues } from \"../../core/computed\";\nimport type { Chunk } from \"../../core/core\";\n\n/**\n * A hook that computes a value based on multiple chunks.\n * Automatically re-computes when any dependency changes.\n */\nexport function useComputed<TDeps extends Chunk<any>[], TResult>(\n dependencies: [...TDeps],\n computeFn: (...args: DependencyValues<TDeps>) => TResult\n): TResult {\n // Create the computed value - memoize it based on dependencies to prevent recreation\n const computedValue = useMemo(\n () => computed(dependencies, computeFn),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [...dependencies]\n );\n\n const [state, setState] = useState<TResult>(() => computedValue.get());\n\n useEffect(() => {\n const unsubscribe = computedValue.subscribe((newValue) => {\n setState(newValue);\n });\n\n return () => {\n unsubscribe();\n };\n }, [computedValue]);\n\n return state;\n}\n","import { useChunk } from \"./useChunk\";\n\nimport type { Chunk } from \"../../core/core\";\n\n/**\n * A lightweight hook that subscribes to a chunk and returns only its current value.\n * Useful for read-only components that don't need to update the chunk.\n */\nexport function useChunkValue<T, S = T>(\n chunk: Chunk<T>,\n selector?: (value: T) => S\n): S {\n const [value] = useChunk(chunk, selector);\n return value;\n}\n","import { useMemo } from \"react\";\n\nimport { useChunkValue } from \"./useChunkValue\";\n\nimport type { Chunk } from \"../../core/core\";\n\n/**\n * A hook that subscribes to a specific property of a chunk.\n * This optimizes renders by only updating when the selected property changes.\n */\nexport function useChunkProperty<T, K extends keyof T>(\n chunk: Chunk<T>,\n property: K\n): T[K] {\n const selector = useMemo(\n () => (state: T) => state[property],\n [property]\n );\n\n return useChunkValue(chunk, selector);\n}\n","import { useState, useEffect } from \"react\";\n\nimport type { Chunk } from \"../../core/core\";\n\n/**\n * Hook to read values from multiple chunks at once.\n * Only re-renders when any of the chunk values change.\n */\nexport function useChunkValues<T extends Chunk<any>[]>(\n chunks: [...T]\n): { [K in keyof T]: T[K] extends Chunk<infer U> ? U : never } {\n type ReturnType = { [K in keyof T]: T[K] extends Chunk<infer U> ? U : never };\n\n const [values, setValues] = useState<ReturnType>(() => {\n return chunks.map(chunk => chunk.get()) as ReturnType;\n });\n\n useEffect(() => {\n const unsubscribes = chunks.map((chunk, index) => {\n return chunk.subscribe((newValue) => {\n setValues(prev => {\n const newValues = [...prev] as ReturnType;\n newValues[index] = newValue;\n return newValues;\n });\n });\n });\n\n return () => {\n unsubscribes.forEach(unsubscribe => unsubscribe());\n };\n }, [chunks]);\n\n return values;\n}\n","import { useState, useEffect, useCallback } from \"react\";\nimport { AsyncChunk, AsyncState } from \"../../core/asyncChunk\";\n\n/**\n * A hook that handles asynchronous state with built-in reactivity.\n * Provides loading, error, and data states with full asyncChunk functionality.\n */\nexport function useAsyncChunk<T, E extends Error = Error, P extends any[] = []>(\n asyncChunk: AsyncChunk<T, E> | (AsyncChunk<T, E> & { setParams: (...params: P) => void }),\n params?: P\n) {\n const [state, setState] = useState<AsyncState<T, E>>(() => asyncChunk.get());\n\n useEffect(() => {\n const unsubscribe = asyncChunk.subscribe((newState) => {\n setState(newState);\n });\n\n return () => {\n unsubscribe();\n };\n }, [asyncChunk]);\n\n // Handle parameter updates automatically\n useEffect(() => {\n if (params && 'setParams' in asyncChunk) {\n (asyncChunk as any).setParams(...params);\n }\n }, [asyncChunk, ...(params || [])]);\n\n useEffect(() => {\n return () => {\n asyncChunk.cleanup();\n };\n }, [asyncChunk]);\n\n // Memoize methods to prevent unnecessary re-renders\n const reload = useCallback((...params: any[]) => {\n if ('setParams' in asyncChunk && params.length > 0) {\n return (asyncChunk as any).reload(...params);\n }\n return asyncChunk.reload();\n }, [asyncChunk]);\n\n const refresh = useCallback((...params: any[]) => {\n if ('setParams' in asyncChunk && params.length > 0) {\n return (asyncChunk as any).refresh(...params);\n }\n return asyncChunk.refresh();\n }, [asyncChunk]);\n\n const mutate = useCallback(\n (mutator: (currentData: T | null) => T) => asyncChunk.mutate(mutator),\n [asyncChunk]\n );\n\n const reset = useCallback(() => asyncChunk.reset(), [asyncChunk]);\n\n const setParams = useCallback((...params: any[]) => {\n if ('setParams' in asyncChunk) {\n (asyncChunk as any).setParams(...params);\n }\n }, [asyncChunk]);\n\n const { data, loading, error, lastFetched } = state;\n\n const result = {\n data,\n loading,\n error,\n lastFetched,\n reload,\n refresh,\n mutate,\n reset,\n };\n\n // Only add setParams if the asyncChunk supports it\n if ('setParams' in asyncChunk) {\n (result as any).setParams = setParams;\n }\n\n return result;\n}\n"]}