ccstate-vue 3.0.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/CHANGELOG.md +10 -0
- package/babel.config.json +3 -0
- package/dist/LICENSE +21 -0
- package/dist/README.md +849 -0
- package/dist/index.cjs +56 -0
- package/dist/index.d.cts +12 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.js +51 -0
- package/dist/package.json +24 -0
- package/package.json +46 -0
- package/rollup.config.mjs +84 -0
- package/src/__tests__/get-set.test.ts +111 -0
- package/src/__tests__/loadable.test.ts +257 -0
- package/src/__tests__/memory.test.ts +92 -0
- package/src/__tests__/resolved.test.ts +116 -0
- package/src/index.ts +3 -0
- package/src/provider.ts +17 -0
- package/src/useGet.ts +30 -0
- package/src/useLoadable.ts +75 -0
- package/src/useResolved.ts +13 -0
- package/src/useSet.ts +22 -0
- package/tsconfig.json +7 -0
- package/vitest.config.ts +3 -0
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
// @vitest-environment happy-dom
|
|
2
|
+
import '@testing-library/jest-dom/vitest';
|
|
3
|
+
import LeakDetector from 'jest-leak-detector';
|
|
4
|
+
import { render, cleanup, screen } from '@testing-library/vue';
|
|
5
|
+
import { expect, it } from 'vitest';
|
|
6
|
+
import { computed, createStore, type Computed } from 'ccstate';
|
|
7
|
+
import { provideStore } from '../provider';
|
|
8
|
+
import { useGet } from '..';
|
|
9
|
+
import { useLastResolved } from '../useResolved';
|
|
10
|
+
import { delay } from 'signal-timers';
|
|
11
|
+
|
|
12
|
+
it('should release memory after view cleanup', async () => {
|
|
13
|
+
let base$:
|
|
14
|
+
| Computed<{
|
|
15
|
+
foo: string;
|
|
16
|
+
}>
|
|
17
|
+
| undefined = computed(() => {
|
|
18
|
+
return {
|
|
19
|
+
foo: 'bar',
|
|
20
|
+
};
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
const Component = {
|
|
24
|
+
setup() {
|
|
25
|
+
const base = useGet(
|
|
26
|
+
base$ as Computed<{
|
|
27
|
+
foo: string;
|
|
28
|
+
}>,
|
|
29
|
+
);
|
|
30
|
+
return { foo: base.value.foo };
|
|
31
|
+
},
|
|
32
|
+
template: `
|
|
33
|
+
<div>
|
|
34
|
+
<p>foo: {{ foo }}</p>
|
|
35
|
+
</div>
|
|
36
|
+
`,
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
const store = createStore();
|
|
40
|
+
const leakDetector = new LeakDetector(store.get(base$));
|
|
41
|
+
render({
|
|
42
|
+
components: { Component },
|
|
43
|
+
setup() {
|
|
44
|
+
provideStore(store);
|
|
45
|
+
},
|
|
46
|
+
template: `<div><Component /></div>`,
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
expect(screen.getByText('foo: bar')).toBeInTheDocument();
|
|
50
|
+
|
|
51
|
+
base$ = undefined;
|
|
52
|
+
cleanup();
|
|
53
|
+
|
|
54
|
+
expect(await leakDetector.isLeaking()).toBe(false);
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
it('should release memory for promise & loadable', async () => {
|
|
58
|
+
let base$: Computed<Promise<string>> | undefined = computed(() => {
|
|
59
|
+
return Promise.resolve('bar');
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
const Component = {
|
|
63
|
+
setup() {
|
|
64
|
+
if (!base$) {
|
|
65
|
+
return {};
|
|
66
|
+
}
|
|
67
|
+
const base = useLastResolved(base$);
|
|
68
|
+
return { base };
|
|
69
|
+
},
|
|
70
|
+
template: `
|
|
71
|
+
<div>
|
|
72
|
+
<p>{{ base }}</p>
|
|
73
|
+
</div>
|
|
74
|
+
`,
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
const store = createStore();
|
|
78
|
+
const leakDetector = new LeakDetector(store.get(base$));
|
|
79
|
+
render({
|
|
80
|
+
components: { Component },
|
|
81
|
+
setup() {
|
|
82
|
+
provideStore(store);
|
|
83
|
+
},
|
|
84
|
+
template: `<div><Component /></div>`,
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
base$ = undefined;
|
|
88
|
+
cleanup();
|
|
89
|
+
await delay(0);
|
|
90
|
+
|
|
91
|
+
expect(await leakDetector.isLeaking()).toBe(false);
|
|
92
|
+
});
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
// @vitest-environment happy-dom
|
|
2
|
+
import '@testing-library/jest-dom/vitest';
|
|
3
|
+
import { cleanup, render, screen } from '@testing-library/vue';
|
|
4
|
+
import { afterEach, expect, it } from 'vitest';
|
|
5
|
+
import { createStore, state } from 'ccstate';
|
|
6
|
+
import { provideStore } from '../provider';
|
|
7
|
+
import { useResolved, useLastResolved } from '../useResolved';
|
|
8
|
+
import { delay } from 'signal-timers';
|
|
9
|
+
|
|
10
|
+
afterEach(() => {
|
|
11
|
+
cleanup();
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
function makeDefered<T>(): {
|
|
15
|
+
resolve: (value: T) => void;
|
|
16
|
+
reject: (error: unknown) => void;
|
|
17
|
+
promise: Promise<T>;
|
|
18
|
+
} {
|
|
19
|
+
const deferred: {
|
|
20
|
+
resolve: (value: T) => void;
|
|
21
|
+
reject: (error: unknown) => void;
|
|
22
|
+
promise: Promise<T>;
|
|
23
|
+
} = {} as {
|
|
24
|
+
resolve: (value: T) => void;
|
|
25
|
+
reject: (error: unknown) => void;
|
|
26
|
+
promise: Promise<T>;
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
deferred.promise = new Promise((resolve, reject) => {
|
|
30
|
+
deferred.resolve = resolve;
|
|
31
|
+
deferred.reject = reject;
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
return deferred;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
it('convert promise to awaited value', async () => {
|
|
38
|
+
const base = state(Promise.resolve('foo'));
|
|
39
|
+
|
|
40
|
+
const Component = {
|
|
41
|
+
setup() {
|
|
42
|
+
const ret = useResolved(base);
|
|
43
|
+
return { ret };
|
|
44
|
+
},
|
|
45
|
+
template: `<div>{{ ret }}</div>`,
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
const store = createStore();
|
|
49
|
+
render({
|
|
50
|
+
components: { Component },
|
|
51
|
+
setup() {
|
|
52
|
+
provideStore(store);
|
|
53
|
+
},
|
|
54
|
+
template: `<div><Component /></div>`,
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
expect(await screen.findByText('foo')).toBeInTheDocument();
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it('loading state', async () => {
|
|
61
|
+
const deferred = makeDefered<string>();
|
|
62
|
+
const base = state(deferred.promise);
|
|
63
|
+
|
|
64
|
+
const Component = {
|
|
65
|
+
setup() {
|
|
66
|
+
const ret = useResolved(base);
|
|
67
|
+
return { ret };
|
|
68
|
+
},
|
|
69
|
+
template: `<div>{{ ret ?? 'loading' }}</div>`,
|
|
70
|
+
};
|
|
71
|
+
|
|
72
|
+
const store = createStore();
|
|
73
|
+
render({
|
|
74
|
+
components: { Component },
|
|
75
|
+
setup() {
|
|
76
|
+
provideStore(store);
|
|
77
|
+
},
|
|
78
|
+
template: `<div><Component /></div>`,
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
expect(await screen.findByText('loading')).toBeInTheDocument();
|
|
82
|
+
deferred.resolve('foo');
|
|
83
|
+
expect(await screen.findByText('foo')).toBeInTheDocument();
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
it('use lastResolved should not update when new promise pending', async () => {
|
|
87
|
+
const async$ = state(Promise.resolve(1));
|
|
88
|
+
|
|
89
|
+
const Component = {
|
|
90
|
+
setup() {
|
|
91
|
+
const number = useLastResolved(async$);
|
|
92
|
+
return { number };
|
|
93
|
+
},
|
|
94
|
+
template: `<div>num{{ number }}</div>`,
|
|
95
|
+
};
|
|
96
|
+
|
|
97
|
+
const store = createStore();
|
|
98
|
+
render({
|
|
99
|
+
components: { Component },
|
|
100
|
+
setup() {
|
|
101
|
+
provideStore(store);
|
|
102
|
+
},
|
|
103
|
+
template: `<div><Component /></div>`,
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
expect(await screen.findByText('num1')).toBeInTheDocument();
|
|
107
|
+
|
|
108
|
+
const defered = makeDefered();
|
|
109
|
+
store.set(async$, defered.promise);
|
|
110
|
+
|
|
111
|
+
await delay(0);
|
|
112
|
+
expect(screen.getByText('num1')).toBeInTheDocument();
|
|
113
|
+
defered.resolve(2);
|
|
114
|
+
await delay(0);
|
|
115
|
+
expect(screen.getByText('num2')).toBeInTheDocument();
|
|
116
|
+
});
|
package/src/index.ts
ADDED
package/src/provider.ts
ADDED
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { inject, provide, type InjectionKey } from 'vue';
|
|
2
|
+
import type { Store } from 'ccstate';
|
|
3
|
+
|
|
4
|
+
export const StoreKey = Symbol('ccstate-vue-store') as InjectionKey<Store>;
|
|
5
|
+
|
|
6
|
+
export const provideStore = (store: Store) => {
|
|
7
|
+
provide(StoreKey, store);
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
export const useStore = (): Store => {
|
|
11
|
+
const store = inject(StoreKey);
|
|
12
|
+
if (store === undefined) {
|
|
13
|
+
throw new Error('Store context not found - did you forget to wrap your app with StoreProvider?');
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
return store;
|
|
17
|
+
};
|
package/src/useGet.ts
ADDED
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { getCurrentInstance, onScopeDispose, shallowReadonly, shallowRef, type ShallowRef } from 'vue';
|
|
2
|
+
import { useStore } from './provider';
|
|
3
|
+
import { command, type Computed, type State } from 'ccstate';
|
|
4
|
+
|
|
5
|
+
export function useGet<Value>(atom: Computed<Value> | State<Value>): Readonly<ShallowRef<Value>> {
|
|
6
|
+
const store = useStore();
|
|
7
|
+
const initialValue = store.get(atom);
|
|
8
|
+
|
|
9
|
+
const vueState = shallowRef(initialValue);
|
|
10
|
+
|
|
11
|
+
const controller = new AbortController();
|
|
12
|
+
store.sub(
|
|
13
|
+
atom,
|
|
14
|
+
command(() => {
|
|
15
|
+
const nextValue = store.get(atom);
|
|
16
|
+
vueState.value = nextValue;
|
|
17
|
+
}),
|
|
18
|
+
{
|
|
19
|
+
signal: controller.signal,
|
|
20
|
+
},
|
|
21
|
+
);
|
|
22
|
+
|
|
23
|
+
if (getCurrentInstance()) {
|
|
24
|
+
onScopeDispose(() => {
|
|
25
|
+
controller.abort();
|
|
26
|
+
});
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
return shallowReadonly(vueState);
|
|
30
|
+
}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { useGet } from './useGet';
|
|
2
|
+
import type { Computed, State } from 'ccstate';
|
|
3
|
+
import { shallowReadonly, shallowRef, watch, type ShallowRef } from 'vue';
|
|
4
|
+
|
|
5
|
+
type Loadable<T> =
|
|
6
|
+
| {
|
|
7
|
+
state: 'loading';
|
|
8
|
+
}
|
|
9
|
+
| {
|
|
10
|
+
state: 'hasData';
|
|
11
|
+
data: T;
|
|
12
|
+
}
|
|
13
|
+
| {
|
|
14
|
+
state: 'hasError';
|
|
15
|
+
error: unknown;
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
function useLoadableInternal<T>(
|
|
19
|
+
atom: State<Promise<T>> | Computed<Promise<T>>,
|
|
20
|
+
keepLastResolved: boolean,
|
|
21
|
+
): Readonly<ShallowRef<Loadable<T>>> {
|
|
22
|
+
const promise = useGet(atom);
|
|
23
|
+
const loadable = shallowRef<Loadable<T>>({
|
|
24
|
+
state: 'loading',
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
watch(
|
|
28
|
+
promise,
|
|
29
|
+
(promiseValue, _, onCleanup) => {
|
|
30
|
+
const ctrl = new AbortController();
|
|
31
|
+
onCleanup(() => {
|
|
32
|
+
ctrl.abort();
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
if (!keepLastResolved) {
|
|
36
|
+
loadable.value = {
|
|
37
|
+
state: 'loading',
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
void promiseValue
|
|
42
|
+
.then((ret) => {
|
|
43
|
+
if (ctrl.signal.aborted) return;
|
|
44
|
+
|
|
45
|
+
loadable.value = {
|
|
46
|
+
state: 'hasData',
|
|
47
|
+
data: ret,
|
|
48
|
+
};
|
|
49
|
+
})
|
|
50
|
+
.catch(() => void 0);
|
|
51
|
+
|
|
52
|
+
void promiseValue.catch((error: unknown) => {
|
|
53
|
+
if (ctrl.signal.aborted) return;
|
|
54
|
+
|
|
55
|
+
loadable.value = {
|
|
56
|
+
state: 'hasError',
|
|
57
|
+
error,
|
|
58
|
+
};
|
|
59
|
+
});
|
|
60
|
+
},
|
|
61
|
+
{
|
|
62
|
+
immediate: true,
|
|
63
|
+
},
|
|
64
|
+
);
|
|
65
|
+
|
|
66
|
+
return shallowReadonly(loadable);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
export function useLoadable<T>(atom: State<Promise<T>> | Computed<Promise<T>>): Readonly<ShallowRef<Loadable<T>>> {
|
|
70
|
+
return useLoadableInternal(atom, false);
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
export function useLastLoadable<T>(atom: State<Promise<T>> | Computed<Promise<T>>): Readonly<ShallowRef<Loadable<T>>> {
|
|
74
|
+
return useLoadableInternal(atom, true);
|
|
75
|
+
}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { useLastLoadable, useLoadable } from './useLoadable';
|
|
2
|
+
import type { Computed, State } from 'ccstate';
|
|
3
|
+
import { computed, type ComputedRef } from 'vue';
|
|
4
|
+
|
|
5
|
+
export function useResolved<T>(atom: State<Promise<T>> | Computed<Promise<T>>): ComputedRef<T | undefined> {
|
|
6
|
+
const loadable = useLoadable(atom);
|
|
7
|
+
return computed(() => (loadable.value.state === 'hasData' ? loadable.value.data : undefined));
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export function useLastResolved<T>(atom: State<Promise<T>> | Computed<Promise<T>>): ComputedRef<T | undefined> {
|
|
11
|
+
const loadable = useLastLoadable(atom);
|
|
12
|
+
return computed(() => (loadable.value.state === 'hasData' ? loadable.value.data : undefined));
|
|
13
|
+
}
|
package/src/useSet.ts
ADDED
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { useStore } from './provider';
|
|
2
|
+
import { type Command, type State, type Updater } from 'ccstate';
|
|
3
|
+
|
|
4
|
+
export function useSet<T>(atom: State<T>): (value: T | Updater<T>) => void;
|
|
5
|
+
export function useSet<T, ARGS extends unknown[]>(atom: Command<T, ARGS>): (...args: ARGS) => T;
|
|
6
|
+
export function useSet<T, ARGS extends unknown[]>(
|
|
7
|
+
atom: State<T> | Command<T, ARGS>,
|
|
8
|
+
): ((value: T | Updater<T>) => void) | ((...args: ARGS) => T) {
|
|
9
|
+
const store = useStore();
|
|
10
|
+
|
|
11
|
+
if ('write' in atom) {
|
|
12
|
+
return (...args: ARGS): T => {
|
|
13
|
+
const ret = store.set(atom, ...args);
|
|
14
|
+
|
|
15
|
+
return ret;
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
return (value: T | Updater<T>) => {
|
|
20
|
+
store.set(atom, value);
|
|
21
|
+
};
|
|
22
|
+
}
|
package/tsconfig.json
ADDED
package/vitest.config.ts
ADDED