Wrapping React.useState with TypeScript

January 19th, 2021 — 6 min read

by Daiga Ellaby
by Daiga Ellaby
No translations available.Add translation

I made a useDarkMode hook that looks like this:

type DarkModeState = 'dark' | 'light'
type SetDarkModeState = React.Dispatch<React.SetStateAction<DarkModeState>>

function useDarkMode() {
  const preferDarkQuery = '(prefers-color-scheme: dark)'
  const [mode, setMode] = React.useState<DarkModeState>(() => {
    const lsVal = window.localStorage.getItem('colorMode')
    if (lsVal) {
      return lsVal === 'dark' ? 'dark' : 'light'
    } else {
      return window.matchMedia(preferDarkQuery).matches ? 'dark' : 'light'
    }
  })

  React.useEffect(() => {
    const mediaQuery = window.matchMedia(preferDarkQuery)
    const handleChange = () => {
      setMode(mediaQuery.matches ? 'dark' : 'light')
    }
    mediaQuery.addEventListener('change', handleChange)
    return () => mediaQuery.removeEventListener('change', handleChange)
  }, [])

  React.useEffect(() => {
    window.localStorage.setItem('colorMode', mode)
  }, [mode])

  // we're doing it this way instead of as an effect so we only
  // set the localStorage value if they explicitly change the default
  return [mode, setMode] as const
}

Then it is used like this:

function App() {
  const [mode, setMode] = useDarkMode()
  return (
    <>
      {/* ... */}
      <Home mode={mode} setMode={setMode} />
      {/* ... */}
      <Page mode={mode} setMode={setMode} />
      {/* ... */}
    </>
  )
}

function Home({
  mode,
  setMode,
}: {
  mode: DarkModeState
  setMode: SetDarkModeState
}) {
  return (
    <>
      {/* ... */}
      <Navigation mode={mode} setMode={setMode} />
      {/* ... */}
    </>
  )
}

function Page({
  mode,
  setMode,
}: {
  mode: DarkModeState
  setMode: SetDarkModeState
}) {
  return (
    <>
      {/* ... */}
      <Navigation mode={mode} setMode={setMode} />
      {/* ... */}
    </>
  )
}

function Navigation({
  mode,
  setMode,
}: {
  mode: DarkModeState
  setMode: SetDarkModeState
}) {
  return (
    <>
      {/* ... */}
      <button onClick={() => setMode(mode === 'light' ? 'dark' : 'light')}>
        {mode === 'light' ? <RiMoonClearLine /> : <RiSunLine />}
      </button>
      {/* ... */}
    </>
  )
}

This works great, and powers the "dark mode" support for all the Epic React workshop apps (for example React Fundamentals).

A closer look

I want to call out a few things about the hook itself that made things work well from a TypeScript perspective. First, let's clear out all the extra stuff and just look at the important bits. We'll even clear out the TypeScript and add it iteratively:

function useDarkMode() {
  const [mode, setMode] = React.useState(() => {
    // ...
    return 'light'
  })

  // ...

  return [mode, setMode]
}

function App() {
  const [mode, setMode] = useDarkMode()
  return (
    <button onClick={() => setMode(mode === 'light' ? 'dark' : 'light')}>
      Toggle from {mode}
    </button>
  )
}

From the get-go, we've got an error when calling setMode:

This expression is not callable.
  Not all constituents of type 'string | React.Dispatch<SetStateAction<string>>' are callable.
    Type 'string' has no call signatures.(2349)

You can read each addition of indentation as "because", so let's read that again:

This expression is not callable. Because not all constituents of type 'string | React.Dispatch<SetStateAction<string>>' are callable. Because type 'string' has no call signatures.(2349)

The "expression" it's referring to is the call to setMode, so it's saying that setMode isn't callable because it can be either React.Dispatch<SetStateAction<string>> (which is a callable function) or string (which is not callable).

For us reading the code we know that setMode is a callable function, so the question is: why is the setMode type both a function and a string?

Let me rewrite something and we'll see if the reason jumps out at you:

const array = useDarkMode()
const mode = array[0]
const setMode = array[1]

The array in this case has the following type:

Array<string | React.Dispatch<React.SetStateAction<string>>>

So the array that's being returned from useDarkMode is an Array with elements that are either a string or a React.Dispatch type. As far as TypeScript is concerned, it has no idea that the first element of the array is the string and the second element is the function. All it knows for sure is that the array has elements of those two types. So when we pull any values out of this array, those values have to be one of the two types.

But React's useState hook manages to ensure when we extract values out of it. Let's take a quick look at their type definition for useState:

function useState<S>(
  initialState: S | (() => S),
): [S, Dispatch<SetStateAction<S>>]

Ah, so they have a return type that is an array with explicit types. So rather than an array of elements that can be one of two types, it's explicitly an array with two elements where the first is the type of state and the second is a Dispatch SetStateAction for that type of state.

So we need to tell TypeScript that we intend to ensure our array values don't ever change. There are a few ways to do this, we could set the return type for our function:

function useDarkMode(): [string, React.Dispatch<React.SetStateAction<string>>] {
  // ...
  return [mode, setMode]
}

Or we could make a specific type for a variable:

function useDarkMode() {
  // ...
  const returnValue: [string, React.Dispatch<React.SetStateAction<string>>] = [
    mode,
    setMode,
  ]
  return returnValue
}

Or, even better, TypeScript has this capability built-in. Because TypeScript already knows the types in our array, so we can just tell TypeScript: "the type for this value is constant" so we can cast our value as a const:

function useDarkMode() {
  // ...
  return [mode, setMode] as const
}

And that makes everything happy without having to spend a ton of time typing out our types 😉

And we can take it a step further because with our Dark Mode functionality, the string can be either dark or light so we can do better than TypeScript's inference and pass the possible values explicitly:

function useDarkMode() {
  const [mode, setMode] = React.useState<'dark' | 'light'>(() => {
    // ...
    return 'light'
  })

  // ...

  return [mode, setMode] as const
}

This will help us when we call setMode to ensure we not only call it with a string, but the right type of string. I also created type aliases for this and the dispatch function to make the prop types easier as I pass these values around my app.

Hope that was interesting and helpful to you! Enjoy 🎉

Epic React

Get Really Good at React

Illustration of a Rocket
Kent C. Dodds
Written by Kent C. Dodds

Kent C. Dodds is a JavaScript software engineer and teacher. Kent's taught hundreds of thousands of people how to make the world a better place with quality software development tools and practices. He lives with his wife and four kids in Utah.

Learn more about Kent

If you found this article helpful.

You will love these ones as well.