Lazy Loading Routes with Vite and React Router V7

Recently, we switched the Street Art Cities dashboard (where users upload artworks, create routes, view insights, etc.) from a massive monolithic Next.js app – that also contained the many pages of our website! – to a standalone React app using React Router and Vite.

We have dashboard for various different roles - hunter, artist, country manager, admin, etc. To make sure not everyone needs to load all the Javascript for each dashboard, even if they don't have access to it, I set up chunking and lazy loading for various sub-routes of the dashboard.

Updating our router

Previously, our main router looked like this:

import HunterRoutes from './routes/hunter';
import AdminRoutes from './routes/admin';

const App = () => {
  return (
    <Routes>
      <Route path="hunter/*" element={<HunterRoutes />} />
      <Route path="admin/*" element={<AdminRoutes />} />
      // ...
    </Routes>
  );
}

This is a perfect setup for lazy loading, since the various dashboards are already clumped together using sub-routes.

We can combine Vite's dynamic import() function and React's lazy() to get this done. I've started by introducing a helper component that combines them:

const Lazy = ({ component }) => {
  const Component = React.lazy(component);

  return (
    <Suspense fallback={<LoadingSpinner />}>
      <Component />
    </Suspense>
  );
};

We can use it like this:

const HunterRoutes = () => import('./routes/hunter');
const AdminRoutes = () => import('./routes/admin');

const App = () => {
  return (
    <Routes>
      <Route path="hunter/*" element={
	    <Lazy component={HunterRoutes} />
	  } />
	  <Route path="admin/*" element={
	    <Lazy component={AdminRoutes} />
	  } />
	  // ...
    </Routes>
  );
}

Important: we need to make sure we define the import() with a pre-set value. Before, I tried to abstract this further by passing path prop into the <Lazy> component, and having the import() in the component, but that means Vite can't resolve the reference during build-time.

That means that even though it works in development mode, in production it will try to load ./routes/hunter, which probably won't exist in your bundled code!

Chunking strategy

By default, Vite (and Rollup, which Vite uses under the hood) will try to intelligently chunk our dependencies. It's not always amazing at it though, so in cases like this, where there is a clear-cut distinction in when users will need certain bits of code.

Rollup allows you to define a manualChunks configuration option where you can pass a function that determines the chunk a certain file of source could should be put into.

Here's how you can use this in the vite.config.js:

export default defineConfig({
  // ...
  build: {
    rollupOptions: {
      output: {
        manualChunks: (id) => {
          const match = /src\/routes\/([\w-]+)\//.exec(id);
          if (match) {
            return match[1];
          }
        },
      },
    },
  },
});

The parameter id here refers to the full path for the input file. In this example, we match files like src/routes/admin/AdminDashboard.jsx and puts them in a chunk based on the first directory within src/routes.

The result is a chunk per dashboard section: Image

This is a massive improvement over the 'before' state: Image

That means that all of our hunters and artists no longer need to load the almost 3 megabytes of JS that is only used in the admin dashboard!