diff --git a/CHANGELOG.md b/CHANGELOG.md index 652d30621b..17fb7c962e 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,23 +4,17 @@ **_Add new changes here as they land_** -### Pending - -- Memory management -- useTransition() compatibility for OSS - ## 0.6 () - React 18 - Leverage new React 18 APIs for improved safety and optimizations. (#1488) - Fixes for `` (#1473, #1444, #1509). - - `useTransition()` is not yet supported for open source React. + - Experimental support for `useTransition()` using hooks with `_TRANSITION_SUPPORT_UNSTABLE` suffix. (#1572, #1560) - Recoil updates now re-render earlier: - Recoil and React state changes from the same batch now stay in sync. (#1076) - Renders now occur before transaction observers instead of after. ### New Features - - Add `refresh()` to the `useRecoilCallback()` interface for refreshing selector caches. (#1413) - Callbacks from selector's `getCallback()` can now mutate, refresh, and transact Recoil state, in addition to reading it, for parity with `useRecoilCallback()`. (#1498) - Recoil StoreID's for `` and `Snapshot` stores accessible via `useRecoilStoreID()` hook (#1417) or `storeID` parameter for atom effects (#1414). diff --git a/packages/recoil/Recoil_index.js b/packages/recoil/Recoil_index.js index 12a712353d..c93a2361e3 100644 --- a/packages/recoil/Recoil_index.js +++ b/packages/recoil/Recoil_index.js @@ -49,9 +49,12 @@ const {retentionZone} = require('./core/Recoil_RetentionZone'); const {freshSnapshot} = require('./core/Recoil_Snapshot'); const { useRecoilState, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, useRecoilStateLoadable, useRecoilValue, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, useRecoilValueLoadable, + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE, useResetRecoilState, useSetRecoilState, } = require('./hooks/Recoil_Hooks'); @@ -119,6 +122,9 @@ module.exports = { useResetRecoilState, useGetRecoilValueInfo_UNSTABLE: useGetRecoilValueInfo, useRecoilRefresher_UNSTABLE: useRecoilRefresher, + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, // Hooks for complex operations useRecoilCallback, diff --git a/packages/recoil/core/Recoil_ReactMode.js b/packages/recoil/core/Recoil_ReactMode.js index 8c82d32e9f..256f4fda7b 100644 --- a/packages/recoil/core/Recoil_ReactMode.js +++ b/packages/recoil/core/Recoil_ReactMode.js @@ -42,7 +42,7 @@ const useSyncExternalStore: ( (React: any).unstable_useSyncExternalStore; type ReactMode = - | 'CONCURRENT_SUPPORT' + | 'TRANSITION_SUPPORT' | 'SYNC_EXTERNAL_STORE' | 'MUTABLE_SOURCE' | 'LEGACY'; @@ -53,13 +53,13 @@ type ReactMode = * 1) earlier * 2) in sync with React updates in the same batch * 3) before transaction observers instead of after. - * concurrent: Is the current mode compatible with Concurrent Mode (i.e. useTransition()) + * concurrent: Is the current mode compatible with Concurrent Mode and useTransition() */ function reactMode(): {mode: ReactMode, early: boolean, concurrent: boolean} { // NOTE: This mode is currently broken with some Suspense cases // see Recoil_selector-test.js - if (gkx('recoil_concurrent_support')) { - return {mode: 'CONCURRENT_SUPPORT', early: true, concurrent: true}; + if (gkx('recoil_transition_support')) { + return {mode: 'TRANSITION_SUPPORT', early: true, concurrent: true}; } if (gkx('recoil_sync_external_store') && useSyncExternalStore != null) { diff --git a/packages/recoil/hooks/Recoil_Hooks.js b/packages/recoil/hooks/Recoil_Hooks.js index 7edc3e2e1c..7ffe31baed 100644 --- a/packages/recoil/hooks/Recoil_Hooks.js +++ b/packages/recoil/hooks/Recoil_Hooks.js @@ -42,6 +42,7 @@ const differenceSets = require('recoil-shared/util/Recoil_differenceSets'); const err = require('recoil-shared/util/Recoil_err'); const expectationViolation = require('recoil-shared/util/Recoil_expectationViolation'); const gkx = require('recoil-shared/util/Recoil_gkx'); +const recoverableViolation = require('recoil-shared/util/Recoil_recoverableViolation'); const useComponentName = require('recoil-shared/util/Recoil_useComponentName'); function handleLoadable( @@ -421,7 +422,7 @@ function useRecoilValueLoadable_MUTABLE_SOURCE( return loadable; } -function useRecoilValueLoadable_CONCURRENT_SUPPORT( +function useRecoilValueLoadable_TRANSITION_SUPPORT( recoilValue: RecoilValue, ): Loadable { const storeRef = useStoreRef(); @@ -578,7 +579,7 @@ function useRecoilValueLoadable(recoilValue: RecoilValue): Loadable { useRetain(recoilValue); } return { - CONCURRENT_SUPPORT: useRecoilValueLoadable_CONCURRENT_SUPPORT, + TRANSITION_SUPPORT: useRecoilValueLoadable_TRANSITION_SUPPORT, SYNC_EXTERNAL_STORE: useRecoilValueLoadable_SYNC_EXTERNAL_STORE, MUTABLE_SOURCE: useRecoilValueLoadable_MUTABLE_SOURCE, LEGACY: useRecoilValueLoadable_LEGACY, @@ -679,6 +680,62 @@ function useSetUnvalidatedAtomValues(): ( }; } +/** + * Experimental variants of hooks with support for useTransition() + */ + +function useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE( + recoilValue: RecoilValue, +): Loadable { + if (__DEV__) { + validateRecoilValue( + recoilValue, + 'useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE', + ); + if (!reactMode().early) { + recoverableViolation( + 'Attepmt to use a hook with UNSTABLE_TRANSITION_SUPPORT in a rendering mode incompatible with concurrent rendering. Try enabling the recoil_sync_external_store or recoil_transition_support GKs.', + 'recoil', + ); + } + } + if (gkx('recoil_memory_managament_2020')) { + // eslint-disable-next-line fb-www/react-hooks + useRetain(recoilValue); + } + return useRecoilValueLoadable_TRANSITION_SUPPORT(recoilValue); +} + +function useRecoilValue_TRANSITION_SUPPORT_UNSTABLE( + recoilValue: RecoilValue, +): T { + if (__DEV__) { + validateRecoilValue( + recoilValue, + 'useRecoilValue_TRANSITION_SUPPORT_UNSTABLE', + ); + } + const storeRef = useStoreRef(); + const loadable = + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE(recoilValue); + return handleLoadable(loadable, recoilValue, storeRef); +} + +function useRecoilState_TRANSITION_SUPPORT_UNSTABLE( + recoilState: RecoilState, +): [T, SetterOrUpdater] { + if (__DEV__) { + validateRecoilValue( + recoilState, + 'useRecoilState_TRANSITION_SUPPORT_UNSTABLE', + ); + } + return [ + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(recoilState), + useSetRecoilState(recoilState), + ]; +} + module.exports = { recoilComponentGetRecoilValueCount_FOR_TESTING, useRecoilInterface: useRecoilInterface_DEPRECATED, @@ -689,4 +746,7 @@ module.exports = { useResetRecoilState, useSetRecoilState, useSetUnvalidatedAtomValues, + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, }; diff --git a/packages/recoil/hooks/__tests__/Recoil_Hooks_TRANSITION_SUPPORT_UNSTABLE-test.js b/packages/recoil/hooks/__tests__/Recoil_Hooks_TRANSITION_SUPPORT_UNSTABLE-test.js new file mode 100644 index 0000000000..ec15bdcfac --- /dev/null +++ b/packages/recoil/hooks/__tests__/Recoil_Hooks_TRANSITION_SUPPORT_UNSTABLE-test.js @@ -0,0 +1,148 @@ +/** + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + * + * @emails oncall+recoil + * @flow strict-local + * @format + */ +'use strict'; + +// Sanity tests for *_TRANSITION_SUPPORT_UNSTABLE() hooks. The actual tests +// for useTransition() support are in Recoil_useTransition-test.js + +const { + getRecoilTestFn, +} = require('recoil-shared/__test_utils__/Recoil_TestingUtils'); + +let React, + act, + selector, + stringAtom, + asyncSelector, + flushPromisesAndTimers, + renderElements, + useRecoilState, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, + useRecoilValue, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, + useRecoilValueLoadable, + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE, + useSetRecoilState, + reactMode; + +const testRecoil = getRecoilTestFn(() => { + React = require('react'); + ({act} = require('ReactTestUtils')); + + selector = require('../../recoil_values/Recoil_selector'); + ({ + stringAtom, + asyncSelector, + flushPromisesAndTimers, + renderElements, + } = require('recoil-shared/__test_utils__/Recoil_TestingUtils')); + ({reactMode} = require('../../core/Recoil_ReactMode')); + ({ + useRecoilState, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, + useRecoilValue, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, + useRecoilValueLoadable, + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE, + useSetRecoilState, + } = require('../Recoil_Hooks')); +}); + +testRecoil('useRecoilValue_TRANSITION_SUPPORT_UNSTABLE', async () => { + if (!reactMode().early) { + return; + } + const myAtom = stringAtom(); + const [mySelector, resolve] = asyncSelector(); + let setAtom; + function Component() { + setAtom = useSetRecoilState(myAtom); + return [ + useRecoilValue(myAtom), + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(myAtom), + useRecoilValue(mySelector), + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(mySelector), + ].join(' '); + } + const c = renderElements(); + expect(c.textContent).toBe('loading'); + + act(() => resolve('RESOLVE')); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('DEFAULT DEFAULT RESOLVE RESOLVE'); + + act(() => setAtom('SET')); + expect(c.textContent).toBe('SET SET RESOLVE RESOLVE'); +}); + +testRecoil('useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE', async () => { + if (!reactMode().early) { + return; + } + const myAtom = stringAtom(); + const [mySelector, resolve] = asyncSelector(); + let setAtom; + function Component() { + setAtom = useSetRecoilState(myAtom); + return [ + useRecoilValueLoadable(myAtom).getValue(), + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE(myAtom).getValue(), + useRecoilValueLoadable(mySelector).getValue(), + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE(mySelector).getValue(), + ].join(' '); + } + const c = renderElements(); + expect(c.textContent).toBe('loading'); + + act(() => resolve('RESOLVE')); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('DEFAULT DEFAULT RESOLVE RESOLVE'); + + act(() => setAtom('SET')); + expect(c.textContent).toBe('SET SET RESOLVE RESOLVE'); +}); + +testRecoil('useRecoilState_TRANSITION_SUPPORT_UNSTABLE', async () => { + if (!reactMode().early) { + return; + } + const myAtom = stringAtom(); + const [myAsyncSelector, resolve] = asyncSelector(); + const mySelector = selector({ + key: 'useRecoilState_TRANSITION_SUPPORT_UNSTABLE selector', + get: () => myAsyncSelector, + set: ({set}, newValue) => set(myAtom, newValue), + }); + let setAtom, setSelector; + function Component() { + const [v1] = useRecoilState(myAtom); + const [v2, setAtomValue] = + useRecoilState_TRANSITION_SUPPORT_UNSTABLE(myAtom); + setAtom = setAtomValue; + const [v3] = useRecoilState(mySelector); + const [v4, setSelectorValue] = + useRecoilState_TRANSITION_SUPPORT_UNSTABLE(mySelector); + setSelector = setSelectorValue; + return [v1, v2, v3, v4].join(' '); + } + const c = renderElements(); + expect(c.textContent).toBe('loading'); + + act(() => resolve('RESOLVE')); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('DEFAULT DEFAULT RESOLVE RESOLVE'); + + act(() => setAtom('SET')); + expect(c.textContent).toBe('SET SET RESOLVE RESOLVE'); + + act(() => setSelector('SETS')); + expect(c.textContent).toBe('SETS SETS RESOLVE RESOLVE'); +}); diff --git a/packages/recoil/hooks/__tests__/Recoil_PublicHooks-test.js b/packages/recoil/hooks/__tests__/Recoil_PublicHooks-test.js index f39cfbe05b..48e6b3032b 100644 --- a/packages/recoil/hooks/__tests__/Recoil_PublicHooks-test.js +++ b/packages/recoil/hooks/__tests__/Recoil_PublicHooks-test.js @@ -458,7 +458,7 @@ describe('Render counts', () => { if ( (reactMode().mode === 'LEGACY' && !gks.includes('recoil_suppress_rerender_in_callback')) || - reactMode().mode === 'CONCURRENT_SUPPORT' + reactMode().mode === 'TRANSITION_SUPPORT' ) { baseCalls += 1; } @@ -579,7 +579,7 @@ describe('Component Subscriptions', () => { if ( (reactMode().mode === 'LEGACY' && !gks.includes('recoil_suppress_rerender_in_callback')) || - reactMode().mode === 'CONCURRENT_SUPPORT' + reactMode().mode === 'TRANSITION_SUPPORT' ) { baseCalls += 1; } diff --git a/packages/recoil/hooks/__tests__/Recoil_useTransition-test.js b/packages/recoil/hooks/__tests__/Recoil_useTransition-test.js index 8e4f5f5a7c..6f392c0374 100644 --- a/packages/recoil/hooks/__tests__/Recoil_useTransition-test.js +++ b/packages/recoil/hooks/__tests__/Recoil_useTransition-test.js @@ -16,19 +16,31 @@ const { } = require('recoil-shared/__test_utils__/Recoil_TestingUtils'); let React, - act, + useState, useTransition, + act, + useRecoilValue, useRecoilState, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, atom, + selectorFamily, renderElements, reactMode, flushPromisesAndTimers; const testRecoil = getRecoilTestFn(() => { React = require('react'); - ({useTransition} = React); + ({useState, useTransition} = React); ({act} = require('ReactTestUtils')); - ({useRecoilState, atom} = require('../../Recoil_index')); + ({ + useRecoilValue, + useRecoilState, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, + atom, + selectorFamily, + } = require('../../Recoil_index')); ({ renderElements, flushPromisesAndTimers, @@ -141,3 +153,234 @@ testRecoil('Works with useTransition', async ({concurrentMode}) => { await flushPromisesAndTimers(); expect(c.textContent).toEqual('Index: 3 - Item 3 = v3'); }); + +testRecoil('useRecoilValue()', async ({concurrentMode}) => { + const myAtom = atom({key: 'useRecoilValue atom', default: 0}); + let resolvers = []; + function resolveSelectors() { + resolvers.forEach(resolve => resolve('RESOLVED')); + resolvers = []; + } + const query = selectorFamily({ + key: 'useRecoilValue selector', + get: + param => + ({get}) => { + const value = get(myAtom); + return new Promise(resolve => { + resolvers.push(resolve); + }).then(str => `${param} ${value} ${str}`); + }, + }); + + function Component({index}) { + const value = useRecoilValue(query(index)); + return ( + <> + {index} {value} + + ); + } + + let startReactTransition, startRecoilTransition, startBothTransition; + function Main() { + const [reactState, setReactState] = useState(0); + const [recoilState, setRecoilState] = useRecoilState(myAtom); + const [inTransition, startTransition] = useTransition(); + startReactTransition = () => { + startTransition(() => { + setReactState(x => x + 1); + }); + }; + startRecoilTransition = () => { + startTransition(() => { + setRecoilState(x => x + 1); + }); + }; + startBothTransition = () => { + startTransition(() => { + setReactState(x => x + 1); + setRecoilState(x => x + 1); + }); + }; + return ( + <> + React:{reactState} Recoil:{recoilState}{' '} + {inTransition ? '[IN TRANSITION] ' : ''}|{' '} + + + + + ); + } + + const c = renderElements(
); + expect(c.textContent).toBe('React:0 Recoil:0 | LOADING'); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:0 Recoil:0 | 0 0 0 RESOLVED'); + + // Transition changing React State + act(startReactTransition); + expect(c.textContent).toBe( + concurrentMode + ? 'React:0 Recoil:0 [IN TRANSITION] | 0 0 0 RESOLVED' + : 'React:1 Recoil:0 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:1 Recoil:0 | 1 1 0 RESOLVED'); + + // Transition changing Recoil State + act(startRecoilTransition); + expect(c.textContent).toBe( + concurrentMode && reactMode().concurrent + ? 'React:1 Recoil:0 [IN TRANSITION] | 1 1 0 RESOLVED' + : 'React:1 Recoil:1 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:1 Recoil:1 | 1 1 1 RESOLVED'); + + // Second transition changing Recoil State + act(startRecoilTransition); + expect(c.textContent).toBe( + concurrentMode && reactMode().concurrent + ? 'React:1 Recoil:1 [IN TRANSITION] | 1 1 1 RESOLVED' + : 'React:1 Recoil:2 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:1 Recoil:2 | 1 1 2 RESOLVED'); + + // Transition with both React and Recoil state + act(startBothTransition); + expect(c.textContent).toBe( + concurrentMode && + (reactMode().concurrent || reactMode().mode === 'MUTABLE_SOURCE') + ? 'React:1 Recoil:2 [IN TRANSITION] | 1 1 2 RESOLVED' + : 'React:2 Recoil:3 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:2 Recoil:3 | 2 2 3 RESOLVED'); +}); + +testRecoil( + 'useRecoilValue_TRANSITION_SUPPORT_UNSTABLE()', + async ({concurrentMode}) => { + const myAtom = atom({key: 'TRANSITION_SUPPORT_UNSTABLE atom', default: 0}); + let resolvers = []; + function resolveSelectors() { + resolvers.forEach(resolve => resolve('RESOLVED')); + resolvers = []; + } + const query = selectorFamily({ + key: 'TRANSITION_SUPPORT_UNSTABLE selector', + get: + param => + ({get}) => { + const value = get(myAtom); + return new Promise(resolve => { + resolvers.push(resolve); + }).then(str => `${param} ${value} ${str}`); + }, + }); + + function Component({index}) { + const value = useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(query(index)); + return ( + <> + {index} {value} + + ); + } + + let startReactTransition, startRecoilTransition, startBothTransition; + function Main() { + const [reactState, setReactState] = useState(0); + const [recoilState, setRecoilState] = + useRecoilState_TRANSITION_SUPPORT_UNSTABLE(myAtom); + const [inTransition, startTransition] = useTransition(); + startReactTransition = () => { + startTransition(() => { + setReactState(x => x + 1); + }); + }; + startRecoilTransition = () => { + startTransition(() => { + setRecoilState(x => x + 1); + }); + }; + startBothTransition = () => { + startTransition(() => { + setReactState(x => x + 1); + setRecoilState(x => x + 1); + }); + }; + return ( + <> + React:{reactState} Recoil:{recoilState}{' '} + {inTransition ? '[IN TRANSITION] ' : ''}|{' '} + + + + + ); + } + + const c = renderElements(
); + expect(c.textContent).toBe('React:0 Recoil:0 | LOADING'); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:0 Recoil:0 | 0 0 0 RESOLVED'); + + // Transition changing React State + act(startReactTransition); + expect(c.textContent).toBe( + concurrentMode + ? 'React:0 Recoil:0 [IN TRANSITION] | 0 0 0 RESOLVED' + : 'React:1 Recoil:0 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:1 Recoil:0 | 1 1 0 RESOLVED'); + + // Transition changing Recoil State + act(startRecoilTransition); + expect(c.textContent).toBe( + concurrentMode && reactMode().early + ? 'React:1 Recoil:0 [IN TRANSITION] | 1 1 0 RESOLVED' + : 'React:1 Recoil:1 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:1 Recoil:1 | 1 1 1 RESOLVED'); + + // Second transition changing Recoil State + act(startRecoilTransition); + expect(c.textContent).toBe( + concurrentMode && reactMode().early + ? 'React:1 Recoil:1 [IN TRANSITION] | 1 1 1 RESOLVED' + : 'React:1 Recoil:2 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:1 Recoil:2 | 1 1 2 RESOLVED'); + + // Transition with both React and Recoil State + act(startBothTransition); + expect(c.textContent).toBe( + concurrentMode + ? 'React:1 Recoil:2 [IN TRANSITION] | 1 1 2 RESOLVED' + : 'React:2 Recoil:3 | LOADING', + ); + act(resolveSelectors); + await flushPromisesAndTimers(); + act(resolveSelectors); + await flushPromisesAndTimers(); + expect(c.textContent).toBe('React:2 Recoil:3 | 2 2 3 RESOLVED'); + }, +); diff --git a/packages/shared/__test_utils__/Recoil_TestingUtils.js b/packages/shared/__test_utils__/Recoil_TestingUtils.js index bdff0e6f77..648ab02208 100644 --- a/packages/shared/__test_utils__/Recoil_TestingUtils.js +++ b/packages/shared/__test_utils__/Recoil_TestingUtils.js @@ -18,6 +18,7 @@ import type {RecoilState, RecoilValue, RecoilValueReadOnly} from 'Recoil'; const {act} = require('ReactTestUtils'); const { RecoilRoot, + atom, selector, useRecoilValue, useResetRecoilState, @@ -202,6 +203,10 @@ function renderElementsWithSuspenseCount( //////////////////////////////////////// let id = 0; +function stringAtom(): RecoilState { + return atom({key: `StringAtom-${id++}`, default: 'DEFAULT'}); +} + const errorThrowingAsyncSelector: ( string, ?RecoilValue, @@ -259,7 +264,11 @@ function asyncSelector( // Useful Components for testing ////////////////////////////////// -function ReadsAtom({atom}: {atom: RecoilValue}): React.Node { +function ReadsAtom({ + atom, // eslint-disable-line no-shadow +}: { + atom: RecoilValue, +}): React.Node { return stableStringify(useRecoilValue(atom)); } @@ -269,14 +278,14 @@ function ReadsAtom({atom}: {atom: RecoilValue}): React.Node { // resetValue() // ] function componentThatReadsAndWritesAtom( - atom: RecoilState, + recoilState: RecoilState, ): [() => React.Node, (T) => void, () => void] { let setValue; let resetValue; const ReadsAndWritesAtom = (): React.Node => { - setValue = useSetRecoilState(atom); - resetValue = useResetRecoilState(atom); - return stableStringify(useRecoilValue(atom)); + setValue = useSetRecoilState(recoilState); + resetValue = useResetRecoilState(recoilState); + return stableStringify(useRecoilValue(recoilState)); }; return [ ReadsAndWritesAtom, @@ -409,7 +418,7 @@ const WWW_GKS_TO_TEST = QUICK_TEST 'recoil_release_on_cascading_update_killswitch_2021', ], // Experimental mode for useTransition() support: - ['recoil_hamt_2020', 'recoil_concurrent_support'], + ['recoil_hamt_2020', 'recoil_transition_support'], ]; /** @@ -449,6 +458,7 @@ module.exports = { renderElementsWithSuspenseCount, ReadsAtom, componentThatReadsAndWritesAtom, + stringAtom, errorThrowingAsyncSelector, resolvingAsyncSelector, loadingAsyncSelector, diff --git a/typescript/index.d.ts b/typescript/index.d.ts index 4f3029152a..e03c1c5d68 100644 --- a/typescript/index.d.ts +++ b/typescript/index.d.ts @@ -240,6 +240,14 @@ */ export function useGetRecoilValueInfo_UNSTABLE(): (recoilValue: RecoilValue) => RecoilStateInfo; +/** + * Experimental version of hooks for useTransition() support + */ + export function useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(recoilValue: RecoilValue): T; + export function useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE(recoilValue: RecoilValue): Loadable; + export function useRecoilState_TRANSITION_SUPPORT_UNSTABLE(recoilState: RecoilState): [T, SetterOrUpdater]; + + /** * Returns a function that will run the callback that was passed when * calling this hook. Useful for accessing Recoil state in response to diff --git a/typescript/tests.ts b/typescript/tests.ts index 827d191cd6..45cb2db867 100644 --- a/typescript/tests.ts +++ b/typescript/tests.ts @@ -30,6 +30,9 @@ useRecoilTransaction_UNSTABLE, useRecoilRefresher_UNSTABLE, useRecoilStoreID, + useRecoilValue_TRANSITION_SUPPORT_UNSTABLE, + useRecoilState_TRANSITION_SUPPORT_UNSTABLE, + useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE, } from 'recoil'; /* eslint-disable @typescript-eslint/no-explicit-any */ @@ -239,6 +242,23 @@ useGetRecoilValueInfo_UNSTABLE()(myAtom); // $ExpectType RecoilStateInfo useGetRecoilValueInfo_UNSTABLE()(mySelector2); // $ExpectType RecoilStateInfo useGetRecoilValueInfo_UNSTABLE()({}); // $ExpectError + +useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(roAtom); // $ExpectType string +useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(waAtom); // $ExpectType string +useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(myAtom); // $ExpectType number +useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(mySelector1); // $ExpectType number +useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(readOnlySelectorSel); // $ExpectType number +useRecoilValue_TRANSITION_SUPPORT_UNSTABLE(writeableSelector); // $ExpectType number +useRecoilValue_TRANSITION_SUPPORT_UNSTABLE({}); // $ExpectError +useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE(myAtom); // $ExpectType Loadable +useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE(readOnlySelectorSel); // $ExpectType Loadable +useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE(writeableSelector); // $ExpectType Loadable +useRecoilValueLoadable_TRANSITION_SUPPORT_UNSTABLE({}); // $ExpectError +useRecoilState_TRANSITION_SUPPORT_UNSTABLE(myAtom); // $ExpectType [number, SetterOrUpdater] +useRecoilState_TRANSITION_SUPPORT_UNSTABLE(writeableSelector); // $ExpectType [number, SetterOrUpdater] +useRecoilState_TRANSITION_SUPPORT_UNSTABLE(readOnlySelectorSel); // $ExpectError +useRecoilState_TRANSITION_SUPPORT_UNSTABLE({}); // $ExpectError + useRecoilCallback(({ snapshot, set, reset, refresh, gotoSnapshot, transact_UNSTABLE }) => async () => { snapshot; // $ExpectType Snapshot snapshot.getID(); // $ExpectType SnapshotID