@based/react 4.3.1 → 4.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/Ctx.d.ts ADDED
@@ -0,0 +1,3 @@
1
+ /// <reference types="react" />
2
+ import { BasedClient } from '@based/client';
3
+ export declare const Ctx: import("react").Context<BasedClient>;
package/dist/Ctx.js ADDED
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Ctx = void 0;
4
+ const react_1 = require("react");
5
+ exports.Ctx = (0, react_1.createContext)(null);
6
+ //# sourceMappingURL=Ctx.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Ctx.js","sourceRoot":"","sources":["../src/Ctx.ts"],"names":[],"mappings":";;;AAAA,iCAAqC;AAGxB,QAAA,GAAG,GAAG,IAAA,qBAAa,EAAc,IAAI,CAAC,CAAA"}
@@ -0,0 +1,6 @@
1
+ import { FC, ReactNode } from 'react';
2
+ import { BasedClient } from '@based/client';
3
+ export declare const Provider: FC<{
4
+ client: BasedClient;
5
+ children: ReactNode;
6
+ }>;
@@ -0,0 +1,10 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Provider = void 0;
4
+ const react_1 = require("react");
5
+ const Ctx_1 = require("./Ctx");
6
+ const Provider = ({ client, children }) => {
7
+ return (0, react_1.createElement)(Ctx_1.Ctx.Provider, { value: client }, children);
8
+ };
9
+ exports.Provider = Provider;
10
+ //# sourceMappingURL=Provider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Provider.js","sourceRoot":"","sources":["../src/Provider.ts"],"names":[],"mappings":";;;AAAA,iCAAoD;AAEpD,+BAA2B;AAEpB,MAAM,QAAQ,GAGhB,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,EAAE,EAAE;IAC5B,OAAO,IAAA,qBAAa,EAAC,SAAG,CAAC,QAAQ,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,EAAE,QAAQ,CAAC,CAAA;AACjE,CAAC,CAAA;AALY,QAAA,QAAQ,YAKpB"}
package/dist/index.d.ts CHANGED
@@ -1,21 +1,8 @@
1
- import { FC, ReactNode } from 'react';
2
- import { BasedClient, AuthState } from '@based/client';
3
- import { BasedError } from '@based/client';
4
- export declare const Provider: FC<{
5
- client: BasedClient;
6
- children: ReactNode;
7
- }>;
8
- export declare const useAuthState: () => AuthState;
9
- export declare const useLoading: () => boolean;
10
- export declare const useConnected: () => {
11
- connected: boolean;
12
- };
13
- export declare const useQuery: <T = any>(name?: string, payload?: any, opts?: {
14
- persistent: boolean;
15
- }) => {
16
- loading: boolean;
17
- data?: T;
18
- error?: BasedError;
19
- checksum?: number;
20
- };
21
- export declare const useClient: () => BasedClient;
1
+ export * from './Ctx';
2
+ export * from './Provider';
3
+ export * from './useAuthState';
4
+ export * from './useClient';
5
+ export * from './useConnected';
6
+ export * from './useLoading';
7
+ export * from './useQuery';
8
+ export * from './useWindow';
package/dist/index.js CHANGED
@@ -1,120 +1,25 @@
1
1
  "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.useClient = exports.useQuery = exports.useConnected = exports.useLoading = exports.useAuthState = exports.Provider = void 0;
4
- const react_1 = require("react");
5
- const Ctx = (0, react_1.createContext)(null);
6
- const Provider = ({ client, children }) => {
7
- return (0, react_1.createElement)(Ctx.Provider, { value: client }, children);
8
- };
9
- exports.Provider = Provider;
10
- const useAuthState = () => {
11
- const client = (0, react_1.useContext)(Ctx);
12
- const [state, setState] = (0, react_1.useState)(client?.authState || {});
13
- (0, react_1.useEffect)(() => {
14
- if (client) {
15
- setState(client.authState);
16
- const listener = (authState) => {
17
- setState(authState);
18
- };
19
- client.on('authstate-change', listener);
20
- return () => client.off('authstate-change', listener);
21
- }
22
- }, [client]);
23
- return state;
24
- };
25
- exports.useAuthState = useAuthState;
26
- const useLoadingListeners = new Set();
27
- const hooksLoading = new Set();
28
- const useLoading = () => {
29
- const [isLoading, setLoading] = (0, react_1.useState)(hooksLoading.size > 0);
30
- (0, react_1.useEffect)(() => {
31
- useLoadingListeners.add(setLoading);
32
- return () => {
33
- useLoadingListeners.delete(setLoading);
34
- };
35
- }, []);
36
- return isLoading;
37
- };
38
- exports.useLoading = useLoading;
39
- const useConnected = () => {
40
- const client = (0, react_1.useContext)(Ctx);
41
- const [connected, setConnected] = (0, react_1.useState)(client.connected);
42
- (0, react_1.useEffect)(() => {
43
- if (client) {
44
- setConnected(client.connected);
45
- const listener = () => {
46
- setConnected(client.connected);
47
- };
48
- client.on('disconnect', listener);
49
- client.on('reconnect', listener);
50
- client.on('connect', listener);
51
- return () => {
52
- client.off('disconnect', listener);
53
- client.off('reconnect', listener);
54
- client.off('connect', listener);
55
- };
56
- }
57
- }, [client]);
58
- return { connected };
59
- };
60
- exports.useConnected = useConnected;
61
- const useQuery = (name, payload, opts) => {
62
- const client = (0, react_1.useContext)(Ctx);
63
- if (client && name) {
64
- const q = client.query(name, payload, opts);
65
- const { id, cache } = q;
66
- const [checksumOrError, update] = (0, react_1.useState)(cache?.checksum);
67
- (0, react_1.useEffect)(() => {
68
- const unsubscribe = q.subscribe((_, checksum) => {
69
- update(checksum);
70
- }, (err) => {
71
- update(err);
72
- });
73
- return () => {
74
- const isLoading = hooksLoading.size > 0;
75
- if (hooksLoading.delete(id) && !(hooksLoading.size > 0) && isLoading) {
76
- useLoadingListeners.forEach((fn) => {
77
- fn(false);
78
- });
79
- }
80
- unsubscribe();
81
- update(0);
82
- };
83
- }, [id]);
84
- if (checksumOrError) {
85
- const isLoading = hooksLoading.size > 0;
86
- if (hooksLoading.delete(id)) {
87
- if (!(hooksLoading.size > 0) && isLoading) {
88
- useLoadingListeners.forEach((fn) => {
89
- fn(false);
90
- });
91
- }
92
- }
93
- if (typeof checksumOrError === 'number') {
94
- if (!cache) {
95
- return { loading: true };
96
- }
97
- return { loading: false, data: cache.value, checksum: checksumOrError };
98
- }
99
- return { loading: false, error: checksumOrError };
100
- }
101
- const isLoading = hooksLoading.size > 0;
102
- if (hooksLoading.add(id)) {
103
- if (!isLoading) {
104
- useLoadingListeners.forEach((fn) => {
105
- fn(true);
106
- });
107
- }
108
- }
109
- return { loading: true };
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
110
7
  }
111
- (0, react_1.useState)();
112
- (0, react_1.useEffect)(() => { }, [null]);
113
- return { loading: true };
114
- };
115
- exports.useQuery = useQuery;
116
- const useClient = () => {
117
- return (0, react_1.useContext)(Ctx);
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
118
15
  };
119
- exports.useClient = useClient;
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./Ctx"), exports);
18
+ __exportStar(require("./Provider"), exports);
19
+ __exportStar(require("./useAuthState"), exports);
20
+ __exportStar(require("./useClient"), exports);
21
+ __exportStar(require("./useConnected"), exports);
22
+ __exportStar(require("./useLoading"), exports);
23
+ __exportStar(require("./useQuery"), exports);
24
+ __exportStar(require("./useWindow"), exports);
120
25
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAAA,wCAAqB;AACrB,6CAA0B;AAC1B,iDAA8B;AAC9B,8CAA2B;AAC3B,iDAA8B;AAC9B,+CAA4B;AAC5B,6CAA0B;AAC1B,8CAA2B"}
@@ -0,0 +1,2 @@
1
+ import { AuthState } from '@based/client';
2
+ export declare const useAuthState: () => AuthState;
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.useAuthState = void 0;
4
+ const react_1 = require("react");
5
+ const Ctx_1 = require("./Ctx");
6
+ const useAuthState = () => {
7
+ const client = (0, react_1.useContext)(Ctx_1.Ctx);
8
+ const [state, setState] = (0, react_1.useState)(client?.authState || {});
9
+ (0, react_1.useEffect)(() => {
10
+ if (client) {
11
+ setState(client.authState);
12
+ const listener = (authState) => {
13
+ setState(authState);
14
+ };
15
+ client.on('authstate-change', listener);
16
+ return () => client.off('authstate-change', listener);
17
+ }
18
+ }, [client]);
19
+ return state;
20
+ };
21
+ exports.useAuthState = useAuthState;
22
+ //# sourceMappingURL=useAuthState.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useAuthState.js","sourceRoot":"","sources":["../src/useAuthState.ts"],"names":[],"mappings":";;;AAAA,iCAAuD;AAEvD,+BAA2B;AAEpB,MAAM,YAAY,GAAG,GAAc,EAAE;IAC1C,MAAM,MAAM,GAAgB,IAAA,kBAAU,EAAC,SAAG,CAAC,CAAA;IAC3C,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,IAAA,gBAAQ,EAAY,MAAM,EAAE,SAAS,IAAI,EAAE,CAAC,CAAA;IAEtE,IAAA,iBAAS,EAAC,GAAG,EAAE;QACb,IAAI,MAAM,EAAE;YACV,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,CAAA;YAC1B,MAAM,QAAQ,GAAG,CAAC,SAAS,EAAE,EAAE;gBAC7B,QAAQ,CAAC,SAAS,CAAC,CAAA;YACrB,CAAC,CAAA;YACD,MAAM,CAAC,EAAE,CAAC,kBAAkB,EAAE,QAAQ,CAAC,CAAA;YACvC,OAAO,GAAG,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,kBAAkB,EAAE,QAAQ,CAAC,CAAA;SACtD;IACH,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAA;IAEZ,OAAO,KAAK,CAAA;AACd,CAAC,CAAA;AAhBY,QAAA,YAAY,gBAgBxB"}
@@ -0,0 +1,2 @@
1
+ import { BasedClient } from '@based/client';
2
+ export declare const useClient: () => BasedClient;
@@ -0,0 +1,10 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.useClient = void 0;
4
+ const react_1 = require("react");
5
+ const Ctx_1 = require("./Ctx");
6
+ const useClient = () => {
7
+ return (0, react_1.useContext)(Ctx_1.Ctx);
8
+ };
9
+ exports.useClient = useClient;
10
+ //# sourceMappingURL=useClient.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useClient.js","sourceRoot":"","sources":["../src/useClient.ts"],"names":[],"mappings":";;;AAAA,iCAAkC;AAElC,+BAA2B;AAEpB,MAAM,SAAS,GAAG,GAAgB,EAAE;IACzC,OAAO,IAAA,kBAAU,EAAC,SAAG,CAAC,CAAA;AACxB,CAAC,CAAA;AAFY,QAAA,SAAS,aAErB"}
@@ -0,0 +1,3 @@
1
+ export declare const useConnected: () => {
2
+ connected: boolean;
3
+ };
@@ -0,0 +1,28 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.useConnected = void 0;
4
+ const react_1 = require("react");
5
+ const Ctx_1 = require("./Ctx");
6
+ const useConnected = () => {
7
+ const client = (0, react_1.useContext)(Ctx_1.Ctx);
8
+ const [connected, setConnected] = (0, react_1.useState)(client.connected);
9
+ (0, react_1.useEffect)(() => {
10
+ if (client) {
11
+ setConnected(client.connected);
12
+ const listener = () => {
13
+ setConnected(client.connected);
14
+ };
15
+ client.on('disconnect', listener);
16
+ client.on('reconnect', listener);
17
+ client.on('connect', listener);
18
+ return () => {
19
+ client.off('disconnect', listener);
20
+ client.off('reconnect', listener);
21
+ client.off('connect', listener);
22
+ };
23
+ }
24
+ }, [client]);
25
+ return { connected };
26
+ };
27
+ exports.useConnected = useConnected;
28
+ //# sourceMappingURL=useConnected.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useConnected.js","sourceRoot":"","sources":["../src/useConnected.ts"],"names":[],"mappings":";;;AAAA,iCAAuD;AAEvD,+BAA2B;AAEpB,MAAM,YAAY,GAAG,GAAG,EAAE;IAC/B,MAAM,MAAM,GAAgB,IAAA,kBAAU,EAAC,SAAG,CAAC,CAAA;IAC3C,MAAM,CAAC,SAAS,EAAE,YAAY,CAAC,GAAG,IAAA,gBAAQ,EAAC,MAAM,CAAC,SAAS,CAAC,CAAA;IAE5D,IAAA,iBAAS,EAAC,GAAG,EAAE;QACb,IAAI,MAAM,EAAE;YACV,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,CAAA;YAC9B,MAAM,QAAQ,GAAG,GAAG,EAAE;gBACpB,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,CAAA;YAChC,CAAC,CAAA;YACD,MAAM,CAAC,EAAE,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAA;YACjC,MAAM,CAAC,EAAE,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAA;YAChC,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAA;YAC9B,OAAO,GAAG,EAAE;gBACV,MAAM,CAAC,GAAG,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAA;gBAClC,MAAM,CAAC,GAAG,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAA;gBACjC,MAAM,CAAC,GAAG,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAA;YACjC,CAAC,CAAA;SACF;IACH,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAA;IAEZ,OAAO,EAAE,SAAS,EAAE,CAAA;AACtB,CAAC,CAAA;AAtBY,QAAA,YAAY,gBAsBxB"}
@@ -0,0 +1,3 @@
1
+ export declare const useLoadingListeners: Set<Function>;
2
+ export declare const hooksLoading: Set<number>;
3
+ export declare const useLoading: () => boolean;
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.useLoading = exports.hooksLoading = exports.useLoadingListeners = void 0;
4
+ const react_1 = require("react");
5
+ exports.useLoadingListeners = new Set();
6
+ exports.hooksLoading = new Set();
7
+ const useLoading = () => {
8
+ const [isLoading, setLoading] = (0, react_1.useState)(exports.hooksLoading.size > 0);
9
+ (0, react_1.useEffect)(() => {
10
+ exports.useLoadingListeners.add(setLoading);
11
+ return () => {
12
+ exports.useLoadingListeners.delete(setLoading);
13
+ };
14
+ }, []);
15
+ return isLoading;
16
+ };
17
+ exports.useLoading = useLoading;
18
+ //# sourceMappingURL=useLoading.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useLoading.js","sourceRoot":"","sources":["../src/useLoading.ts"],"names":[],"mappings":";;;AAAA,iCAA2C;AAE9B,QAAA,mBAAmB,GAAkB,IAAI,GAAG,EAAE,CAAA;AAC9C,QAAA,YAAY,GAAgB,IAAI,GAAG,EAAE,CAAA;AAC3C,MAAM,UAAU,GAAG,GAAG,EAAE;IAC7B,MAAM,CAAC,SAAS,EAAE,UAAU,CAAC,GAAG,IAAA,gBAAQ,EAAC,oBAAY,CAAC,IAAI,GAAG,CAAC,CAAC,CAAA;IAC/D,IAAA,iBAAS,EAAC,GAAG,EAAE;QACb,2BAAmB,CAAC,GAAG,CAAC,UAAU,CAAC,CAAA;QACnC,OAAO,GAAG,EAAE;YACV,2BAAmB,CAAC,MAAM,CAAC,UAAU,CAAC,CAAA;QACxC,CAAC,CAAA;IACH,CAAC,EAAE,EAAE,CAAC,CAAA;IACN,OAAO,SAAS,CAAA;AAClB,CAAC,CAAA;AATY,QAAA,UAAU,cAStB"}
@@ -0,0 +1,9 @@
1
+ import { BasedError } from '@based/client';
2
+ export declare const useQuery: <T = any>(name?: string, payload?: any, opts?: {
3
+ persistent: boolean;
4
+ }) => {
5
+ loading: boolean;
6
+ data?: T;
7
+ error?: BasedError;
8
+ checksum?: number;
9
+ };
@@ -0,0 +1,62 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.useQuery = void 0;
4
+ const react_1 = require("react");
5
+ const Ctx_1 = require("./Ctx");
6
+ const useLoading_1 = require("./useLoading");
7
+ const useQuery = (name, payload, opts) => {
8
+ const client = (0, react_1.useContext)(Ctx_1.Ctx);
9
+ if (client && name) {
10
+ const q = client.query(name, payload, opts);
11
+ const { id, cache } = q;
12
+ const [checksumOrError, update] = (0, react_1.useState)(cache?.checksum);
13
+ (0, react_1.useEffect)(() => {
14
+ const unsubscribe = q.subscribe((_, checksum) => {
15
+ update(checksum);
16
+ }, (err) => {
17
+ update(err);
18
+ });
19
+ return () => {
20
+ const isLoading = useLoading_1.hooksLoading.size > 0;
21
+ if (useLoading_1.hooksLoading.delete(id) && !(useLoading_1.hooksLoading.size > 0) && isLoading) {
22
+ useLoading_1.useLoadingListeners.forEach((fn) => {
23
+ fn(false);
24
+ });
25
+ }
26
+ unsubscribe();
27
+ update(0);
28
+ };
29
+ }, [id]);
30
+ if (checksumOrError) {
31
+ const isLoading = useLoading_1.hooksLoading.size > 0;
32
+ if (useLoading_1.hooksLoading.delete(id)) {
33
+ if (!(useLoading_1.hooksLoading.size > 0) && isLoading) {
34
+ useLoading_1.useLoadingListeners.forEach((fn) => {
35
+ fn(false);
36
+ });
37
+ }
38
+ }
39
+ if (typeof checksumOrError === 'number') {
40
+ if (!cache) {
41
+ return { loading: true };
42
+ }
43
+ return { loading: false, data: cache.value, checksum: checksumOrError };
44
+ }
45
+ return { loading: false, error: checksumOrError };
46
+ }
47
+ const isLoading = useLoading_1.hooksLoading.size > 0;
48
+ if (useLoading_1.hooksLoading.add(id)) {
49
+ if (!isLoading) {
50
+ useLoading_1.useLoadingListeners.forEach((fn) => {
51
+ fn(true);
52
+ });
53
+ }
54
+ }
55
+ return { loading: true };
56
+ }
57
+ (0, react_1.useState)();
58
+ (0, react_1.useEffect)(() => { }, [null]);
59
+ return { loading: true };
60
+ };
61
+ exports.useQuery = useQuery;
62
+ //# sourceMappingURL=useQuery.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useQuery.js","sourceRoot":"","sources":["../src/useQuery.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ type UseWindowState = {
2
+ loading: boolean;
3
+ items: any[];
4
+ checksum?: number;
5
+ };
6
+ export declare const useWindow: (name: string, getPayload: ({ offset, limit }: {
7
+ offset: any;
8
+ limit: any;
9
+ }) => any, opts: {
10
+ path: string[];
11
+ pages: number[];
12
+ size: number;
13
+ persistent?: boolean;
14
+ }, dependencies?: any) => UseWindowState;
15
+ export {};
@@ -0,0 +1,125 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.useWindow = void 0;
4
+ const react_1 = require("react");
5
+ const Ctx_1 = require("./Ctx");
6
+ const hash_1 = require("@saulx/hash");
7
+ const useWindow = (name, getPayload, opts, dependencies) => {
8
+ const [checksum, setCheckum] = (0, react_1.useState)(0);
9
+ const cache = (0, react_1.useRef)();
10
+ const client = (0, react_1.useContext)(Ctx_1.Ctx);
11
+ const queries = (0, react_1.useRef)();
12
+ const unsubs = (0, react_1.useRef)();
13
+ const raf = (0, react_1.useRef)();
14
+ const currModifyHash = (0, react_1.useRef)();
15
+ const currResetHash = (0, react_1.useRef)();
16
+ const modifyHash = `${opts.size}.${opts.persistent}`;
17
+ if (dependencies) {
18
+ // complete reset
19
+ const resetHash = (0, hash_1.hash)(dependencies);
20
+ if (currResetHash.current !== resetHash) {
21
+ currModifyHash.current = null;
22
+ cache.current = null;
23
+ }
24
+ }
25
+ if (currModifyHash.current !== modifyHash) {
26
+ // reset queries but keep cache
27
+ for (const n in unsubs.current) {
28
+ unsubs.current[n]();
29
+ }
30
+ if (raf.current) {
31
+ cancelAnimationFrame(raf.current);
32
+ raf.current = null;
33
+ }
34
+ queries.current = [];
35
+ unsubs.current = {};
36
+ currModifyHash.current = modifyHash;
37
+ if (!cache.current) {
38
+ cache.current = { items: [], loading: true, checksum: 0 };
39
+ }
40
+ }
41
+ (0, react_1.useEffect)(() => {
42
+ return () => {
43
+ for (const n in unsubs.current) {
44
+ unsubs.current[n]();
45
+ }
46
+ if (raf.current) {
47
+ cancelAnimationFrame(raf.current);
48
+ }
49
+ };
50
+ }, []);
51
+ if (client && name) {
52
+ const { pages, size, persistent = false, path } = opts;
53
+ const active = new Set();
54
+ // add new subs
55
+ pages.forEach((n) => {
56
+ // pages start at 1 => shift to 0
57
+ n -= 1;
58
+ active.add(n);
59
+ if (!(n in queries.current)) {
60
+ const payload = getPayload({ offset: n * size, limit: size });
61
+ const q = client.query(name, payload, { persistent });
62
+ queries.current[n] = q;
63
+ }
64
+ if (n in unsubs.current)
65
+ return;
66
+ unsubs.current[n] = queries.current[n].subscribe(() => {
67
+ if (raf.current)
68
+ return;
69
+ raf.current = requestAnimationFrame(() => {
70
+ raf.current = null;
71
+ setCheckum(queries.current.reduce((combined, { cache }) => {
72
+ const checksum = cache?.checksum;
73
+ return checksum ? combined + checksum : combined;
74
+ }, 0));
75
+ });
76
+ });
77
+ });
78
+ // remove inactive subs
79
+ queries.current.forEach((_, n) => {
80
+ if (n in unsubs.current && !active.has(n)) {
81
+ unsubs.current[n]();
82
+ delete unsubs.current[n];
83
+ }
84
+ });
85
+ if (cache.current.checksum === checksum) {
86
+ return cache.current;
87
+ }
88
+ let l = queries.current.length;
89
+ cache.current.loading = false;
90
+ cache.current.checksum = checksum;
91
+ while (l--) {
92
+ const q = queries.current[l];
93
+ const m = size * l + size;
94
+ let i = size * l;
95
+ if (q) {
96
+ if (q.cache) {
97
+ let data = q.cache.value;
98
+ if (data) {
99
+ for (const i of path) {
100
+ data = data[i];
101
+ }
102
+ for (let j = 0; i < m; i++) {
103
+ const item = data[j++];
104
+ if (!item)
105
+ break;
106
+ cache.current.items[i] = item;
107
+ }
108
+ }
109
+ }
110
+ else {
111
+ cache.current.loading = true;
112
+ }
113
+ }
114
+ // fill up empty items with null
115
+ for (; i < m; i++) {
116
+ if (!(i in cache.current.items)) {
117
+ cache.current.items[i] = null;
118
+ }
119
+ }
120
+ }
121
+ }
122
+ return cache.current;
123
+ };
124
+ exports.useWindow = useWindow;
125
+ //# sourceMappingURL=useWindow.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useWindow.js","sourceRoot":"","sources":["../src/useWindow.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,12 +1,12 @@
1
1
  {
2
2
  "name": "@based/react",
3
- "version": "4.3.1",
3
+ "version": "4.4.0",
4
4
  "license": "MIT",
5
5
  "main": "dist/index.js",
6
6
  "scripts": {
7
7
  "browser": "npx aristotle watch -t ./test/browser.tsx",
8
8
  "server": "npx ts-node ./test/based-server.ts",
9
- "build": "npx tsc",
9
+ "build": "npx tsc",
10
10
  "watch": "npm run build -- --watch",
11
11
  "clean": "rimraf {.turbo,dist,node_modules}"
12
12
  },
package/src/Ctx.ts ADDED
@@ -0,0 +1,4 @@
1
+ import { createContext } from 'react'
2
+ import { BasedClient } from '@based/client'
3
+
4
+ export const Ctx = createContext<BasedClient>(null)
@@ -0,0 +1,10 @@
1
+ import { createElement, FC, ReactNode } from 'react'
2
+ import { BasedClient } from '@based/client'
3
+ import { Ctx } from './Ctx'
4
+
5
+ export const Provider: FC<{
6
+ client: BasedClient
7
+ children: ReactNode
8
+ }> = ({ client, children }) => {
9
+ return createElement(Ctx.Provider, { value: client }, children)
10
+ }
package/src/index.ts CHANGED
@@ -1,161 +1,8 @@
1
- import {
2
- useContext,
3
- createContext,
4
- createElement,
5
- useState,
6
- useEffect,
7
- FC,
8
- ReactNode,
9
- } from 'react'
10
- import { BasedClient, AuthState } from '@based/client'
11
- import { BasedError } from '@based/client'
12
-
13
- const Ctx = createContext<BasedClient>(null)
14
-
15
- export const Provider: FC<{
16
- client: BasedClient
17
- children: ReactNode
18
- }> = ({ client, children }) => {
19
- return createElement(Ctx.Provider, { value: client }, children)
20
- }
21
-
22
- export const useAuthState = (): AuthState => {
23
- const client: BasedClient = useContext(Ctx)
24
- const [state, setState] = useState<AuthState>(client?.authState || {})
25
-
26
- useEffect(() => {
27
- if (client) {
28
- setState(client.authState)
29
- const listener = (authState) => {
30
- setState(authState)
31
- }
32
- client.on('authstate-change', listener)
33
- return () => client.off('authstate-change', listener)
34
- }
35
- }, [client])
36
-
37
- return state
38
- }
39
-
40
- const useLoadingListeners: Set<Function> = new Set()
41
- const hooksLoading: Set<number> = new Set()
42
-
43
- export const useLoading = () => {
44
- const [isLoading, setLoading] = useState(hooksLoading.size > 0)
45
- useEffect(() => {
46
- useLoadingListeners.add(setLoading)
47
- return () => {
48
- useLoadingListeners.delete(setLoading)
49
- }
50
- }, [])
51
- return isLoading
52
- }
53
-
54
- export const useConnected = () => {
55
- const client: BasedClient = useContext(Ctx)
56
- const [connected, setConnected] = useState(client.connected)
57
-
58
- useEffect(() => {
59
- if (client) {
60
- setConnected(client.connected)
61
- const listener = () => {
62
- setConnected(client.connected)
63
- }
64
- client.on('disconnect', listener)
65
- client.on('reconnect', listener)
66
- client.on('connect', listener)
67
- return () => {
68
- client.off('disconnect', listener)
69
- client.off('reconnect', listener)
70
- client.off('connect', listener)
71
- }
72
- }
73
- }, [client])
74
-
75
- return { connected }
76
- }
77
-
78
- export const useQuery = <T = any>(
79
- name?: string,
80
- payload?: any,
81
- opts?: {
82
- persistent: boolean
83
- }
84
- ): {
85
- loading: boolean
86
- data?: T
87
- error?: BasedError
88
- checksum?: number
89
- } => {
90
- const client: BasedClient = useContext(Ctx)
91
-
92
- if (client && name) {
93
- const q = client.query(name, payload, opts)
94
- const { id, cache } = q
95
- const [checksumOrError, update] = useState<number | BasedError>(
96
- cache?.checksum
97
- )
98
-
99
- useEffect(() => {
100
- const unsubscribe = q.subscribe(
101
- (_, checksum) => {
102
- update(checksum)
103
- },
104
- (err) => {
105
- update(err)
106
- }
107
- )
108
- return () => {
109
- const isLoading = hooksLoading.size > 0
110
- if (hooksLoading.delete(id) && !(hooksLoading.size > 0) && isLoading) {
111
- useLoadingListeners.forEach((fn) => {
112
- fn(false)
113
- })
114
- }
115
- unsubscribe()
116
- update(0)
117
- }
118
- }, [id])
119
-
120
- if (checksumOrError) {
121
- const isLoading = hooksLoading.size > 0
122
- if (hooksLoading.delete(id)) {
123
- if (!(hooksLoading.size > 0) && isLoading) {
124
- useLoadingListeners.forEach((fn) => {
125
- fn(false)
126
- })
127
- }
128
- }
129
-
130
- if (typeof checksumOrError === 'number') {
131
- if (!cache) {
132
- return { loading: true }
133
- }
134
-
135
- return { loading: false, data: cache.value, checksum: checksumOrError }
136
- }
137
-
138
- return { loading: false, error: checksumOrError }
139
- }
140
-
141
- const isLoading = hooksLoading.size > 0
142
- if (hooksLoading.add(id)) {
143
- if (!isLoading) {
144
- useLoadingListeners.forEach((fn) => {
145
- fn(true)
146
- })
147
- }
148
- }
149
-
150
- return { loading: true }
151
- }
152
-
153
- useState()
154
- useEffect(() => {}, [null])
155
-
156
- return { loading: true }
157
- }
158
-
159
- export const useClient = (): BasedClient => {
160
- return useContext(Ctx)
161
- }
1
+ export * from './Ctx'
2
+ export * from './Provider'
3
+ export * from './useAuthState'
4
+ export * from './useClient'
5
+ export * from './useConnected'
6
+ export * from './useLoading'
7
+ export * from './useQuery'
8
+ export * from './useWindow'
@@ -0,0 +1,21 @@
1
+ import { useContext, useState, useEffect } from 'react'
2
+ import { BasedClient, AuthState } from '@based/client'
3
+ import { Ctx } from './Ctx'
4
+
5
+ export const useAuthState = (): AuthState => {
6
+ const client: BasedClient = useContext(Ctx)
7
+ const [state, setState] = useState<AuthState>(client?.authState || {})
8
+
9
+ useEffect(() => {
10
+ if (client) {
11
+ setState(client.authState)
12
+ const listener = (authState) => {
13
+ setState(authState)
14
+ }
15
+ client.on('authstate-change', listener)
16
+ return () => client.off('authstate-change', listener)
17
+ }
18
+ }, [client])
19
+
20
+ return state
21
+ }
@@ -0,0 +1,7 @@
1
+ import { useContext } from 'react'
2
+ import { BasedClient } from '@based/client'
3
+ import { Ctx } from './Ctx'
4
+
5
+ export const useClient = (): BasedClient => {
6
+ return useContext(Ctx)
7
+ }
@@ -0,0 +1,27 @@
1
+ import { useContext, useState, useEffect } from 'react'
2
+ import { BasedClient } from '@based/client'
3
+ import { Ctx } from './Ctx'
4
+
5
+ export const useConnected = () => {
6
+ const client: BasedClient = useContext(Ctx)
7
+ const [connected, setConnected] = useState(client.connected)
8
+
9
+ useEffect(() => {
10
+ if (client) {
11
+ setConnected(client.connected)
12
+ const listener = () => {
13
+ setConnected(client.connected)
14
+ }
15
+ client.on('disconnect', listener)
16
+ client.on('reconnect', listener)
17
+ client.on('connect', listener)
18
+ return () => {
19
+ client.off('disconnect', listener)
20
+ client.off('reconnect', listener)
21
+ client.off('connect', listener)
22
+ }
23
+ }
24
+ }, [client])
25
+
26
+ return { connected }
27
+ }
@@ -0,0 +1,14 @@
1
+ import { useState, useEffect } from 'react'
2
+
3
+ export const useLoadingListeners: Set<Function> = new Set()
4
+ export const hooksLoading: Set<number> = new Set()
5
+ export const useLoading = () => {
6
+ const [isLoading, setLoading] = useState(hooksLoading.size > 0)
7
+ useEffect(() => {
8
+ useLoadingListeners.add(setLoading)
9
+ return () => {
10
+ useLoadingListeners.delete(setLoading)
11
+ }
12
+ }, [])
13
+ return isLoading
14
+ }
@@ -0,0 +1,85 @@
1
+ import { useContext, useState, useEffect } from 'react'
2
+ import { BasedClient, BasedError } from '@based/client'
3
+ import { Ctx } from './Ctx'
4
+ import { hooksLoading, useLoadingListeners } from './useLoading'
5
+
6
+ export const useQuery = <T = any>(
7
+ name?: string,
8
+ payload?: any,
9
+ opts?: {
10
+ persistent: boolean
11
+ }
12
+ ): {
13
+ loading: boolean
14
+ data?: T
15
+ error?: BasedError
16
+ checksum?: number
17
+ } => {
18
+ const client: BasedClient = useContext(Ctx)
19
+
20
+ if (client && name) {
21
+ const q = client.query(name, payload, opts)
22
+ const { id, cache } = q
23
+ const [checksumOrError, update] = useState<number | BasedError>(
24
+ cache?.checksum
25
+ )
26
+
27
+ useEffect(() => {
28
+ const unsubscribe = q.subscribe(
29
+ (_, checksum) => {
30
+ update(checksum)
31
+ },
32
+ (err) => {
33
+ update(err)
34
+ }
35
+ )
36
+ return () => {
37
+ const isLoading = hooksLoading.size > 0
38
+ if (hooksLoading.delete(id) && !(hooksLoading.size > 0) && isLoading) {
39
+ useLoadingListeners.forEach((fn) => {
40
+ fn(false)
41
+ })
42
+ }
43
+ unsubscribe()
44
+ update(0)
45
+ }
46
+ }, [id])
47
+
48
+ if (checksumOrError) {
49
+ const isLoading = hooksLoading.size > 0
50
+ if (hooksLoading.delete(id)) {
51
+ if (!(hooksLoading.size > 0) && isLoading) {
52
+ useLoadingListeners.forEach((fn) => {
53
+ fn(false)
54
+ })
55
+ }
56
+ }
57
+
58
+ if (typeof checksumOrError === 'number') {
59
+ if (!cache) {
60
+ return { loading: true }
61
+ }
62
+
63
+ return { loading: false, data: cache.value, checksum: checksumOrError }
64
+ }
65
+
66
+ return { loading: false, error: checksumOrError }
67
+ }
68
+
69
+ const isLoading = hooksLoading.size > 0
70
+ if (hooksLoading.add(id)) {
71
+ if (!isLoading) {
72
+ useLoadingListeners.forEach((fn) => {
73
+ fn(true)
74
+ })
75
+ }
76
+ }
77
+
78
+ return { loading: true }
79
+ }
80
+
81
+ useState()
82
+ useEffect(() => {}, [null])
83
+
84
+ return { loading: true }
85
+ }
@@ -0,0 +1,149 @@
1
+ import { useState, useEffect, useContext, useRef } from 'react'
2
+ import { Ctx } from './Ctx'
3
+ import { BasedClient, BasedQuery } from '@based/client'
4
+ import { hash } from '@saulx/hash'
5
+
6
+ type UseWindowState = {
7
+ loading: boolean
8
+ items: any[]
9
+ checksum?: number
10
+ }
11
+
12
+ export const useWindow = (
13
+ name: string,
14
+ getPayload: ({ offset, limit }) => any,
15
+ opts: {
16
+ path: string[]
17
+ pages: number[]
18
+ size: number
19
+ persistent?: boolean
20
+ },
21
+ dependencies?: any
22
+ ): UseWindowState => {
23
+ const [checksum, setCheckum] = useState(0)
24
+ const cache = useRef<UseWindowState>()
25
+ const client: BasedClient = useContext(Ctx)
26
+ const queries = useRef<BasedQuery[]>()
27
+ const unsubs = useRef<{}>()
28
+ const raf = useRef<number>()
29
+ const currModifyHash = useRef<string>()
30
+ const currResetHash = useRef<number>()
31
+ const modifyHash = `${opts.size}.${opts.persistent}`
32
+
33
+ if (dependencies) {
34
+ // complete reset
35
+ const resetHash = hash(dependencies)
36
+ if (currResetHash.current !== resetHash) {
37
+ currModifyHash.current = null
38
+ cache.current = null
39
+ }
40
+ }
41
+
42
+ if (currModifyHash.current !== modifyHash) {
43
+ // reset queries but keep cache
44
+ for (const n in unsubs.current) {
45
+ unsubs.current[n]()
46
+ }
47
+ if (raf.current) {
48
+ cancelAnimationFrame(raf.current)
49
+ raf.current = null
50
+ }
51
+ queries.current = []
52
+ unsubs.current = {}
53
+ currModifyHash.current = modifyHash
54
+
55
+ if (!cache.current) {
56
+ cache.current = { items: [], loading: true, checksum: 0 }
57
+ }
58
+ }
59
+
60
+ useEffect(() => {
61
+ return () => {
62
+ for (const n in unsubs.current) {
63
+ unsubs.current[n]()
64
+ }
65
+ if (raf.current) {
66
+ cancelAnimationFrame(raf.current)
67
+ }
68
+ }
69
+ }, [])
70
+
71
+ if (client && name) {
72
+ const { pages, size, persistent = false, path } = opts
73
+ const active = new Set()
74
+
75
+ // add new subs
76
+ pages.forEach((n) => {
77
+ // pages start at 1 => shift to 0
78
+ n -= 1
79
+ active.add(n)
80
+ if (!(n in queries.current)) {
81
+ const payload = getPayload({ offset: n * size, limit: size })
82
+ const q = client.query(name, payload, { persistent })
83
+ queries.current[n] = q
84
+ }
85
+ if (n in unsubs.current) return
86
+ unsubs.current[n] = queries.current[n].subscribe(() => {
87
+ if (raf.current) return
88
+ raf.current = requestAnimationFrame(() => {
89
+ raf.current = null
90
+ setCheckum(
91
+ queries.current.reduce((combined, { cache }) => {
92
+ const checksum = cache?.checksum
93
+ return checksum ? combined + checksum : combined
94
+ }, 0)
95
+ )
96
+ })
97
+ })
98
+ })
99
+
100
+ // remove inactive subs
101
+ queries.current.forEach((_, n) => {
102
+ if (n in unsubs.current && !active.has(n)) {
103
+ unsubs.current[n]()
104
+ delete unsubs.current[n]
105
+ }
106
+ })
107
+
108
+ if (cache.current.checksum === checksum) {
109
+ return cache.current
110
+ }
111
+
112
+ let l = queries.current.length
113
+ cache.current.loading = false
114
+ cache.current.checksum = checksum
115
+
116
+ while (l--) {
117
+ const q = queries.current[l]
118
+ const m = size * l + size
119
+ let i = size * l
120
+
121
+ if (q) {
122
+ if (q.cache) {
123
+ let data = q.cache.value
124
+ if (data) {
125
+ for (const i of path) {
126
+ data = data[i]
127
+ }
128
+ for (let j = 0; i < m; i++) {
129
+ const item = data[j++]
130
+ if (!item) break
131
+ cache.current.items[i] = item
132
+ }
133
+ }
134
+ } else {
135
+ cache.current.loading = true
136
+ }
137
+ }
138
+
139
+ // fill up empty items with null
140
+ for (; i < m; i++) {
141
+ if (!(i in cache.current.items)) {
142
+ cache.current.items[i] = null
143
+ }
144
+ }
145
+ }
146
+ }
147
+
148
+ return cache.current
149
+ }
@@ -16,10 +16,37 @@ const counter: BasedQueryFunction<{ speed: number }, { cnt: number }> = (
16
16
  }
17
17
  }
18
18
 
19
+ const fakeDb = (_based, { offset, limit }, update) => {
20
+ let i
21
+ let cnt = 0
22
+ const timer = setTimeout(() => {
23
+ const doit = () => {
24
+ cnt++
25
+ const things = Array.from(Array(limit)).map((_, i) => {
26
+ return {
27
+ id: `${i + offset} - ${cnt}`,
28
+ }
29
+ })
30
+ update({ things })
31
+ }
32
+
33
+ i = setInterval(doit, 1e3)
34
+ doit()
35
+ }, 100)
36
+ return () => {
37
+ clearTimeout(timer)
38
+ clearInterval(i)
39
+ }
40
+ }
41
+
19
42
  const server = new BasedServer({
20
43
  port: 8081,
21
44
  functions: {
22
45
  configs: {
46
+ 'fake-db': {
47
+ type: 'query',
48
+ fn: fakeDb,
49
+ },
23
50
  counter: {
24
51
  type: 'query',
25
52
  fn: counter,
package/test/browser.tsx CHANGED
@@ -1,7 +1,7 @@
1
1
  import React, { useState } from 'react'
2
2
  import based from '@based/client'
3
3
  import { createRoot } from 'react-dom/client'
4
- import { Provider, useQuery, useLoading } from '../src'
4
+ import { Provider, useQuery, useLoading, useWindow } from '../src'
5
5
 
6
6
  const client = based({
7
7
  url: 'ws://localhost:8081',
@@ -37,10 +37,11 @@ const Tester = () => {
37
37
  height: 50,
38
38
  background: somethingIsLoading ? 'red' : 'blue',
39
39
  }}
40
- ></div>
41
- {payloads.map((v) => {
40
+ />
41
+ {payloads.map((v, i) => {
42
42
  return (
43
43
  <div
44
+ key={i}
44
45
  style={{
45
46
  border: '1px solid green',
46
47
  padding: 5,
@@ -53,12 +54,68 @@ const Tester = () => {
53
54
  )
54
55
  })}
55
56
  </div>
56
-
57
- <pre>{JSON.stringify(x, null, 2)}</pre>
58
57
  </div>
59
58
  )
60
59
  }
61
60
 
61
+ const UseWindowTester = () => {
62
+ const [pages, setPages] = useState([1, 2])
63
+ const [size, setSize] = useState(4)
64
+ const { items, loading } = useWindow(
65
+ 'fake-db',
66
+ ({ offset, limit }) => {
67
+ return {
68
+ offset,
69
+ limit,
70
+ }
71
+ },
72
+ {
73
+ path: ['things'], // where are the items in the response?
74
+ pages, // array of page numbers - starts at 1
75
+ size, // amount of items per page
76
+ }
77
+ )
78
+
79
+ return (
80
+ <>
81
+ <div style={{ display: 'flex' }}>
82
+ <pre>{JSON.stringify({ loading, items }, null, 2)}</pre>
83
+ <div>
84
+ {items.map((item, index) => {
85
+ return <div key={index}>{item?.id || '-'}</div>
86
+ })}
87
+ </div>
88
+ </div>
89
+ <button
90
+ style={{
91
+ background: 'lightgrey',
92
+ padding: 16,
93
+ }}
94
+ onClick={() => {
95
+ setPages(
96
+ pages.map((n) => {
97
+ return n + 3
98
+ })
99
+ )
100
+ }}
101
+ >
102
+ Move ({pages.join(',')})
103
+ </button>
104
+ <button
105
+ style={{
106
+ background: 'lightgrey',
107
+ padding: 16,
108
+ }}
109
+ onClick={() => {
110
+ setSize(size + 1)
111
+ }}
112
+ >
113
+ Resize ({size})
114
+ </button>
115
+ </>
116
+ )
117
+ }
118
+
62
119
  function App() {
63
120
  return (
64
121
  <div
@@ -68,6 +125,7 @@ function App() {
68
125
  >
69
126
  <Provider client={client}>
70
127
  <Tester />
128
+ <UseWindowTester />
71
129
  </Provider>
72
130
  </div>
73
131
  )