Search code examples
javascriptreactjsreact-hooksmobxmobx-react

Is it possible to use React Hooks outside of functional component, or i have to use mobx or redux?


I am new to React, and when I was reading about the docs, I found there were two ways to implement React components, functional-based and class-based. I know before React 16.8 it's not possible to manage state in functional components, but after that there is React Hooks.

The problem is, there seems to be one restriction for React Hooks, they can only be used inside functional components. Take a server-client as an example, which needs to change an isAuthenticated state while 401 received.

//client.js
import { useUserDispatch, signOut } from "auth";

export function request(url, args) {
  var dispatch = useUserDispatch();
  return fetch(url, args).then(response => {
  if (response.status === 401) {
    logout(dispatch);
  }
}
);

//auth.js
import React from "react";

var UserStateContext = React.createContext();
var UserDispatchContext = React.createContext();

function userReducer(state, action) {
  ...
}

function UserProvider({ children }) {
  var [state, dispatch] = React.useReducer(userReducer, {
    isAuthenticated: false,
  });

  return (
    <UserStateContext.Provider value={state}>
      <UserDispatchContext.Provider value={dispatch}>
        {children}
      </UserDispatchContext.Provider>
    </UserStateContext.Provider>
  );
}

function useUserState() {
  return React.useContext(UserStateContext);
}

function useUserDispatch() {
  return React.useContext(UserDispatchContext);
}

function signOut(dispatch) {
  dispatch({});
}

export { UserProvider, useUserState, useUserDispatch, loginUser, signOut };

The client code above will produce error "Hooks can only be called inside of the body of a function component". So maybe I have to move line var dispatch = useUserDispatch() upward to the component where request is called, and pass dispatch as props to request. I feel this is not right, no only request is forced to care about some meaningless(to it) dispatch, but also this dispatch will spread everywhere a component needs to request.

For class-based components, this.state doesn't solve this problem either, but at least I can use mobx.

So are there some other ideal ways to solve this problem?


Solution

  • Yes, you have to use Redux or MobX to solve this problem. You have to maintain isAuthenticated state in the global state of Redux or MobX. Then make an action that could be named like, toggleAuthState and pass is to the child component and toggle the state from there.

    Also you can use functional components for this case. Class based components is not mandatory to use MobX or Redux. If you maintain a HOC as a Container then you can pass the actions and states to the child.

    I am showing an example of using a container as a HOC:

    // Container
    import React from "react"
    import * as actions from "../actions" 
    import ChildComponent from "../components/ChildComponent"
    import { connect } from "react-redux"
    import { bindActionCreators } from "redux"
    
    const Container = props => <ChildComponent { ...props } />
    
    const mapStateToProps = state => ({ ...state })
    
    const mapDispatchToProps = dispatch => bindActionCreators(actions, dispatch)
    
    export default connect(mapStateToProps, mapDispatchToProps)(Container)
    

    Then in ChildComponent you can use your states and dispatch actions whenever you need.