Dependent queries in React Query

โœ๏ธ

Lets see how we can make dependent queries with React Query

12 Feb, 2022 ยท 5 min read

In some cases, you might need only to fire a query once a specific condition is met.

Some examples of this:

  • Wait for user input
  • Wait for the main query to return the user ID
  • We are still retrieving an ID from the storage
  • Wait for a search query

And so on.

Using user input in this example, I'll show you how to make this happen in React Query.

You can see the result in the video below. We only start using the query once we have valid input. We put the query into idle mode as long as it doesn't exist.

React Query dependent queries

We can leverage the enabled property to make queries dependent on a variable. This will tell React Query if this query should be enabled or not, and it can accept anything that calculates to a boolean.

You can use it like this:

const { isIdle, data } = useQuery('your-key', yourQueryFn, {
  enabled: conditionIsTrue,
});

Let's try it out and create a user input. Only once this user input is more than zero should it be querying anything.

This code will be based on the React Query Pokemon app we built before.

The input will be placed in the App component.

function App() {
  const [number, setNumber] = useState(0);

  return (
    <QueryClientProvider client={queryClient}>
      <input
        type='number'
        value={number}
        max='10220'
        onChange={(e) => setNumber(e.target.value)}
      />
      <button onClick={() => setNumber(0)}>reset</button>
      <PokemonList number={number} />
    </QueryClientProvider>
  );
}

We keep track of a state number and update the state on change. This state gets passed to our PokemonList component.

Let's look at how the PokemonList component can receive this number and how we can make our query dependent on it.

function PokemonList({ number }) {
  const { isIdle, data, isLoading } = useQuery(
    ['pokemon', number],
    () => fetchPokemon({ number }),
    {
      enabled: number > 0,
    }
  );

  return <></>;
}

We receive the number and assign it to a specific unique key, as you can see above. Then we invoke the fetchPokemon function and pass along the number to this function. The dependency comes in at the enabled property where we tell react query only to enable this once it's bigger than zero.

Let's take a look at what our fetchPokemon function looks like now:

const fetchPokemon = async ({ number }) => {
  const request = await fetch(`https://pokeapi.co/api/v2/pokemon/${number}`);
  return await request.json();
};

And then all that's left is to fix the actual return in the component. We'll add some data for this Pokemon and keep track of our isIdle and isLoading states so the user knows what's going on.

function PokemonList({ number }) {
  const { isIdle, data, isLoading } = useQuery(
    ['pokemon', number],
    () => fetchPokemon({ number }),
    {
      enabled: number > 0,
    }
  );

  return (
    <div className='App'>
      {isIdle ? (
        <p>Is idle...</p>
      ) : isLoading ? (
        <p>Loading...</p>
      ) : (
        <ul>
          <li>
            <strong>Name</strong>: {data.name}
          </li>
          <li>
            <strong>Weight</strong>: {data.weight}
          </li>
          <li>
            <strong>Image</strong>:
          </li>
          <li>
            <img
              src={data.sprites?.front_shiny ?? data.sprites?.front_default}
              alt={data.name}
            />
          </li>
        </ul>
      )}
    </div>
  );
}

And that's it! We made our first dependent react query function.

You can try it out in this Code Sandbox:

Thank you for reading, and let's connect!

Thank you for reading my blog. Feel free to subscribe to my email newsletter and connect on Facebook or Twitter

Spread the knowledge with fellow developers on Twitter
Tweet this tip
Powered by Webmentions - Learn more

Read next ๐Ÿ“–

React cleaner use of setTimeout

15 Jul, 2022 ยท 4 min read

React cleaner use of setTimeout

Upgrading to React 18

22 Apr, 2022 ยท 3 min read

Upgrading to React 18