sibujs 1.0.1 → 1.0.2
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/chunk-OHQQWZ7P.js +969 -0
- package/dist/data.cjs +24 -4
- package/dist/data.js +1 -1
- package/dist/extras.cjs +24 -4
- package/dist/extras.js +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,969 @@
|
|
|
1
|
+
import {
|
|
2
|
+
context
|
|
3
|
+
} from "./chunk-Z65KYU7I.js";
|
|
4
|
+
import {
|
|
5
|
+
derived
|
|
6
|
+
} from "./chunk-FGOEVHY3.js";
|
|
7
|
+
import {
|
|
8
|
+
effect
|
|
9
|
+
} from "./chunk-PZEGYCF5.js";
|
|
10
|
+
import {
|
|
11
|
+
batch,
|
|
12
|
+
signal
|
|
13
|
+
} from "./chunk-YECR7UIA.js";
|
|
14
|
+
|
|
15
|
+
// src/data/retry.ts
|
|
16
|
+
function calculateDelay(attempt, strategy, baseDelay, maxDelay, jitter) {
|
|
17
|
+
let delay;
|
|
18
|
+
switch (strategy) {
|
|
19
|
+
case "exponential":
|
|
20
|
+
delay = baseDelay * 2 ** attempt;
|
|
21
|
+
break;
|
|
22
|
+
case "linear":
|
|
23
|
+
delay = baseDelay * (attempt + 1);
|
|
24
|
+
break;
|
|
25
|
+
case "fixed":
|
|
26
|
+
delay = baseDelay;
|
|
27
|
+
break;
|
|
28
|
+
}
|
|
29
|
+
delay = Math.min(delay, maxDelay);
|
|
30
|
+
if (jitter > 0) {
|
|
31
|
+
const jitterRange = delay * jitter;
|
|
32
|
+
delay += (Math.random() * 2 - 1) * jitterRange;
|
|
33
|
+
}
|
|
34
|
+
return Math.max(0, delay);
|
|
35
|
+
}
|
|
36
|
+
async function withRetry(fn, options, onRetry, signal2) {
|
|
37
|
+
const maxRetries = options?.maxRetries ?? 3;
|
|
38
|
+
const strategy = options?.strategy ?? "exponential";
|
|
39
|
+
const baseDelay = options?.baseDelay ?? 1e3;
|
|
40
|
+
const maxDelay = options?.maxDelay ?? 3e4;
|
|
41
|
+
const jitter = options?.jitter ?? 0.1;
|
|
42
|
+
const shouldRetry = options?.shouldRetry ?? (() => true);
|
|
43
|
+
let lastError;
|
|
44
|
+
for (let attempt = 0; attempt <= maxRetries; attempt++) {
|
|
45
|
+
if (signal2?.aborted) throw new DOMException("Aborted", "AbortError");
|
|
46
|
+
try {
|
|
47
|
+
return await fn();
|
|
48
|
+
} catch (error) {
|
|
49
|
+
lastError = error;
|
|
50
|
+
if (attempt >= maxRetries || !shouldRetry(error, attempt)) throw error;
|
|
51
|
+
const delay = calculateDelay(attempt, strategy, baseDelay, maxDelay, jitter);
|
|
52
|
+
onRetry?.(error, attempt, delay);
|
|
53
|
+
await new Promise((resolve, reject) => {
|
|
54
|
+
const timer = setTimeout(resolve, delay);
|
|
55
|
+
if (signal2) {
|
|
56
|
+
const onAbort = () => {
|
|
57
|
+
clearTimeout(timer);
|
|
58
|
+
reject(new DOMException("Aborted", "AbortError"));
|
|
59
|
+
};
|
|
60
|
+
signal2.addEventListener("abort", onAbort, { once: true });
|
|
61
|
+
}
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
throw lastError;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// src/data/query.ts
|
|
69
|
+
var queryCache = /* @__PURE__ */ new Map();
|
|
70
|
+
function getOrCreateEntry(key, initialData) {
|
|
71
|
+
let entry = queryCache.get(key);
|
|
72
|
+
if (!entry) {
|
|
73
|
+
entry = {
|
|
74
|
+
data: initialData,
|
|
75
|
+
error: void 0,
|
|
76
|
+
dataUpdatedAt: initialData !== void 0 ? Date.now() : 0,
|
|
77
|
+
subscribers: 0,
|
|
78
|
+
gcTimer: null,
|
|
79
|
+
promise: null,
|
|
80
|
+
listeners: /* @__PURE__ */ new Set(),
|
|
81
|
+
refetchers: /* @__PURE__ */ new Set()
|
|
82
|
+
};
|
|
83
|
+
queryCache.set(key, entry);
|
|
84
|
+
}
|
|
85
|
+
return entry;
|
|
86
|
+
}
|
|
87
|
+
function query(key, fetcher, options = {}) {
|
|
88
|
+
const {
|
|
89
|
+
staleTime = 0,
|
|
90
|
+
cacheTime = 3e5,
|
|
91
|
+
enabled = true,
|
|
92
|
+
retry: retryOptions,
|
|
93
|
+
initialData,
|
|
94
|
+
refetchInterval,
|
|
95
|
+
refetchOnWindowFocus = false,
|
|
96
|
+
refetchOnReconnect = false,
|
|
97
|
+
onSuccess,
|
|
98
|
+
onError,
|
|
99
|
+
onSettled
|
|
100
|
+
} = options;
|
|
101
|
+
const resolveKey = typeof key === "function" ? key : () => key;
|
|
102
|
+
const [data, setData] = signal(initialData);
|
|
103
|
+
const [isFetching, setIsFetching] = signal(false);
|
|
104
|
+
const [error, setError] = signal(void 0);
|
|
105
|
+
let abortController = null;
|
|
106
|
+
let disposed = false;
|
|
107
|
+
let currentKey = null;
|
|
108
|
+
let intervalTimer = null;
|
|
109
|
+
const loading = derived(() => isFetching() && data() === void 0);
|
|
110
|
+
const isStale = derived(() => {
|
|
111
|
+
data();
|
|
112
|
+
if (!currentKey) return true;
|
|
113
|
+
const entry = queryCache.get(currentKey);
|
|
114
|
+
if (!entry || entry.dataUpdatedAt === 0) return true;
|
|
115
|
+
return Date.now() - entry.dataUpdatedAt >= staleTime;
|
|
116
|
+
});
|
|
117
|
+
async function doFetch() {
|
|
118
|
+
if (disposed || !currentKey || !enabled) return;
|
|
119
|
+
const key2 = currentKey;
|
|
120
|
+
let entry = queryCache.get(key2);
|
|
121
|
+
if (!entry) {
|
|
122
|
+
entry = getOrCreateEntry(key2);
|
|
123
|
+
entry.subscribers++;
|
|
124
|
+
entry.listeners.add(onCacheUpdate);
|
|
125
|
+
entry.refetchers.add(doFetch);
|
|
126
|
+
}
|
|
127
|
+
if (entry.promise) {
|
|
128
|
+
setIsFetching(true);
|
|
129
|
+
try {
|
|
130
|
+
await entry.promise;
|
|
131
|
+
} catch {
|
|
132
|
+
}
|
|
133
|
+
onCacheUpdate();
|
|
134
|
+
return;
|
|
135
|
+
}
|
|
136
|
+
abortController?.abort();
|
|
137
|
+
abortController = new AbortController();
|
|
138
|
+
const signal2 = abortController.signal;
|
|
139
|
+
setIsFetching(true);
|
|
140
|
+
const promise = withRetry(() => fetcher({ signal: signal2, key: key2 }), retryOptions, void 0, signal2);
|
|
141
|
+
entry.promise = promise;
|
|
142
|
+
try {
|
|
143
|
+
const result = await promise;
|
|
144
|
+
entry.promise = null;
|
|
145
|
+
if (disposed || currentKey !== key2) return;
|
|
146
|
+
entry.data = result;
|
|
147
|
+
entry.dataUpdatedAt = Date.now();
|
|
148
|
+
entry.error = void 0;
|
|
149
|
+
batch(() => {
|
|
150
|
+
setData(result);
|
|
151
|
+
setIsFetching(false);
|
|
152
|
+
setError(void 0);
|
|
153
|
+
});
|
|
154
|
+
for (const listener of entry.listeners) listener();
|
|
155
|
+
onSuccess?.(result);
|
|
156
|
+
} catch (err) {
|
|
157
|
+
entry.promise = null;
|
|
158
|
+
if (disposed || currentKey !== key2) return;
|
|
159
|
+
if (err instanceof DOMException && err.name === "AbortError") return;
|
|
160
|
+
const errorObj = err instanceof Error ? err : new Error(String(err));
|
|
161
|
+
entry.error = errorObj;
|
|
162
|
+
batch(() => {
|
|
163
|
+
setError(errorObj);
|
|
164
|
+
setIsFetching(false);
|
|
165
|
+
});
|
|
166
|
+
for (const listener of entry.listeners) listener();
|
|
167
|
+
onError?.(errorObj);
|
|
168
|
+
} finally {
|
|
169
|
+
if (!disposed && currentKey === key2) onSettled?.();
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
function onCacheUpdate() {
|
|
173
|
+
if (disposed || !currentKey) return;
|
|
174
|
+
const entry = queryCache.get(currentKey);
|
|
175
|
+
if (!entry) {
|
|
176
|
+
batch(() => {
|
|
177
|
+
setData(void 0);
|
|
178
|
+
setError(void 0);
|
|
179
|
+
setIsFetching(false);
|
|
180
|
+
});
|
|
181
|
+
return;
|
|
182
|
+
}
|
|
183
|
+
batch(() => {
|
|
184
|
+
setData(entry.data);
|
|
185
|
+
setError(entry.error);
|
|
186
|
+
if (!entry.promise) setIsFetching(false);
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
const effectCleanup = effect(() => {
|
|
190
|
+
const key2 = resolveKey();
|
|
191
|
+
if (currentKey !== null && currentKey !== key2) {
|
|
192
|
+
const oldEntry = queryCache.get(currentKey);
|
|
193
|
+
if (oldEntry) {
|
|
194
|
+
oldEntry.listeners.delete(onCacheUpdate);
|
|
195
|
+
oldEntry.refetchers.delete(doFetch);
|
|
196
|
+
oldEntry.subscribers--;
|
|
197
|
+
if (oldEntry.subscribers <= 0 && cacheTime >= 0) {
|
|
198
|
+
const oldKey = currentKey;
|
|
199
|
+
oldEntry.gcTimer = setTimeout(() => queryCache.delete(oldKey), cacheTime);
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
currentKey = key2;
|
|
204
|
+
const entry = getOrCreateEntry(key2, initialData);
|
|
205
|
+
entry.subscribers++;
|
|
206
|
+
if (entry.gcTimer !== null) {
|
|
207
|
+
clearTimeout(entry.gcTimer);
|
|
208
|
+
entry.gcTimer = null;
|
|
209
|
+
}
|
|
210
|
+
entry.listeners.add(onCacheUpdate);
|
|
211
|
+
entry.refetchers.add(doFetch);
|
|
212
|
+
if (entry.data !== void 0) {
|
|
213
|
+
batch(() => {
|
|
214
|
+
setData(entry.data);
|
|
215
|
+
setError(entry.error);
|
|
216
|
+
});
|
|
217
|
+
}
|
|
218
|
+
const isDataStale = entry.dataUpdatedAt === 0 || Date.now() - entry.dataUpdatedAt >= staleTime;
|
|
219
|
+
if (enabled && (entry.data === void 0 || isDataStale)) {
|
|
220
|
+
doFetch();
|
|
221
|
+
}
|
|
222
|
+
});
|
|
223
|
+
if (refetchInterval && refetchInterval > 0) {
|
|
224
|
+
intervalTimer = setInterval(() => {
|
|
225
|
+
if (!disposed && currentKey && enabled) doFetch();
|
|
226
|
+
}, refetchInterval);
|
|
227
|
+
}
|
|
228
|
+
let focusHandler = null;
|
|
229
|
+
let onlineHandler = null;
|
|
230
|
+
if (typeof globalThis !== "undefined" && typeof globalThis.addEventListener === "function") {
|
|
231
|
+
if (refetchOnWindowFocus) {
|
|
232
|
+
focusHandler = () => {
|
|
233
|
+
if (!disposed && currentKey && enabled) doFetch();
|
|
234
|
+
};
|
|
235
|
+
globalThis.addEventListener("focus", focusHandler);
|
|
236
|
+
}
|
|
237
|
+
if (refetchOnReconnect) {
|
|
238
|
+
onlineHandler = () => {
|
|
239
|
+
if (!disposed && currentKey && enabled) doFetch();
|
|
240
|
+
};
|
|
241
|
+
globalThis.addEventListener("online", onlineHandler);
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
function dispose() {
|
|
245
|
+
disposed = true;
|
|
246
|
+
abortController?.abort();
|
|
247
|
+
effectCleanup();
|
|
248
|
+
if (intervalTimer) clearInterval(intervalTimer);
|
|
249
|
+
if (currentKey) {
|
|
250
|
+
const entry = queryCache.get(currentKey);
|
|
251
|
+
if (entry) {
|
|
252
|
+
entry.listeners.delete(onCacheUpdate);
|
|
253
|
+
entry.refetchers.delete(doFetch);
|
|
254
|
+
entry.subscribers--;
|
|
255
|
+
if (entry.subscribers <= 0 && cacheTime >= 0) {
|
|
256
|
+
const key2 = currentKey;
|
|
257
|
+
entry.gcTimer = setTimeout(() => queryCache.delete(key2), cacheTime);
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
if (focusHandler) globalThis.removeEventListener("focus", focusHandler);
|
|
262
|
+
if (onlineHandler) globalThis.removeEventListener("online", onlineHandler);
|
|
263
|
+
}
|
|
264
|
+
return {
|
|
265
|
+
data,
|
|
266
|
+
loading,
|
|
267
|
+
fetching: isFetching,
|
|
268
|
+
error,
|
|
269
|
+
isStale,
|
|
270
|
+
refetch: doFetch,
|
|
271
|
+
dispose
|
|
272
|
+
};
|
|
273
|
+
}
|
|
274
|
+
function invalidateQueries(keyOrPredicate) {
|
|
275
|
+
const predicate = typeof keyOrPredicate === "function" ? keyOrPredicate : (k) => k === keyOrPredicate;
|
|
276
|
+
for (const [key, entry] of queryCache.entries()) {
|
|
277
|
+
if (predicate(key)) {
|
|
278
|
+
entry.dataUpdatedAt = 0;
|
|
279
|
+
for (const refetcher of entry.refetchers) refetcher();
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
function getQueryData(key) {
|
|
284
|
+
return queryCache.get(key)?.data;
|
|
285
|
+
}
|
|
286
|
+
function setQueryData(key, data) {
|
|
287
|
+
const entry = queryCache.get(key);
|
|
288
|
+
if (!entry) return;
|
|
289
|
+
const newData = typeof data === "function" ? data(entry.data) : data;
|
|
290
|
+
entry.data = newData;
|
|
291
|
+
entry.dataUpdatedAt = Date.now();
|
|
292
|
+
for (const listener of entry.listeners) listener();
|
|
293
|
+
}
|
|
294
|
+
function clearQueryCache() {
|
|
295
|
+
const activeListeners = [];
|
|
296
|
+
const activeRefetchers = [];
|
|
297
|
+
for (const entry of queryCache.values()) {
|
|
298
|
+
if (entry.gcTimer) clearTimeout(entry.gcTimer);
|
|
299
|
+
if (entry.subscribers > 0) {
|
|
300
|
+
for (const listener of entry.listeners) activeListeners.push(listener);
|
|
301
|
+
for (const refetcher of entry.refetchers) activeRefetchers.push(refetcher);
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
queryCache.clear();
|
|
305
|
+
for (const listener of activeListeners) listener();
|
|
306
|
+
for (const refetcher of activeRefetchers) refetcher();
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
// src/data/mutation.ts
|
|
310
|
+
function mutation(mutationFn, options = {}) {
|
|
311
|
+
const [data, setData] = signal(void 0);
|
|
312
|
+
const [loading, setLoading] = signal(false);
|
|
313
|
+
const [error, setError] = signal(void 0);
|
|
314
|
+
const [status, setStatus] = signal("idle");
|
|
315
|
+
const isSuccess = derived(() => status() === "success");
|
|
316
|
+
const isIdle = derived(() => status() === "idle");
|
|
317
|
+
async function execute(variables) {
|
|
318
|
+
let context2;
|
|
319
|
+
batch(() => {
|
|
320
|
+
setLoading(true);
|
|
321
|
+
setError(void 0);
|
|
322
|
+
setStatus("loading");
|
|
323
|
+
});
|
|
324
|
+
try {
|
|
325
|
+
if (options.onMutate) {
|
|
326
|
+
context2 = await options.onMutate(variables);
|
|
327
|
+
}
|
|
328
|
+
const result = await withRetry(() => mutationFn(variables), options.retry);
|
|
329
|
+
batch(() => {
|
|
330
|
+
setData(result);
|
|
331
|
+
setLoading(false);
|
|
332
|
+
setStatus("success");
|
|
333
|
+
});
|
|
334
|
+
options.onSuccess?.(result, variables, context2);
|
|
335
|
+
options.onSettled?.(result, void 0, variables, context2);
|
|
336
|
+
return result;
|
|
337
|
+
} catch (err) {
|
|
338
|
+
const errorObj = err instanceof Error ? err : new Error(String(err));
|
|
339
|
+
batch(() => {
|
|
340
|
+
setError(errorObj);
|
|
341
|
+
setLoading(false);
|
|
342
|
+
setStatus("error");
|
|
343
|
+
});
|
|
344
|
+
options.onError?.(errorObj, variables, context2);
|
|
345
|
+
options.onSettled?.(void 0, errorObj, variables, context2);
|
|
346
|
+
throw errorObj;
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
function reset() {
|
|
350
|
+
batch(() => {
|
|
351
|
+
setData(void 0);
|
|
352
|
+
setError(void 0);
|
|
353
|
+
setLoading(false);
|
|
354
|
+
setStatus("idle");
|
|
355
|
+
});
|
|
356
|
+
}
|
|
357
|
+
return {
|
|
358
|
+
data,
|
|
359
|
+
loading,
|
|
360
|
+
error,
|
|
361
|
+
isSuccess,
|
|
362
|
+
isIdle,
|
|
363
|
+
mutate: (variables) => {
|
|
364
|
+
execute(variables).catch(() => {
|
|
365
|
+
});
|
|
366
|
+
},
|
|
367
|
+
mutateAsync: execute,
|
|
368
|
+
reset
|
|
369
|
+
};
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
// src/data/infiniteQuery.ts
|
|
373
|
+
function infiniteQuery(key, fetcher, options) {
|
|
374
|
+
const {
|
|
375
|
+
getNextPageParam,
|
|
376
|
+
getPreviousPageParam,
|
|
377
|
+
initialPageParam = 0,
|
|
378
|
+
enabled = true,
|
|
379
|
+
retry: retryOptions,
|
|
380
|
+
onSuccess,
|
|
381
|
+
onError
|
|
382
|
+
} = options;
|
|
383
|
+
const resolveKey = typeof key === "function" ? key : () => key;
|
|
384
|
+
const [pages, setPages] = signal([]);
|
|
385
|
+
const [isFetching, setIsFetching] = signal(false);
|
|
386
|
+
const [isFetchingNext, setIsFetchingNext] = signal(false);
|
|
387
|
+
const [isFetchingPrev, setIsFetchingPrev] = signal(false);
|
|
388
|
+
const [error, setError] = signal(void 0);
|
|
389
|
+
const [nextPageParam, setNextPageParam] = signal(initialPageParam);
|
|
390
|
+
const [prevPageParam, setPrevPageParam] = signal(void 0);
|
|
391
|
+
const data = derived(() => {
|
|
392
|
+
const p = pages();
|
|
393
|
+
return p.length > 0 ? p : void 0;
|
|
394
|
+
});
|
|
395
|
+
const loading = derived(() => isFetching() && pages().length === 0);
|
|
396
|
+
const hasNextPage = derived(() => nextPageParam() !== void 0);
|
|
397
|
+
const hasPreviousPage = derived(() => prevPageParam() !== void 0);
|
|
398
|
+
let abortController = null;
|
|
399
|
+
let disposed = false;
|
|
400
|
+
async function fetchPage(pageParam, direction) {
|
|
401
|
+
if (disposed) return;
|
|
402
|
+
abortController?.abort();
|
|
403
|
+
abortController = new AbortController();
|
|
404
|
+
const signal2 = abortController.signal;
|
|
405
|
+
batch(() => {
|
|
406
|
+
setIsFetching(true);
|
|
407
|
+
if (direction === "next") setIsFetchingNext(true);
|
|
408
|
+
if (direction === "prev") setIsFetchingPrev(true);
|
|
409
|
+
setError(void 0);
|
|
410
|
+
});
|
|
411
|
+
try {
|
|
412
|
+
const page = await withRetry(() => fetcher({ signal: signal2, pageParam }), retryOptions, void 0, signal2);
|
|
413
|
+
if (disposed) return;
|
|
414
|
+
const currentPages = pages();
|
|
415
|
+
let newPages;
|
|
416
|
+
if (direction === "prev") {
|
|
417
|
+
newPages = [page, ...currentPages];
|
|
418
|
+
} else {
|
|
419
|
+
newPages = [...currentPages, page];
|
|
420
|
+
}
|
|
421
|
+
const nextParam = getNextPageParam(page, newPages);
|
|
422
|
+
const prevParam = getPreviousPageParam?.(newPages[0], newPages);
|
|
423
|
+
batch(() => {
|
|
424
|
+
setPages(newPages);
|
|
425
|
+
setNextPageParam(nextParam);
|
|
426
|
+
setPrevPageParam(prevParam);
|
|
427
|
+
setIsFetching(false);
|
|
428
|
+
setIsFetchingNext(false);
|
|
429
|
+
setIsFetchingPrev(false);
|
|
430
|
+
});
|
|
431
|
+
onSuccess?.(newPages);
|
|
432
|
+
} catch (err) {
|
|
433
|
+
if (disposed) return;
|
|
434
|
+
if (err instanceof DOMException && err.name === "AbortError") return;
|
|
435
|
+
const errorObj = err instanceof Error ? err : new Error(String(err));
|
|
436
|
+
batch(() => {
|
|
437
|
+
setError(errorObj);
|
|
438
|
+
setIsFetching(false);
|
|
439
|
+
setIsFetchingNext(false);
|
|
440
|
+
setIsFetchingPrev(false);
|
|
441
|
+
});
|
|
442
|
+
onError?.(errorObj);
|
|
443
|
+
}
|
|
444
|
+
}
|
|
445
|
+
const effectCleanup = effect(() => {
|
|
446
|
+
resolveKey();
|
|
447
|
+
if (enabled) {
|
|
448
|
+
setPages([]);
|
|
449
|
+
setNextPageParam(initialPageParam);
|
|
450
|
+
setPrevPageParam(void 0);
|
|
451
|
+
fetchPage(initialPageParam, "initial");
|
|
452
|
+
}
|
|
453
|
+
});
|
|
454
|
+
function fetchNextPage() {
|
|
455
|
+
const param = nextPageParam();
|
|
456
|
+
if (param === void 0) return Promise.resolve();
|
|
457
|
+
return fetchPage(param, "next");
|
|
458
|
+
}
|
|
459
|
+
function fetchPreviousPage() {
|
|
460
|
+
const param = prevPageParam();
|
|
461
|
+
if (param === void 0) return Promise.resolve();
|
|
462
|
+
return fetchPage(param, "prev");
|
|
463
|
+
}
|
|
464
|
+
async function refetch() {
|
|
465
|
+
batch(() => {
|
|
466
|
+
setPages([]);
|
|
467
|
+
setNextPageParam(initialPageParam);
|
|
468
|
+
setPrevPageParam(void 0);
|
|
469
|
+
});
|
|
470
|
+
await fetchPage(initialPageParam, "initial");
|
|
471
|
+
}
|
|
472
|
+
function dispose() {
|
|
473
|
+
disposed = true;
|
|
474
|
+
abortController?.abort();
|
|
475
|
+
effectCleanup();
|
|
476
|
+
}
|
|
477
|
+
return {
|
|
478
|
+
data,
|
|
479
|
+
pages,
|
|
480
|
+
loading,
|
|
481
|
+
fetching: isFetching,
|
|
482
|
+
fetchingNextPage: isFetchingNext,
|
|
483
|
+
fetchingPreviousPage: isFetchingPrev,
|
|
484
|
+
error,
|
|
485
|
+
hasNextPage,
|
|
486
|
+
hasPreviousPage,
|
|
487
|
+
fetchNextPage,
|
|
488
|
+
fetchPreviousPage,
|
|
489
|
+
refetch,
|
|
490
|
+
dispose
|
|
491
|
+
};
|
|
492
|
+
}
|
|
493
|
+
|
|
494
|
+
// src/data/previous.ts
|
|
495
|
+
function previous(getter) {
|
|
496
|
+
const [previous2, setPrevious] = signal(void 0);
|
|
497
|
+
let current = getter();
|
|
498
|
+
effect(() => {
|
|
499
|
+
const next = getter();
|
|
500
|
+
if (!Object.is(next, current)) {
|
|
501
|
+
setPrevious(current);
|
|
502
|
+
current = next;
|
|
503
|
+
}
|
|
504
|
+
});
|
|
505
|
+
return previous2;
|
|
506
|
+
}
|
|
507
|
+
|
|
508
|
+
// src/data/debounce.ts
|
|
509
|
+
function debounce(getter, delay) {
|
|
510
|
+
const [debounced, setDebounced] = signal(getter());
|
|
511
|
+
let timer = null;
|
|
512
|
+
effect(() => {
|
|
513
|
+
const value = getter();
|
|
514
|
+
if (timer !== null) clearTimeout(timer);
|
|
515
|
+
timer = setTimeout(() => {
|
|
516
|
+
setDebounced(value);
|
|
517
|
+
timer = null;
|
|
518
|
+
}, delay);
|
|
519
|
+
});
|
|
520
|
+
return debounced;
|
|
521
|
+
}
|
|
522
|
+
|
|
523
|
+
// src/data/throttle.ts
|
|
524
|
+
function throttle(getter, interval) {
|
|
525
|
+
const [throttled, setThrottled] = signal(getter());
|
|
526
|
+
let cooldown = false;
|
|
527
|
+
let pending = null;
|
|
528
|
+
let lastEmitted = getter();
|
|
529
|
+
effect(() => {
|
|
530
|
+
const value = getter();
|
|
531
|
+
if (!cooldown) {
|
|
532
|
+
if (!Object.is(value, lastEmitted)) {
|
|
533
|
+
setThrottled(value);
|
|
534
|
+
lastEmitted = value;
|
|
535
|
+
cooldown = true;
|
|
536
|
+
pending = null;
|
|
537
|
+
setTimeout(() => {
|
|
538
|
+
cooldown = false;
|
|
539
|
+
if (pending !== null) {
|
|
540
|
+
const trailingValue = pending.value;
|
|
541
|
+
pending = null;
|
|
542
|
+
setThrottled(trailingValue);
|
|
543
|
+
lastEmitted = trailingValue;
|
|
544
|
+
}
|
|
545
|
+
}, interval);
|
|
546
|
+
}
|
|
547
|
+
} else {
|
|
548
|
+
pending = { value };
|
|
549
|
+
}
|
|
550
|
+
});
|
|
551
|
+
return throttled;
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
// src/data/resource.ts
|
|
555
|
+
function resource(sourceOrFetcher, fetcherOrOptions, maybeOptions) {
|
|
556
|
+
let source = null;
|
|
557
|
+
let fetcher;
|
|
558
|
+
let options;
|
|
559
|
+
if (typeof fetcherOrOptions === "function") {
|
|
560
|
+
source = sourceOrFetcher;
|
|
561
|
+
fetcher = fetcherOrOptions;
|
|
562
|
+
options = maybeOptions ?? {};
|
|
563
|
+
} else {
|
|
564
|
+
const rawFetcher = sourceOrFetcher;
|
|
565
|
+
fetcher = (_source, info) => rawFetcher(info);
|
|
566
|
+
options = fetcherOrOptions ?? {};
|
|
567
|
+
}
|
|
568
|
+
const [data, setData] = signal(options.initialValue);
|
|
569
|
+
const [loading, setLoading] = signal(false);
|
|
570
|
+
const [error, setError] = signal(void 0);
|
|
571
|
+
let currentData = options.initialValue;
|
|
572
|
+
let abortController = null;
|
|
573
|
+
let disposed = false;
|
|
574
|
+
let effectCleanup = null;
|
|
575
|
+
let fetchVersion = 0;
|
|
576
|
+
async function doFetch(sourceValue) {
|
|
577
|
+
if (disposed) return;
|
|
578
|
+
abortController?.abort();
|
|
579
|
+
abortController = new AbortController();
|
|
580
|
+
const version = ++fetchVersion;
|
|
581
|
+
const signal2 = abortController.signal;
|
|
582
|
+
const prev = currentData;
|
|
583
|
+
batch(() => {
|
|
584
|
+
setLoading(true);
|
|
585
|
+
setError(void 0);
|
|
586
|
+
});
|
|
587
|
+
options.onStart?.();
|
|
588
|
+
try {
|
|
589
|
+
const result = await withRetry(() => fetcher(sourceValue, { signal: signal2, prev }), options.retry, void 0, signal2);
|
|
590
|
+
if (version !== fetchVersion || disposed) return;
|
|
591
|
+
currentData = result;
|
|
592
|
+
batch(() => {
|
|
593
|
+
setData(result);
|
|
594
|
+
setLoading(false);
|
|
595
|
+
});
|
|
596
|
+
options.onSuccess?.(result);
|
|
597
|
+
} catch (err) {
|
|
598
|
+
if (version !== fetchVersion || disposed) return;
|
|
599
|
+
if (err instanceof DOMException && err.name === "AbortError") return;
|
|
600
|
+
const errorObj = err instanceof Error ? err : new Error(String(err));
|
|
601
|
+
batch(() => {
|
|
602
|
+
setError(errorObj);
|
|
603
|
+
setLoading(false);
|
|
604
|
+
});
|
|
605
|
+
options.onError?.(errorObj);
|
|
606
|
+
} finally {
|
|
607
|
+
if (version === fetchVersion) {
|
|
608
|
+
options.onSettled?.();
|
|
609
|
+
}
|
|
610
|
+
}
|
|
611
|
+
}
|
|
612
|
+
if (source) {
|
|
613
|
+
effectCleanup = effect(() => {
|
|
614
|
+
const sourceValue = source();
|
|
615
|
+
doFetch(sourceValue);
|
|
616
|
+
});
|
|
617
|
+
} else if (options.immediate !== false) {
|
|
618
|
+
doFetch(void 0);
|
|
619
|
+
}
|
|
620
|
+
return {
|
|
621
|
+
data,
|
|
622
|
+
loading,
|
|
623
|
+
error,
|
|
624
|
+
refetch: () => doFetch(source ? source() : void 0),
|
|
625
|
+
mutate: (value) => {
|
|
626
|
+
const newValue = typeof value === "function" ? value(currentData) : value;
|
|
627
|
+
currentData = newValue;
|
|
628
|
+
setData(newValue);
|
|
629
|
+
},
|
|
630
|
+
abort: () => abortController?.abort(),
|
|
631
|
+
dispose: () => {
|
|
632
|
+
disposed = true;
|
|
633
|
+
abortController?.abort();
|
|
634
|
+
effectCleanup?.();
|
|
635
|
+
}
|
|
636
|
+
};
|
|
637
|
+
}
|
|
638
|
+
|
|
639
|
+
// src/data/offlineStore.ts
|
|
640
|
+
function openDB(name, version, keyPath) {
|
|
641
|
+
return new Promise((resolve, reject) => {
|
|
642
|
+
const request = indexedDB.open(name, version);
|
|
643
|
+
request.onerror = () => reject(request.error);
|
|
644
|
+
request.onsuccess = () => resolve(request.result);
|
|
645
|
+
request.onupgradeneeded = () => {
|
|
646
|
+
const db = request.result;
|
|
647
|
+
if (!db.objectStoreNames.contains("items")) {
|
|
648
|
+
db.createObjectStore("items", { keyPath });
|
|
649
|
+
}
|
|
650
|
+
if (!db.objectStoreNames.contains("_changes")) {
|
|
651
|
+
const changeStore = db.createObjectStore("_changes", { autoIncrement: true });
|
|
652
|
+
changeStore.createIndex("timestamp", "timestamp");
|
|
653
|
+
}
|
|
654
|
+
if (!db.objectStoreNames.contains("_meta")) {
|
|
655
|
+
db.createObjectStore("_meta");
|
|
656
|
+
}
|
|
657
|
+
};
|
|
658
|
+
});
|
|
659
|
+
}
|
|
660
|
+
function idbGetAll(db, store) {
|
|
661
|
+
return new Promise((resolve, reject) => {
|
|
662
|
+
const tx = db.transaction(store, "readonly");
|
|
663
|
+
const req = tx.objectStore(store).getAll();
|
|
664
|
+
req.onsuccess = () => resolve(req.result);
|
|
665
|
+
req.onerror = () => reject(req.error);
|
|
666
|
+
});
|
|
667
|
+
}
|
|
668
|
+
function idbGet(db, store, key) {
|
|
669
|
+
return new Promise((resolve, reject) => {
|
|
670
|
+
const tx = db.transaction(store, "readonly");
|
|
671
|
+
const req = tx.objectStore(store).get(key);
|
|
672
|
+
req.onsuccess = () => resolve(req.result);
|
|
673
|
+
req.onerror = () => reject(req.error);
|
|
674
|
+
});
|
|
675
|
+
}
|
|
676
|
+
function idbPut(db, store, item) {
|
|
677
|
+
return new Promise((resolve, reject) => {
|
|
678
|
+
const tx = db.transaction(store, "readwrite");
|
|
679
|
+
tx.objectStore(store).put(item);
|
|
680
|
+
tx.oncomplete = () => resolve();
|
|
681
|
+
tx.onerror = () => reject(tx.error);
|
|
682
|
+
});
|
|
683
|
+
}
|
|
684
|
+
function idbDelete(db, store, key) {
|
|
685
|
+
return new Promise((resolve, reject) => {
|
|
686
|
+
const tx = db.transaction(store, "readwrite");
|
|
687
|
+
tx.objectStore(store).delete(key);
|
|
688
|
+
tx.oncomplete = () => resolve();
|
|
689
|
+
tx.onerror = () => reject(tx.error);
|
|
690
|
+
});
|
|
691
|
+
}
|
|
692
|
+
function idbClear(db, store) {
|
|
693
|
+
return new Promise((resolve, reject) => {
|
|
694
|
+
const tx = db.transaction(store, "readwrite");
|
|
695
|
+
tx.objectStore(store).clear();
|
|
696
|
+
tx.oncomplete = () => resolve();
|
|
697
|
+
tx.onerror = () => reject(tx.error);
|
|
698
|
+
});
|
|
699
|
+
}
|
|
700
|
+
async function offlineStore(options) {
|
|
701
|
+
const { name, version = 1, keyPath = "id", autoSync = true } = options;
|
|
702
|
+
const db = await openDB(name, version, keyPath);
|
|
703
|
+
const initialData = await idbGetAll(db, "items");
|
|
704
|
+
const initialChanges = await idbGetAll(db, "_changes");
|
|
705
|
+
const savedLastSync = await idbGet(db, "_meta", "lastSynced");
|
|
706
|
+
const [data, setData] = signal(initialData);
|
|
707
|
+
const [isSyncing, setIsSyncing] = signal(false);
|
|
708
|
+
const [lastSynced, setLastSynced] = signal(savedLastSync ?? null);
|
|
709
|
+
const [pendingCount, setPendingCount] = signal(initialChanges.length);
|
|
710
|
+
let adapter = options.adapter;
|
|
711
|
+
async function refreshData() {
|
|
712
|
+
const items = await idbGetAll(db, "items");
|
|
713
|
+
setData(items);
|
|
714
|
+
const changes = await idbGetAll(db, "_changes");
|
|
715
|
+
setPendingCount(changes.length);
|
|
716
|
+
}
|
|
717
|
+
async function put(item) {
|
|
718
|
+
await idbPut(db, "items", item);
|
|
719
|
+
await idbPut(db, "_changes", { type: "put", item, timestamp: Date.now() });
|
|
720
|
+
await refreshData();
|
|
721
|
+
}
|
|
722
|
+
async function remove(key) {
|
|
723
|
+
const existing = await idbGet(db, "items", key);
|
|
724
|
+
if (existing) {
|
|
725
|
+
await idbDelete(db, "items", key);
|
|
726
|
+
await idbPut(db, "_changes", { type: "delete", item: existing, timestamp: Date.now() });
|
|
727
|
+
await refreshData();
|
|
728
|
+
}
|
|
729
|
+
}
|
|
730
|
+
async function get(key) {
|
|
731
|
+
return idbGet(db, "items", key);
|
|
732
|
+
}
|
|
733
|
+
function query2(filter) {
|
|
734
|
+
return data().filter(filter);
|
|
735
|
+
}
|
|
736
|
+
async function sync() {
|
|
737
|
+
if (!adapter || isSyncing()) return;
|
|
738
|
+
setIsSyncing(true);
|
|
739
|
+
try {
|
|
740
|
+
const changes = await idbGetAll(db, "_changes");
|
|
741
|
+
if (changes.length > 0) {
|
|
742
|
+
const result = await adapter.push(changes);
|
|
743
|
+
if (result.ok) {
|
|
744
|
+
await idbClear(db, "_changes");
|
|
745
|
+
}
|
|
746
|
+
}
|
|
747
|
+
const remoteItems = await adapter.pull(lastSynced());
|
|
748
|
+
for (const item of remoteItems) {
|
|
749
|
+
await idbPut(db, "items", item);
|
|
750
|
+
}
|
|
751
|
+
const now = Date.now();
|
|
752
|
+
await idbPut(db, "_meta", now);
|
|
753
|
+
setLastSynced(now);
|
|
754
|
+
await refreshData();
|
|
755
|
+
} catch {
|
|
756
|
+
} finally {
|
|
757
|
+
setIsSyncing(false);
|
|
758
|
+
}
|
|
759
|
+
}
|
|
760
|
+
function attach(newAdapter) {
|
|
761
|
+
adapter = newAdapter;
|
|
762
|
+
}
|
|
763
|
+
function close() {
|
|
764
|
+
db.close();
|
|
765
|
+
}
|
|
766
|
+
if (autoSync && typeof window !== "undefined") {
|
|
767
|
+
window.addEventListener("online", () => {
|
|
768
|
+
sync();
|
|
769
|
+
});
|
|
770
|
+
}
|
|
771
|
+
return {
|
|
772
|
+
data,
|
|
773
|
+
get,
|
|
774
|
+
put,
|
|
775
|
+
remove,
|
|
776
|
+
query: query2,
|
|
777
|
+
isSyncing,
|
|
778
|
+
lastSynced,
|
|
779
|
+
sync,
|
|
780
|
+
attach,
|
|
781
|
+
pendingCount,
|
|
782
|
+
close
|
|
783
|
+
};
|
|
784
|
+
}
|
|
785
|
+
function syncAdapter(config) {
|
|
786
|
+
return config;
|
|
787
|
+
}
|
|
788
|
+
|
|
789
|
+
// src/data/routeLoader.ts
|
|
790
|
+
var LoaderContext = context(null);
|
|
791
|
+
function executeLoader(loader, context2, options) {
|
|
792
|
+
const res = resource(({ signal: signal2 }) => loader(context2, { signal: signal2 }), options);
|
|
793
|
+
LoaderContext.provide(res);
|
|
794
|
+
return res;
|
|
795
|
+
}
|
|
796
|
+
function loaderData() {
|
|
797
|
+
const resource2 = LoaderContext.get();
|
|
798
|
+
if (!resource2) {
|
|
799
|
+
throw new Error("loaderData must be used inside a route with a loader");
|
|
800
|
+
}
|
|
801
|
+
return {
|
|
802
|
+
data: resource2.data,
|
|
803
|
+
loading: resource2.loading,
|
|
804
|
+
error: resource2.error
|
|
805
|
+
};
|
|
806
|
+
}
|
|
807
|
+
async function preloadRoute(route, context2) {
|
|
808
|
+
if (!route.loader) return void 0;
|
|
809
|
+
const controller = new AbortController();
|
|
810
|
+
return route.loader(context2, { signal: controller.signal });
|
|
811
|
+
}
|
|
812
|
+
|
|
813
|
+
// src/ui/socket.ts
|
|
814
|
+
function socket(url, options) {
|
|
815
|
+
const autoReconnect = options?.autoReconnect ?? false;
|
|
816
|
+
const reconnectDelay = options?.reconnectDelay ?? 1e3;
|
|
817
|
+
const maxReconnects = options?.maxReconnects ?? Infinity;
|
|
818
|
+
const heartbeat = options?.heartbeat;
|
|
819
|
+
const protocols = options?.protocols;
|
|
820
|
+
const [data, setData] = signal(null);
|
|
821
|
+
const [status, setStatus] = signal("closed");
|
|
822
|
+
let ws = null;
|
|
823
|
+
let reconnectCount = 0;
|
|
824
|
+
let reconnectTimer = null;
|
|
825
|
+
let heartbeatTimer = null;
|
|
826
|
+
let disposed = false;
|
|
827
|
+
function getUrl() {
|
|
828
|
+
return typeof url === "function" ? url() : url;
|
|
829
|
+
}
|
|
830
|
+
function connect() {
|
|
831
|
+
if (disposed) return;
|
|
832
|
+
setStatus("connecting");
|
|
833
|
+
ws = new WebSocket(getUrl(), protocols);
|
|
834
|
+
ws.onopen = () => {
|
|
835
|
+
setStatus("open");
|
|
836
|
+
reconnectCount = 0;
|
|
837
|
+
startHeartbeat();
|
|
838
|
+
};
|
|
839
|
+
ws.onmessage = (event) => {
|
|
840
|
+
setData(event.data);
|
|
841
|
+
};
|
|
842
|
+
ws.onclose = () => {
|
|
843
|
+
setStatus("closed");
|
|
844
|
+
stopHeartbeat();
|
|
845
|
+
if (autoReconnect && !disposed && reconnectCount < maxReconnects) {
|
|
846
|
+
reconnectCount++;
|
|
847
|
+
reconnectTimer = setTimeout(() => {
|
|
848
|
+
connect();
|
|
849
|
+
}, reconnectDelay);
|
|
850
|
+
}
|
|
851
|
+
};
|
|
852
|
+
ws.onerror = () => {
|
|
853
|
+
};
|
|
854
|
+
}
|
|
855
|
+
function startHeartbeat() {
|
|
856
|
+
if (!heartbeat) return;
|
|
857
|
+
stopHeartbeat();
|
|
858
|
+
heartbeatTimer = setInterval(() => {
|
|
859
|
+
if (ws && ws.readyState === WebSocket.OPEN) {
|
|
860
|
+
ws.send(heartbeat.message);
|
|
861
|
+
}
|
|
862
|
+
}, heartbeat.interval);
|
|
863
|
+
}
|
|
864
|
+
function stopHeartbeat() {
|
|
865
|
+
if (heartbeatTimer !== null) {
|
|
866
|
+
clearInterval(heartbeatTimer);
|
|
867
|
+
heartbeatTimer = null;
|
|
868
|
+
}
|
|
869
|
+
}
|
|
870
|
+
function send(msg) {
|
|
871
|
+
if (ws && ws.readyState === WebSocket.OPEN) {
|
|
872
|
+
ws.send(msg);
|
|
873
|
+
}
|
|
874
|
+
}
|
|
875
|
+
function close() {
|
|
876
|
+
if (reconnectTimer !== null) {
|
|
877
|
+
clearTimeout(reconnectTimer);
|
|
878
|
+
reconnectTimer = null;
|
|
879
|
+
}
|
|
880
|
+
stopHeartbeat();
|
|
881
|
+
if (ws) {
|
|
882
|
+
setStatus("closing");
|
|
883
|
+
ws.close();
|
|
884
|
+
}
|
|
885
|
+
}
|
|
886
|
+
function dispose() {
|
|
887
|
+
disposed = true;
|
|
888
|
+
close();
|
|
889
|
+
}
|
|
890
|
+
connect();
|
|
891
|
+
return { data, status, send, close, dispose };
|
|
892
|
+
}
|
|
893
|
+
|
|
894
|
+
// src/ui/stream.ts
|
|
895
|
+
function stream(url, options) {
|
|
896
|
+
const autoReconnect = options?.autoReconnect ?? false;
|
|
897
|
+
const [data, setData] = signal(null);
|
|
898
|
+
const [event, setEvent] = signal(null);
|
|
899
|
+
const [status, setStatus] = signal("connecting");
|
|
900
|
+
let source = null;
|
|
901
|
+
let disposed = false;
|
|
902
|
+
let reconnectTimer = null;
|
|
903
|
+
function connect() {
|
|
904
|
+
if (disposed) return;
|
|
905
|
+
setStatus("connecting");
|
|
906
|
+
source = new EventSource(url, {
|
|
907
|
+
withCredentials: options?.withCredentials ?? false
|
|
908
|
+
});
|
|
909
|
+
source.onopen = () => {
|
|
910
|
+
setStatus("open");
|
|
911
|
+
};
|
|
912
|
+
source.onmessage = (evt) => {
|
|
913
|
+
setData(evt.data);
|
|
914
|
+
setEvent(evt.type);
|
|
915
|
+
};
|
|
916
|
+
source.onerror = () => {
|
|
917
|
+
if (source && source.readyState === EventSource.CLOSED) {
|
|
918
|
+
setStatus("closed");
|
|
919
|
+
source = null;
|
|
920
|
+
if (autoReconnect && !disposed) {
|
|
921
|
+
reconnectTimer = setTimeout(() => {
|
|
922
|
+
reconnectTimer = null;
|
|
923
|
+
connect();
|
|
924
|
+
}, 1e3);
|
|
925
|
+
}
|
|
926
|
+
}
|
|
927
|
+
};
|
|
928
|
+
}
|
|
929
|
+
function close() {
|
|
930
|
+
if (reconnectTimer !== null) {
|
|
931
|
+
clearTimeout(reconnectTimer);
|
|
932
|
+
reconnectTimer = null;
|
|
933
|
+
}
|
|
934
|
+
if (source) {
|
|
935
|
+
source.close();
|
|
936
|
+
setStatus("closed");
|
|
937
|
+
source = null;
|
|
938
|
+
}
|
|
939
|
+
}
|
|
940
|
+
function dispose() {
|
|
941
|
+
disposed = true;
|
|
942
|
+
close();
|
|
943
|
+
}
|
|
944
|
+
connect();
|
|
945
|
+
return { data, event, status, close, dispose };
|
|
946
|
+
}
|
|
947
|
+
|
|
948
|
+
export {
|
|
949
|
+
calculateDelay,
|
|
950
|
+
withRetry,
|
|
951
|
+
query,
|
|
952
|
+
invalidateQueries,
|
|
953
|
+
getQueryData,
|
|
954
|
+
setQueryData,
|
|
955
|
+
clearQueryCache,
|
|
956
|
+
mutation,
|
|
957
|
+
infiniteQuery,
|
|
958
|
+
previous,
|
|
959
|
+
debounce,
|
|
960
|
+
throttle,
|
|
961
|
+
resource,
|
|
962
|
+
offlineStore,
|
|
963
|
+
syncAdapter,
|
|
964
|
+
executeLoader,
|
|
965
|
+
loaderData,
|
|
966
|
+
preloadRoute,
|
|
967
|
+
socket,
|
|
968
|
+
stream
|
|
969
|
+
};
|
package/dist/data.cjs
CHANGED
|
@@ -546,8 +546,13 @@ function query(key, fetcher, options = {}) {
|
|
|
546
546
|
async function doFetch() {
|
|
547
547
|
if (disposed || !currentKey || !enabled) return;
|
|
548
548
|
const key2 = currentKey;
|
|
549
|
-
|
|
550
|
-
if (!entry)
|
|
549
|
+
let entry = queryCache.get(key2);
|
|
550
|
+
if (!entry) {
|
|
551
|
+
entry = getOrCreateEntry(key2);
|
|
552
|
+
entry.subscribers++;
|
|
553
|
+
entry.listeners.add(onCacheUpdate);
|
|
554
|
+
entry.refetchers.add(doFetch);
|
|
555
|
+
}
|
|
551
556
|
if (entry.promise) {
|
|
552
557
|
setIsFetching(true);
|
|
553
558
|
try {
|
|
@@ -596,9 +601,16 @@ function query(key, fetcher, options = {}) {
|
|
|
596
601
|
function onCacheUpdate() {
|
|
597
602
|
if (disposed || !currentKey) return;
|
|
598
603
|
const entry = queryCache.get(currentKey);
|
|
599
|
-
if (!entry)
|
|
604
|
+
if (!entry) {
|
|
605
|
+
batch(() => {
|
|
606
|
+
setData(void 0);
|
|
607
|
+
setError(void 0);
|
|
608
|
+
setIsFetching(false);
|
|
609
|
+
});
|
|
610
|
+
return;
|
|
611
|
+
}
|
|
600
612
|
batch(() => {
|
|
601
|
-
|
|
613
|
+
setData(entry.data);
|
|
602
614
|
setError(entry.error);
|
|
603
615
|
if (!entry.promise) setIsFetching(false);
|
|
604
616
|
});
|
|
@@ -709,10 +721,18 @@ function setQueryData(key, data) {
|
|
|
709
721
|
for (const listener of entry.listeners) listener();
|
|
710
722
|
}
|
|
711
723
|
function clearQueryCache() {
|
|
724
|
+
const activeListeners = [];
|
|
725
|
+
const activeRefetchers = [];
|
|
712
726
|
for (const entry of queryCache.values()) {
|
|
713
727
|
if (entry.gcTimer) clearTimeout(entry.gcTimer);
|
|
728
|
+
if (entry.subscribers > 0) {
|
|
729
|
+
for (const listener of entry.listeners) activeListeners.push(listener);
|
|
730
|
+
for (const refetcher of entry.refetchers) activeRefetchers.push(refetcher);
|
|
731
|
+
}
|
|
714
732
|
}
|
|
715
733
|
queryCache.clear();
|
|
734
|
+
for (const listener of activeListeners) listener();
|
|
735
|
+
for (const refetcher of activeRefetchers) refetcher();
|
|
716
736
|
}
|
|
717
737
|
|
|
718
738
|
// src/data/mutation.ts
|
package/dist/data.js
CHANGED
package/dist/extras.cjs
CHANGED
|
@@ -789,8 +789,13 @@ function query(key, fetcher, options = {}) {
|
|
|
789
789
|
async function doFetch() {
|
|
790
790
|
if (disposed || !currentKey || !enabled) return;
|
|
791
791
|
const key2 = currentKey;
|
|
792
|
-
|
|
793
|
-
if (!entry)
|
|
792
|
+
let entry = queryCache.get(key2);
|
|
793
|
+
if (!entry) {
|
|
794
|
+
entry = getOrCreateEntry(key2);
|
|
795
|
+
entry.subscribers++;
|
|
796
|
+
entry.listeners.add(onCacheUpdate);
|
|
797
|
+
entry.refetchers.add(doFetch);
|
|
798
|
+
}
|
|
794
799
|
if (entry.promise) {
|
|
795
800
|
setIsFetching(true);
|
|
796
801
|
try {
|
|
@@ -839,9 +844,16 @@ function query(key, fetcher, options = {}) {
|
|
|
839
844
|
function onCacheUpdate() {
|
|
840
845
|
if (disposed || !currentKey) return;
|
|
841
846
|
const entry = queryCache.get(currentKey);
|
|
842
|
-
if (!entry)
|
|
847
|
+
if (!entry) {
|
|
848
|
+
batch(() => {
|
|
849
|
+
setData(void 0);
|
|
850
|
+
setError(void 0);
|
|
851
|
+
setIsFetching(false);
|
|
852
|
+
});
|
|
853
|
+
return;
|
|
854
|
+
}
|
|
843
855
|
batch(() => {
|
|
844
|
-
|
|
856
|
+
setData(entry.data);
|
|
845
857
|
setError(entry.error);
|
|
846
858
|
if (!entry.promise) setIsFetching(false);
|
|
847
859
|
});
|
|
@@ -952,10 +964,18 @@ function setQueryData(key, data2) {
|
|
|
952
964
|
for (const listener of entry.listeners) listener();
|
|
953
965
|
}
|
|
954
966
|
function clearQueryCache() {
|
|
967
|
+
const activeListeners = [];
|
|
968
|
+
const activeRefetchers = [];
|
|
955
969
|
for (const entry of queryCache.values()) {
|
|
956
970
|
if (entry.gcTimer) clearTimeout(entry.gcTimer);
|
|
971
|
+
if (entry.subscribers > 0) {
|
|
972
|
+
for (const listener of entry.listeners) activeListeners.push(listener);
|
|
973
|
+
for (const refetcher of entry.refetchers) activeRefetchers.push(refetcher);
|
|
974
|
+
}
|
|
957
975
|
}
|
|
958
976
|
queryCache.clear();
|
|
977
|
+
for (const listener of activeListeners) listener();
|
|
978
|
+
for (const refetcher of activeRefetchers) refetcher();
|
|
959
979
|
}
|
|
960
980
|
|
|
961
981
|
// src/data/mutation.ts
|
package/dist/extras.js
CHANGED
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "sibujs",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.2",
|
|
4
4
|
"description": "A lightweight, function-based frontend framework that combines the best of React, Svelte, and Vue — with zero VDOM and maximum simplicity. Designed for developers who want fine-grained reactivity and full control without compilation or magic.",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"frontend",
|