When it comes to status managers, wheels are flying around. In the Class era, Redux and Mobx dominated the entire market, and there were hardly any students who hadn’t used Redux. With the birth of Hooks, a new set of wheels emerged, including unstate-next, Constate and so on. Of course, regardless of the wheel, the problem is the same: ** state sharing across components. ** While solving this core problem, the following features should be satisfied as much as possible:

  • TypeScript support
  • Friendly asynchronous support
  • State interdependence is supported
  • Support for both Class and Hooks components
  • Using a simple

Recoil experience

Let’s take a look at Recoil, facebook’s official status manager solution.

The preparatory work

To use Recoil, we need to wrap a RecoilRoot in the outermost layer of the project, which, like most state managers, uses context to pass data across components.

import React from 'react';
import { RecoilRoot } from 'recoil';

function App() {
  return (
    <RecoilRoot>
        ...
    </RecoilRoot>
  );
}
Copy the code

Sharing across component states

States are easiest to define and use. In Recoil, a state is defined using Atom.

const inputValueState = atom({
  key: "inputValue".default: ""
});
Copy the code

As shown in the code above, we define an inputValue state whose default value is an empty string. Note the key field, which should be globally unique. This key is mainly used for debugging, persisting data (a unique identifier for data recovery), and making it easy to see the global Atoms tree. Consuming state is also simpler, with useRecoilState consuming state.

import React from "react";
import { useRecoilState } from "recoil";
import { inputValue } from ".. /store";

const InputA = (a)= > {
  const [value, setValue] = useRecoilState(inputValueState);

  return <input value={value} onChange={e= > setValue(e.target.value)} />;
};

export default InputA;
Copy the code

Isn’t that easy? This is the basic usage of Recoil. I’ve written a demo here, so you can try it out.


State interdependence

Some state depends on some other state, and then you define that state using selector. For example, we need to define a new state, filterdInputValue, which is the value after filtering the numbers in inputValue.

const filterdInputValue = selector({
  key: "filterdInputValue".get: ({get}) = > {
    // Get can read other states
    const inputValue = get(inputValueState);
    return inputValue.replace(/[0-9]/ig.""); }});Copy the code

Selector is pretty simple, just to implement state dependency. You can try it out in this demo.

Asynchronous support

Good asynchronous request support is essential for the state manager. Recoil provides a useRecoilValueLoadable to handle asynchronous requests. Go straight to the example:

const currentUserNameQuery = selector({
  key: "CurrentUserName".get: async() = > {const response = await queryUserInfo();
    returnresponse.name; }});Copy the code

We need to define the asynchronous state by selector. If get is a Promise, that state is asynchronous, and we need to use useRecoilValueLoadable to consume that state.

const UserName = () => {
  const userNameLoadable = useRecoilValueLoadable(currentUserNameQuery);
  switch (userNameLoadable.state) {
    case "hasValue":
      return <div>{userNameLoadable.contents}</div>;
    case "loading":
      return <div>Loading...</div>;
    case "hasError":
      throw userNameLoadable.contents;
  }
};
Copy the code

As you can see from the above example, the state returned by useRecoilValueLoadable can be read from the state field to read the status of the asynchronous request. I wrote a demo so you can try it out.






useRecoilValueLoadable
React.Suspense
Suspense

const UserName = () => { const userName = useRecoilValue(currentUserNameQuery); return <>{userName}</> } }; function MyApp() { return ( <React.Suspense fallback={<div>Loading... </div>}> <UserName /> </React.Suspense> ); }Copy the code

evaluation

advantages

  • It would be a good thing if state managers were everywhere.
  • React Concurrent mode is well supported.

insufficient

Recoil is still under development, and documentation is not complete. Based on the situation, here are my feelings.

1. Ts is not implemented and currently not supported

I was surprised to find this, but I didn’t find it until I wrote this article. It’s weird. It should be natural for Recoil to support typescript. Maybe @types/ Recoil will be needed later.

2. Currently, no Class component consumption status is supported.

This feature should be a must and should not completely discard the Class component. The next version will definitely support this feature. Low cost of implementation, it would be anti-human not to support it.

3. There are too many apis, which have some starting costs.



atom
selector

4. Consumption is complicated

When we need to consume a state, we need to import two things, which is tedious.

import { useRecoilState } from "recoil";
import { inputValueState } from ".. /store";

/ / usage
useRecoilState(inputValueState);
Copy the code

It should be possible to consume the string key directly, but this is the same as the Redux problem. Ts is not supported.

import { useRecoilState } from "recoil";

useRecoilState('inputValueState');
Copy the code

Either way, importing two things is not a good use.

5. Not enough highlights

I didn’t see anything that caught my eye, I didn’t use impulse. Wait and see

Afterword.

Recoil overall, relatively moderate, need to wait and see the development. Another recommendation is hox, the simplest React state manager I’m currently using. It has only one API, is very intuitive, has no overhead costs, and fully embraces Hooks 😋.

Thank you for reading

Pay attention to the public number “front-end technology expert”, pull you into the communication group, we communicate and progress together.