DEV Community

Cover image for React Query and optimistic updates
Chris Bongers
Chris Bongers

Posted on • Originally published at daily-dev-tips.com

React Query and optimistic updates

In the previous article, we looked at React Query Mutations, which are great for updating the data once we receive a mutation callback.

However, how great would it be if we could do an optimistic update to make our application even faster?

Let's see what that even means?

We'll have the original list of Pokemon we saw yesterday, and once we decide to add a new Pokemon to this list, we fire an API request.

At the same time, we ask React Query to add this Pokemon already and not care if the mutation was correct or not.

The only thing we would care about is if it failed for some reason. In that case, we should revert to its previous state.

React Query optimistic updates

Alright let's start with the mutation we had in the previous article:

const {mutate: addNewPokemon} = useMutation(
  (newPokemon) => {
    // return axios.post('API', newPokemon);
    return {name: newPokemon};
  },
  {
    onSuccess: async (newPokemon) => {
      queryClient.setQueryData('pokemon', (old) => [...old, newPokemon]);
    },
  }
);
Enter fullscreen mode Exit fullscreen mode

Instead of this onSuccess callback, we can leverage the onMutate option.

This option gets fired right away and doesn't care about the state of the actual mutation.

onMutate: async (newPokemon) => {
    await queryClient.cancelQueries('pokemon');
    const previousPokemon = queryClient.getQueryData('pokemon');
    queryClient.setQueryData('pokemon', [
      ...previousPokemon,
      { name: newPokemon },
    ]);
    return { previousPokemon, newPokemon };
},
Enter fullscreen mode Exit fullscreen mode

Let's see what's going on here.
We first cancel the existing query so React Query won't start updating it in between us trying to set it manually.

Then we get the current data object for this query.
And manipulate it, as we did before.

Then we return the previous data. This return context can be accessed in the onError function.

Speaking off the error function, this function gets triggered if the mutation fails.
It will get the context from the onMutate return object.

What we want to do is reset the previous state.

onError: (err, newPokemon, context) => {
    queryClient.setQueryData('pokemon', context.previousPokemon);
},
Enter fullscreen mode Exit fullscreen mode

Let's complete the function by introducing a failing request.
What should happen when we run this function:

  • mutation gets triggers
  • onMutate temporary adds the new Pokemon to the list
  • mutation returns a failed request
  • onError gets called, and we reset the state
  • everything is back to before
const {mutate: addNewPokemon} = useMutation(
  async (newPokemon) => {
    const request = await fetch('https://pokeapi.co/api/v2/pokemon', {
      method: 'POST',
      data: {pokemon: newPokemon},
    });
    const {results} = await request.json();
    return results;
  },
  {
    onMutate: async (newPokemon) => {
      await queryClient.cancelQueries('pokemon');
      const previousPokemon = queryClient.getQueryData('pokemon');
      queryClient.setQueryData('pokemon', [...previousPokemon, {name: newPokemon}]);
      return {previousPokemon, newPokemon};
    },
    onError: (err, newPokemon, context) => {
      queryClient.setQueryData('pokemon', context.previousPokemon);
    },
  }
);
Enter fullscreen mode Exit fullscreen mode

I've also created this Code Sandbox environment so you can try it out directly.

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

Latest comments (0)