muya 2.0.6 → 2.0.7
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/cjs/index.js +1 -1
- package/esm/create-state.js +1 -1
- package/esm/select.js +1 -1
- package/esm/utils/common.js +1 -1
- package/package.json +1 -1
- package/src/__tests__/select.test.tsx +39 -3
- package/src/create-state.ts +1 -1
- package/src/select.ts +28 -4
- package/src/types.ts +1 -1
- package/src/utils/common.ts +1 -1
- package/types/types.d.ts +1 -1
- package/types/utils/common.d.ts +4 -0
package/cjs/index.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";var
|
|
1
|
+
"use strict";var I=Object.defineProperty;var M=Object.getOwnPropertyDescriptor;var j=Object.getOwnPropertyNames;var N=Object.prototype.hasOwnProperty;var _=(e,t)=>{for(var r in t)I(e,r,{get:t[r],enumerable:!0})},K=(e,t,r,n)=>{if(t&&typeof t=="object"||typeof t=="function")for(let o of j(t))!N.call(e,o)&&o!==r&&I(e,o,{get:()=>t[o],enumerable:!(n=M(t,o))||n.enumerable});return e};var Y=e=>K(I({},"__esModule",{value:!0}),e);var Z={};_(Z,{EMPTY_SELECTOR:()=>D,create:()=>R,isAbortError:()=>v,isArray:()=>k,isEqualBase:()=>T,isError:()=>G,isFunction:()=>b,isMap:()=>E,isPromise:()=>l,isSet:()=>g,isSetValueFunction:()=>A,isState:()=>Q,isUndefined:()=>m,select:()=>C,shallow:()=>L,useValue:()=>O});module.exports=Y(Z);var d=class extends Error{static Error="AbortError"};function J(e,t){t&&t.abort();let r=new AbortController,{signal:n}=r;return{promise:new Promise((s,i)=>{n.addEventListener("abort",()=>{i(new d)}),e.then(s).catch(i)}),controller:r}}function x(e,t=T){if(!m(e.current)){if(!m(e.previous)&&t(e.current,e.previous))return!1;e.previous=e.current}return!0}function p(e,t,r){if(!l(r))return r;e.abortController&&e.abortController.abort();let{promise:n,controller:o}=J(r,e.abortController);return e.abortController=o,n.then(s=>{e.current=s,t()}).catch(s=>{v(s)||(e.current=s,t())})}function l(e){return e instanceof Promise}function b(e){return typeof e=="function"}function E(e){return e instanceof Map}function g(e){return e instanceof Set}function k(e){return Array.isArray(e)}function T(e,t){return e===t?!0:!!Object.is(e,t)}function A(e){return typeof e=="function"}function v(e){return e instanceof d}function G(e){return e instanceof Error}function m(e){return e===void 0}function Q(e){return b(e)&&"get"in e&&"set"in e&&"isSet"in e&&e.isSet===!0}var D=e=>e;function q(){let e=new Map,t=new Set,r=performance.now(),n=!1;function o(){let i=performance.now(),a=i-r,{size:u}=t;if(a<.2&&u>0&&u<10){r=i,s();return}n||(n=!0,Promise.resolve().then(()=>{n=!1,r=performance.now(),s()}))}function s(){if(t.size===0)return;let i=new Set;for(let a of t){if(e.has(a.id)){i.add(a.id);let{onResolveItem:u}=e.get(a.id);u&&u(a.value)}t.delete(a)}if(t.size>0){o();return}for(let a of i)e.get(a)?.onFinish()}return{add(i,a){return e.set(i,a),()=>{e.delete(i)}},schedule(i,a){t.add({value:a,id:i}),o()}}}function C(e,t,r){let n={};function o(){let c=[],f=!1;for(let S of e){let y=S.get();l(y)&&(f=!0),c.push(y)}return f?new Promise((S,y)=>{Promise.all(c).then(U=>{if(U.some(H=>m(H)))return y(new d);let z=t(...U);S(z)})}):t(...c)}function s(){if(m(n.current)){let c=o();n.current=p(n,a.emitter.emit,c)}return n.current}let i=[];for(let c of e){let f=c.emitter.subscribe(()=>{h.schedule(a.id,null)});i.push(f)}let a=P({destroy(){for(let c of i)c();u(),a.emitter.clear(),n.current=void 0},get:s}),u=h.add(a.id,{onFinish(){let c=o();n.current=p(n,a.emitter.emit,c),x(n,r)&&a.emitter.emit()}});return a}var V=require("react");function O(e,t=D){let{emitter:r}=e,n=(0,V.useSyncExternalStore)(e.emitter.subscribe,()=>t(r.getSnapshot()),()=>t(r.getInitialSnapshot?r.getInitialSnapshot():r.getSnapshot()));if((0,V.useDebugValue)(n),l(n)||G(n))throw n;return n}function F(e,t){let r=new Set,n=[];return{clear:()=>{for(let o of n)o();r.clear()},subscribe:o=>(r.add(o),()=>{r.delete(o)}),emit:(...o)=>{for(let s of r)s(...o)},contains:o=>r.has(o),getSnapshot:e,getInitialSnapshot:t,getSize:()=>r.size,subscribeToOtherEmitter(o){let s=o.subscribe(()=>{this.emit()});n.push(s)}}}var W=0;function X(){return W++}function P(e){let{get:t,destroy:r,set:n}=e,o=!!n,s=function(i){return O(s,i)};return s.isSet=o,s.id=X(),s.emitter=F(t),s.destroy=r,s.listen=function(i){return this.emitter.subscribe(()=>{let a=t();l(a)||i(t())})},s.withName=function(i){return this.stateName=i,this},s.select=function(i,a=T){return C([s],i,a)},s.get=t,s.set=n,s}var h=q();function R(e,t=T){let r={};function n(){try{if(m(r.current)){let u=b(e)?e():e,c=p(r,i.emitter.emit,u);return r.current=c,r.current}return r.current}catch(u){r.current=u}return r.current}async function o(u,c){await u;let f=c(r.current),w=p(r,i.emitter.emit,f);r.current=w}function s(u){let c=n(),f=A(u);if(f&&l(c)){o(c,u);return}r.abortController&&r.abortController.abort();let w=f?u(c):u,S=p(r,i.emitter.emit,w);r.current=S}let i=P({get:n,destroy(){n(),a(),i.emitter.clear(),r.current=void 0},set(u){h.schedule(i.id,u)}}),a=h.add(i.id,{onFinish(){r.current=n(),x(r,t)&&i.emitter.emit()},onResolveItem:s});return b(e)||n(),i}function L(e,t){if(e==t)return!0;if(typeof e!="object"||e==null||typeof t!="object"||t==null)return!1;if(E(e)&&E(t)){if(e.size!==t.size)return!1;for(let[o,s]of e)if(!Object.is(s,t.get(o)))return!1;return!0}if(g(e)&&g(t)){if(e.size!==t.size)return!1;for(let o of e)if(!t.has(o))return!1;return!0}if(k(e)&&k(t)){if(e.length!==t.length)return!1;for(let[o,s]of e.entries())if(!Object.is(s,t[o]))return!1;return!0}let r=Object.keys(e),n=Object.keys(t);if(r.length!==n.length)return!1;for(let o of r)if(!Object.prototype.hasOwnProperty.call(t,o)||!Object.is(e[o],t[o]))return!1;return!0}
|
package/esm/create-state.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
import{select as u}from"./select";import{useValue as l}from"./use-value";import{createEmitter as S}from"./utils/create-emitter";import{isEqualBase as c,isPromise as m}from"./utils/is";let d=0;function f(){return d++}function
|
|
1
|
+
import{select as u}from"./select";import{useValue as l}from"./use-value";import{createEmitter as S}from"./utils/create-emitter";import{isEqualBase as c,isPromise as m}from"./utils/is";let d=0;function f(){return d++}function G(i){const{get:r,destroy:n,set:a}=i,o=!!a,t=function(e){return l(t,e)};return t.isSet=o,t.id=f(),t.emitter=S(r),t.destroy=n,t.listen=function(e){return this.emitter.subscribe(()=>{const s=r();m(s)||e(r())})},t.withName=function(e){return this.stateName=e,this},t.select=function(e,s=c){return u([t],e,s)},t.get=r,t.set=a,t}export{G as createState};
|
package/esm/select.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
import{stateScheduler as
|
|
1
|
+
import{stateScheduler as m}from"./create";import{createState as w}from"./create-state";import{subscribeToDevelopmentTools as b}from"./debug/development-tools";import{AbortError as k,canUpdate as v,handleAsyncUpdate as f}from"./utils/common";import{isPromise as A,isUndefined as d}from"./utils/is";function U(c,u,p){const t={};function a(){const e=[];let r=!1;for(const s of c){const o=s.get();A(o)&&(r=!0),e.push(o)}return r?new Promise((s,o)=>{Promise.all(e).then(l=>{if(l.some(y=>d(y)))return o(new k);const h=u(...l);s(h)})}):u(...e)}function S(){if(d(t.current)){const e=a();t.current=f(t,n.emitter.emit,e)}return t.current}const i=[];for(const e of c){const r=e.emitter.subscribe(()=>{m.schedule(n.id,null)});i.push(r)}const n=w({destroy(){for(const e of i)e();T(),n.emitter.clear(),t.current=void 0},get:S}),T=m.add(n.id,{onFinish(){const e=a();t.current=f(t,n.emitter.emit,e),v(t,p)&&n.emitter.emit()}});return b(n),n}export{U as select};
|
package/esm/utils/common.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
import{isAbortError as a,isEqualBase as b,isPromise as u,isUndefined as i}from"./is";class
|
|
1
|
+
import{isAbortError as a,isEqualBase as b,isPromise as u,isUndefined as i}from"./is";class p extends Error{static Error="AbortError"}function T(r,o){o&&o.abort();const t=new AbortController,{signal:n}=t;return{promise:new Promise((e,s)=>{n.addEventListener("abort",()=>{s(new p)}),r.then(e).catch(s)}),controller:t}}function f(r,o=b){if(!i(r.current)){if(!i(r.previous)&&o(r.current,r.previous))return!1;r.previous=r.current}return!0}function m(r,o,t){if(!u(t))return t;r.abortController&&r.abortController.abort();const{promise:n,controller:l}=T(t,r.abortController);return r.abortController=l,n.then(e=>{r.current=e,o()}).catch(e=>{a(e)||(r.current=e,o())})}export{p as AbortError,f as canUpdate,T as cancelablePromise,m as handleAsyncUpdate};
|
package/package.json
CHANGED
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import { create } from '../create'
|
|
2
2
|
import { select } from '../select'
|
|
3
|
-
import { waitFor } from '@testing-library/react'
|
|
3
|
+
import { renderHook, waitFor } from '@testing-library/react'
|
|
4
4
|
import { longPromise } from './test-utils'
|
|
5
|
+
import { Suspense } from 'react'
|
|
5
6
|
|
|
6
7
|
describe('select', () => {
|
|
7
8
|
it('should derive state from a single dependency', async () => {
|
|
@@ -101,7 +102,7 @@ describe('select', () => {
|
|
|
101
102
|
await longPromise(100)
|
|
102
103
|
return (await value) + 1
|
|
103
104
|
})
|
|
104
|
-
const selectedState2 = selectedState.select(async (value) =>
|
|
105
|
+
const selectedState2 = selectedState.select(async (value) => value + 1)
|
|
105
106
|
const listener = jest.fn()
|
|
106
107
|
selectedState2.listen(listener)
|
|
107
108
|
await waitFor(() => {
|
|
@@ -154,7 +155,7 @@ describe('select', () => {
|
|
|
154
155
|
it('should select state from async initial state', async () => {
|
|
155
156
|
const state = create(longPromise(100))
|
|
156
157
|
const selectedState = state.select(async (value) => {
|
|
157
|
-
return
|
|
158
|
+
return value + 2
|
|
158
159
|
})
|
|
159
160
|
await waitFor(() => {
|
|
160
161
|
expect(selectedState.get()).toBe(2)
|
|
@@ -169,4 +170,39 @@ describe('select', () => {
|
|
|
169
170
|
expect(selectedState.get()).toBe(2)
|
|
170
171
|
})
|
|
171
172
|
})
|
|
173
|
+
|
|
174
|
+
it('should select state from async state and do not change second time as it just boolean value', async () => {
|
|
175
|
+
const state = create(longPromise(100))
|
|
176
|
+
const selectedState = state.select((value) => {
|
|
177
|
+
const result = value > 0
|
|
178
|
+
expect(value).not.toBeUndefined()
|
|
179
|
+
return result
|
|
180
|
+
})
|
|
181
|
+
const render = jest.fn()
|
|
182
|
+
|
|
183
|
+
const { result } = renderHook(
|
|
184
|
+
() => {
|
|
185
|
+
render()
|
|
186
|
+
const value = selectedState()
|
|
187
|
+
return value
|
|
188
|
+
},
|
|
189
|
+
{ wrapper: ({ children }) => <Suspense fallback="loading">{children}</Suspense> },
|
|
190
|
+
)
|
|
191
|
+
|
|
192
|
+
await waitFor(() => {
|
|
193
|
+
expect(result.current).toBe(false)
|
|
194
|
+
expect(selectedState.get()).toBe(false)
|
|
195
|
+
// re-render twice, as it hit suspense, because value is not resolved yet
|
|
196
|
+
expect(render).toHaveBeenCalledTimes(2)
|
|
197
|
+
})
|
|
198
|
+
|
|
199
|
+
state.set(1)
|
|
200
|
+
|
|
201
|
+
await waitFor(() => {
|
|
202
|
+
expect(result.current).toBe(true)
|
|
203
|
+
expect(selectedState.get()).toBe(true)
|
|
204
|
+
// next time it re-render only once, as value is already resolved
|
|
205
|
+
expect(render).toHaveBeenCalledTimes(3)
|
|
206
|
+
})
|
|
207
|
+
})
|
|
172
208
|
})
|
package/src/create-state.ts
CHANGED
|
@@ -45,7 +45,7 @@ export function createState<T>(options: GetStateOptions<T>): FullState<T> {
|
|
|
45
45
|
return this
|
|
46
46
|
}
|
|
47
47
|
state.select = function (selector, isSelectorEqual = isEqualBase) {
|
|
48
|
-
return select([state], selector, isSelectorEqual)
|
|
48
|
+
return select([state as never], selector, isSelectorEqual)
|
|
49
49
|
}
|
|
50
50
|
state.get = get
|
|
51
51
|
state.set = set as State<T>['set']
|
package/src/select.ts
CHANGED
|
@@ -2,8 +2,8 @@ import { stateScheduler } from './create'
|
|
|
2
2
|
import { createState } from './create-state'
|
|
3
3
|
import { subscribeToDevelopmentTools } from './debug/development-tools'
|
|
4
4
|
import type { Cache, GetState, IsEqual } from './types'
|
|
5
|
-
import { canUpdate, handleAsyncUpdate } from './utils/common'
|
|
6
|
-
import { isUndefined } from './utils/is'
|
|
5
|
+
import { AbortError, canUpdate, handleAsyncUpdate } from './utils/common'
|
|
6
|
+
import { isPromise, isUndefined } from './utils/is'
|
|
7
7
|
|
|
8
8
|
type StateDependencies<T extends Array<unknown>> = {
|
|
9
9
|
[K in keyof T]: GetState<T[K]>
|
|
@@ -21,8 +21,32 @@ export function select<T = unknown, S extends Array<unknown> = []>(
|
|
|
21
21
|
const cache: Cache<T> = {}
|
|
22
22
|
|
|
23
23
|
function computedValue(): T {
|
|
24
|
-
const values = states.map((state) => state.get()) as S
|
|
25
|
-
|
|
24
|
+
// const values = states.map((state) => state.get()) as S
|
|
25
|
+
|
|
26
|
+
const values: unknown[] = []
|
|
27
|
+
let hasPromise = false
|
|
28
|
+
for (const state of states) {
|
|
29
|
+
const value = state.get()
|
|
30
|
+
if (isPromise(value)) {
|
|
31
|
+
hasPromise = true
|
|
32
|
+
}
|
|
33
|
+
values.push(value)
|
|
34
|
+
}
|
|
35
|
+
if (hasPromise) {
|
|
36
|
+
return new Promise((resolve, reject) => {
|
|
37
|
+
Promise.all(values).then((resolvedValues) => {
|
|
38
|
+
// check if some of value is undefined
|
|
39
|
+
// eslint-disable-next-line sonarjs/no-nested-functions
|
|
40
|
+
if (resolvedValues.some((element) => isUndefined(element))) {
|
|
41
|
+
return reject(new AbortError())
|
|
42
|
+
}
|
|
43
|
+
const resolved = selector(...(resolvedValues as S))
|
|
44
|
+
resolve(resolved)
|
|
45
|
+
})
|
|
46
|
+
}) as T
|
|
47
|
+
}
|
|
48
|
+
const result = selector(...(values as S))
|
|
49
|
+
return result
|
|
26
50
|
}
|
|
27
51
|
|
|
28
52
|
function getValue(): T {
|
package/src/types.ts
CHANGED
|
@@ -48,7 +48,7 @@ export interface GetState<T> {
|
|
|
48
48
|
* Select particular slice of the state.
|
|
49
49
|
* It will create "another" state in read-only mode (without set).
|
|
50
50
|
*/
|
|
51
|
-
select: <S>(selector: (state: Awaited<T>
|
|
51
|
+
select: <S>(selector: (state: Awaited<T>) => S, isEqual?: IsEqual<S>) => GetState<S>
|
|
52
52
|
}
|
|
53
53
|
|
|
54
54
|
export interface State<T> extends GetState<T> {
|
package/src/utils/common.ts
CHANGED
|
@@ -12,7 +12,7 @@ export class AbortError extends Error {
|
|
|
12
12
|
/**
|
|
13
13
|
* Cancelable promise function, return promise and controller
|
|
14
14
|
*/
|
|
15
|
-
function cancelablePromise<T>(promise: Promise<T>, previousController?: AbortController): CancelablePromise<T> {
|
|
15
|
+
export function cancelablePromise<T>(promise: Promise<T>, previousController?: AbortController): CancelablePromise<T> {
|
|
16
16
|
if (previousController) {
|
|
17
17
|
previousController.abort()
|
|
18
18
|
}
|
package/types/types.d.ts
CHANGED
|
@@ -45,7 +45,7 @@ export interface GetState<T> {
|
|
|
45
45
|
* Select particular slice of the state.
|
|
46
46
|
* It will create "another" state in read-only mode (without set).
|
|
47
47
|
*/
|
|
48
|
-
select: <S>(selector: (state: Awaited<T>
|
|
48
|
+
select: <S>(selector: (state: Awaited<T>) => S, isEqual?: IsEqual<S>) => GetState<S>;
|
|
49
49
|
}
|
|
50
50
|
export interface State<T> extends GetState<T> {
|
|
51
51
|
/**
|
package/types/utils/common.d.ts
CHANGED
|
@@ -6,6 +6,10 @@ export interface CancelablePromise<T> {
|
|
|
6
6
|
export declare class AbortError extends Error {
|
|
7
7
|
static readonly Error = "AbortError";
|
|
8
8
|
}
|
|
9
|
+
/**
|
|
10
|
+
* Cancelable promise function, return promise and controller
|
|
11
|
+
*/
|
|
12
|
+
export declare function cancelablePromise<T>(promise: Promise<T>, previousController?: AbortController): CancelablePromise<T>;
|
|
9
13
|
/**
|
|
10
14
|
* Check if the cache value is different from the previous value.
|
|
11
15
|
*/
|