@qiaopeng/tanstack-query-plus 0.2.15 → 0.3.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/README.md +1 -0
- package/dist/utils/index.d.ts +1 -1
- package/dist/utils/index.d.ts.map +1 -1
- package/dist/utils/index.js +1 -1
- package/dist/utils/placeholderData.d.ts +1 -0
- package/dist/utils/placeholderData.d.ts.map +1 -1
- package/dist/utils/placeholderData.js +27 -0
- package/package.json +5 -1
- package/dist/types/consistency.d.ts +0 -13
- package/dist/types/consistency.d.ts.map +0 -1
- package/dist/types/consistency.js +0 -1
- package/dist/utils/invalidationManager.d.ts +0 -4
- package/dist/utils/invalidationManager.d.ts.map +0 -1
- package/dist/utils/invalidationManager.js +0 -28
package/README.md
CHANGED
|
@@ -1088,6 +1088,7 @@ function useUpdateProduct({ page, pageSize }) {
|
|
|
1088
1088
|
|
|
1089
1089
|
适用操作与行为说明:
|
|
1090
1090
|
- 编辑/删除:在 `sync+invalidate` 模式下,会对已缓存的家族变体按 `id` 合并或移除;随后统一失效,最终以服务端为准
|
|
1091
|
+
- **自动竞态保护**:当检测到 `consistency` 模式开启时,会自动取消该列表家族下所有正在进行的旧请求,防止旧数据覆盖新缓存
|
|
1091
1092
|
- 新增/状态变更:默认不做跨页注入,仅当前页处理并家族失效;需要跨页放置时请在服务端裁决归属
|
|
1092
1093
|
|
|
1093
1094
|
现在你已经掌握了数据变更和乐观更新。接下来,让我们学习如何处理无限滚动和分页场景。
|
package/dist/utils/index.d.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
export { createFieldEnricher, createOptimisticBase, createTempId, type FieldMappingConfig } from "./fieldMapper.js";
|
|
2
2
|
export { getNetworkInfo, getNetworkSpeed, isFastNetwork, isSlowNetwork, type NavigatorWithConnection } from "./network.js";
|
|
3
3
|
export { batchRemoveItems, batchUpdateItems, conditionalUpdateItems, createAddItemConfig, createListOperationConfig, createRemoveItemConfig, createUpdateItemConfig, type ListOperationVariables, listUpdater, reorderItems } from "./optimisticUtils.js";
|
|
4
|
-
export { keepPreviousData } from "./placeholderData.js";
|
|
4
|
+
export { keepPreviousData, stableListPlaceholder } from "./placeholderData.js";
|
|
5
5
|
export { getPrefetchManager, type InteractionRecord, type NetworkSpeed, type PredictionResult, type PrefetchConfig, type PrefetchStats, type PrefetchTask, resetPrefetchManager, SmartPrefetchManager } from "./prefetchManager.js";
|
|
6
6
|
export { createQueryKeyFactory, createSimpleQueryKeyFactory, extractParamsFromKey, isQueryKeyEqual, type NormalizeConfig, normalizeQueryParams, type QueryKeyFactory, type QueryKeyFactoryConfig } from "./queryKey.js";
|
|
7
7
|
export { compose, selectById, selectByIds, selectCount, selectField, selectFields, selectFirst, selectItems, selectLast, selectMap, selectors, selectTotal, selectWhere } from "./selectors.js";
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/utils/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,oBAAoB,EAAE,YAAY,EAAE,KAAK,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AACpH,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,aAAa,EAAE,aAAa,EAAE,KAAK,uBAAuB,EAAE,MAAM,cAAc,CAAC;AAC3H,OAAO,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,yBAAyB,EAAE,sBAAsB,EAAE,sBAAsB,EAAE,KAAK,sBAAsB,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AAC1P,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/utils/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,oBAAoB,EAAE,YAAY,EAAE,KAAK,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AACpH,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,aAAa,EAAE,aAAa,EAAE,KAAK,uBAAuB,EAAE,MAAM,cAAc,CAAC;AAC3H,OAAO,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,yBAAyB,EAAE,sBAAsB,EAAE,sBAAsB,EAAE,KAAK,sBAAsB,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AAC1P,OAAO,EAAE,gBAAgB,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAC/E,OAAO,EAAE,kBAAkB,EAAE,KAAK,iBAAiB,EAAE,KAAK,YAAY,EAAE,KAAK,gBAAgB,EAAE,KAAK,cAAc,EAAE,KAAK,aAAa,EAAE,KAAK,YAAY,EAAE,oBAAoB,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AACpO,OAAO,EAAE,qBAAqB,EAAE,2BAA2B,EAAE,oBAAoB,EAAE,eAAe,EAAE,KAAK,eAAe,EAAE,oBAAoB,EAAE,KAAK,eAAe,EAAE,KAAK,qBAAqB,EAAE,MAAM,eAAe,CAAC;AACxN,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,WAAW,EAAE,WAAW,EAAE,WAAW,EAAE,YAAY,EAAE,WAAW,EAAE,WAAW,EAAE,UAAU,EAAE,SAAS,EAAE,SAAS,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAChM,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,eAAe,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAC3F,OAAO,EAAE,mBAAmB,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,KAAK,gBAAgB,EAAE,MAAM,kBAAkB,CAAC"}
|
package/dist/utils/index.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
export { createFieldEnricher, createOptimisticBase, createTempId } from "./fieldMapper.js";
|
|
2
2
|
export { getNetworkInfo, getNetworkSpeed, isFastNetwork, isSlowNetwork } from "./network.js";
|
|
3
3
|
export { batchRemoveItems, batchUpdateItems, conditionalUpdateItems, createAddItemConfig, createListOperationConfig, createRemoveItemConfig, createUpdateItemConfig, listUpdater, reorderItems } from "./optimisticUtils.js";
|
|
4
|
-
export { keepPreviousData } from "./placeholderData.js";
|
|
4
|
+
export { keepPreviousData, stableListPlaceholder } from "./placeholderData.js";
|
|
5
5
|
export { getPrefetchManager, resetPrefetchManager, SmartPrefetchManager } from "./prefetchManager.js";
|
|
6
6
|
export { createQueryKeyFactory, createSimpleQueryKeyFactory, extractParamsFromKey, isQueryKeyEqual, normalizeQueryParams } from "./queryKey.js";
|
|
7
7
|
export { compose, selectById, selectByIds, selectCount, selectField, selectFields, selectFirst, selectItems, selectLast, selectMap, selectors, selectTotal, selectWhere } from "./selectors.js";
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"placeholderData.d.ts","sourceRoot":"","sources":["../../src/utils/placeholderData.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC"}
|
|
1
|
+
{"version":3,"file":"placeholderData.d.ts","sourceRoot":"","sources":["../../src/utils/placeholderData.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AAEzD,wBAAgB,qBAAqB,CAAC,CAAC,GAAG,GAAG,EAAE,YAAY,EAAE,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,SAAS,CAuBzF"}
|
|
@@ -1 +1,28 @@
|
|
|
1
1
|
export { keepPreviousData } from "@tanstack/react-query";
|
|
2
|
+
export function stableListPlaceholder(previousData) {
|
|
3
|
+
if (previousData === undefined || previousData === null)
|
|
4
|
+
return previousData;
|
|
5
|
+
if (Array.isArray(previousData)) {
|
|
6
|
+
return [];
|
|
7
|
+
}
|
|
8
|
+
if (typeof previousData === "object") {
|
|
9
|
+
const obj = { ...previousData };
|
|
10
|
+
if (Array.isArray(obj.Rows)) {
|
|
11
|
+
obj.Rows = [];
|
|
12
|
+
if (typeof obj.Total === "number")
|
|
13
|
+
obj.Total = obj.Total;
|
|
14
|
+
return obj;
|
|
15
|
+
}
|
|
16
|
+
if (Array.isArray(obj.items)) {
|
|
17
|
+
obj.items = [];
|
|
18
|
+
if (typeof obj.total === "number")
|
|
19
|
+
obj.total = obj.total;
|
|
20
|
+
return obj;
|
|
21
|
+
}
|
|
22
|
+
if (Array.isArray(obj.data)) {
|
|
23
|
+
obj.data = [];
|
|
24
|
+
return obj;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
return previousData;
|
|
28
|
+
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@qiaopeng/tanstack-query-plus",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.3.1",
|
|
4
4
|
"description": "Enhanced TanStack Query toolkit: defaults, hooks, persistence, offline, utils",
|
|
5
5
|
"author": "qiaopeng",
|
|
6
6
|
"license": "MIT",
|
|
@@ -95,10 +95,14 @@
|
|
|
95
95
|
}
|
|
96
96
|
},
|
|
97
97
|
"devDependencies": {
|
|
98
|
+
"@tanstack/react-query": "^5",
|
|
99
|
+
"@tanstack/react-query-persist-client": "^5",
|
|
98
100
|
"@tanstack/react-query-devtools": "^5",
|
|
99
101
|
"@types/node": "^20.12.7",
|
|
100
102
|
"@types/react": "^18.3.2",
|
|
101
103
|
"@types/react-dom": "^18.3.0",
|
|
104
|
+
"react": ">=18",
|
|
105
|
+
"react-dom": ">=18",
|
|
102
106
|
"react-intersection-observer": "^9",
|
|
103
107
|
"tsup": "^8.0.0",
|
|
104
108
|
"typescript": "^5.4.0"
|
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
import type { QueryKey } from "@tanstack/react-query";
|
|
2
|
-
export type ConsistencyMode = "invalidate" | "sync+invalidate";
|
|
3
|
-
export interface FamilyConsistencyOptions {
|
|
4
|
-
baseKey: QueryKey;
|
|
5
|
-
mode?: ConsistencyMode;
|
|
6
|
-
idField?: string;
|
|
7
|
-
listSelector?: (data: unknown) => unknown[] | {
|
|
8
|
-
items: unknown[];
|
|
9
|
-
total?: number;
|
|
10
|
-
} | null;
|
|
11
|
-
maxKeys?: number;
|
|
12
|
-
}
|
|
13
|
-
//# sourceMappingURL=consistency.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"consistency.d.ts","sourceRoot":"","sources":["../../src/types/consistency.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AACtD,MAAM,MAAM,eAAe,GAAG,YAAY,GAAG,iBAAiB,CAAC;AAC/D,MAAM,WAAW,wBAAwB;IACvC,OAAO,EAAE,QAAQ,CAAC;IAClB,IAAI,CAAC,EAAE,eAAe,CAAC;IACvB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,YAAY,CAAC,EAAE,CAAC,IAAI,EAAE,OAAO,KAAK,OAAO,EAAE,GAAG;QAAE,KAAK,EAAE,OAAO,EAAE,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IAC1F,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export {};
|
|
@@ -1,4 +0,0 @@
|
|
|
1
|
-
import { QueryClient } from "@tanstack/react-query";
|
|
2
|
-
export declare function scheduleInvalidations(queryClient: QueryClient, tasks: any[], delay: number): void;
|
|
3
|
-
export declare function cancelScheduledInvalidations(tasks: any[]): void;
|
|
4
|
-
//# sourceMappingURL=invalidationManager.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"invalidationManager.d.ts","sourceRoot":"","sources":["../../src/utils/invalidationManager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAUpD,wBAAgB,qBAAqB,CAAC,WAAW,EAAE,WAAW,EAAE,KAAK,EAAE,GAAG,EAAE,EAAE,KAAK,EAAE,MAAM,QAe1F;AAED,wBAAgB,4BAA4B,CAAC,KAAK,EAAE,GAAG,EAAE,QAQxD"}
|
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
const timers = new Map();
|
|
2
|
-
function getTaskKey(task) {
|
|
3
|
-
// Serialize the task to identify duplicates.
|
|
4
|
-
// We mainly care about queryKey and exact/refetchType properties.
|
|
5
|
-
return JSON.stringify(task);
|
|
6
|
-
}
|
|
7
|
-
export function scheduleInvalidations(queryClient, tasks, delay) {
|
|
8
|
-
tasks.forEach(task => {
|
|
9
|
-
const key = getTaskKey(task);
|
|
10
|
-
if (timers.has(key)) {
|
|
11
|
-
clearTimeout(timers.get(key));
|
|
12
|
-
}
|
|
13
|
-
const timer = setTimeout(() => {
|
|
14
|
-
queryClient.invalidateQueries(task);
|
|
15
|
-
timers.delete(key);
|
|
16
|
-
}, delay);
|
|
17
|
-
timers.set(key, timer);
|
|
18
|
-
});
|
|
19
|
-
}
|
|
20
|
-
export function cancelScheduledInvalidations(tasks) {
|
|
21
|
-
tasks.forEach(task => {
|
|
22
|
-
const key = getTaskKey(task);
|
|
23
|
-
if (timers.has(key)) {
|
|
24
|
-
clearTimeout(timers.get(key));
|
|
25
|
-
timers.delete(key);
|
|
26
|
-
}
|
|
27
|
-
});
|
|
28
|
-
}
|