When you’ve decided to use React, take a moment to consider the site framework (for building, routing, and other dev APIs) that best suits your project.
When React decided to sunset Create React App, many folks remembered it fondly. In a brief rant, I noted:
I used it for so many demos and examples over the years. But time changes things. The React team's focus has shifted, and we now have other tooling that makes working with React easy, depending on the scenario.
Times change. Tooling comes. And tooling goes. And it goes for a variety of reasons.
In this case, there were a great many things at play, including the evolving and expanding rendering needs of JS devs, the React team's priorities, and the rapidly advancing tooling built up around React.
What I found weird and misleading (and what sparked the rant) is how the React docs recommend starting new React projects.
Rather than focus on helping developers pick what's best for their project, the docs present three "full-stack" frameworks, downplay others, leave out a few popular choices, make easy getting started paths sound complex, and add distracting notes about rendering patterns.
Let's reframe how to think about getting started with your next React project by centering you and what is best for your project.
I'm assuming you're reading this after deciding on React and your UI framework. If that's not the case, or if you haven't considered why React is (or isn't) the right choice for your project, start there. And if you've determined that React is right for your project, come back!
React was my go-to UI framework for many years — I used it everywhere I could. Today, I think for a moment before starting a project. I find myself landing on other solutions (or a combination of solutions) about as often as I pick React.
Tools get marketed to developers, which causes developers to lead architectural decisions with tools that have been marketed to them. We need to push against this. Leading with tooling is leading with a solution without consideration for the problem, getting in the way of making the best architectural decisions for a project.
I usually think about a new project's architecture step-by-step, like this:
#2 and #3 often require extensive research when presented with a new problem. Otherwise, you should be able to work through these quickly. The point is to take a moment to consider and choose solutions based on the problems they are addressing, rather than jumping right to the solution.
I categorize most new projects I've spun up in the last decade into one of three groups:
We'll consider these our "problem areas" and explore a solution for each of these categories.
I've intentionally left out native (and hybrid) applications. I haven't built for these cases in several years and don't have an opinion on the best choice (although it seems like Expo is a popular solution these days).
While you can go much deeper based on your scenario (and I encourage you to), I've aimed to present a single recommendation for the areas listed above.
To summarize these recommendations into a single sentence, I'll share a bit of a mantra that has developed around the virtual halls of Netlify: "Astro for sites, TanStack for apps, Vite under the hood."
These three projects lead with what's best for the developer and the developer community. Together, they cover the majority of use cases for any new project, even outside the context of React.
Let's explore how to apply that mantra to the use case categories mentioned above.
When you need to spin up a simple prototype or an app with limited complexity, where you're working with established or external APIs, Vite is a great solution.
It's super simple to get started. With one command, you're up and running with a new application.
npm create vite@latest my-react-app -- --template react-ts
The React docs consider this "building a React app from scratch" because it doesn't support all modern React features out of the box. In my experience, most React apps (especially simple examples and prototypes) don't need most of React's features. Vite is a great place to start, not a scary custom option.
To add client-side routing and create a single-page application (SPA) to a Vite + React app, install React Router.
npm install react-router
Then, add your routes to the root of your app.
import React from "react";
import ReactDOM from "react-dom/client";
import { BrowserRouter, Routes, Route } from "react-router";
import App from "./app";
const root = document.getElementById("root");
ReactDOM.createRoot(root).render(
<BrowserRouter>
<Routes>
<Route path="/" element={<App />} />
</Routes>
</BrowserRouter>
);
Voila! SPA ready to go. (There's a lot more you can do with the router.)
(At the time of writing) When you prompt AI code-generating agent tools like Bolt or Lovable and aren't explicit about the stack you want to work with, they choose Vite + React every time. They are programmed that way because it's a good architectural decision that leads with simplicity and gets you far.
If the agent realizes that you need multiple routes, it will likely install react-router
.
Astro is an ideal choice for content sites, where most of the content doesn't change rapidly. The creators of the content typically don't consume content on the site. Think of blogs or marketing sites, not dynamic applications where users change content.
I typically classify these projects broadly as "websites" (vs "web applications," which we'll cover below).
Astro is optimized for content sites, and it works incredibly well for the typical use case but is also jam-packed with features for flexibly covering a variety of needs.
For example, if your site can be mostly pre-rendered but your home page is changing constantly, Astro has you covered — you can render your home page from the server while pre-rendering everything else.
The same goes for client-side code. When you have highly interactive areas of your site, you can implement what Astro calls a client island. Let's take a quick look.
Use the delightful CLI wizard to create a new Astro project. Choose your preferred settings. (For demo purposes, choose the basic, minimal starter.)
npm create astro@latest
Once the site is set up, you can add React in one command.
npx astro add react
The correct packages will be installed, and configuration will be adjusted to use React.
To see React in action, create a simple counter component.
src/components/Counter.jsx
import { useState } from "react";
export default function Counter() {
const [count, setCount] = useState(0);
return <button onClick={() => setCount(count + 1)}>Count: {count}</button>;
}
Replace the welcome component on the home page with the counter component.
src/pages/index.astro
---
import Layout from "../layouts/Layout.astro";
import Counter from "../components/Counter.jsx";
---
<Layout>
<Counter />
</Layout>
Start up the dev server and visit localhost:4321.
npm run dev
You'll see that the component renders, but the counter doesn't count. This is because Astro renders only HTML and CSS by default. This keeps your site's build output clean, simple, and super-duper fast, injecting client-side JavaScript code only where you've specifically asked for it.
To use a client island, add the client:load
client directive to the page where you've used the counter component.
---
import Layout from "../layouts/Layout.astro";
import Counter from "../components/Counter.jsx";
---
<Layout>
<Counter client:load />
</Layout>
Reload the page, and your counter will count!
This is incredibly useful because you can spin up little React components wherever you need to without requiring your entire site to rely on client-side JavaScript code.
React apps are a collection of components with routing. With Astro, the routing is handled by pages, either pre-rendered or rendered at runtime from a server. Then, your components are hydrated only as needed and from the point at which you specify.
Here's a demo of the minimum amount of work to get React to run on the client for an Astro site.
The last category is dynamic applications, where users drive constant change to the data, consume that same data on the application, and want to see updates instantly.
As I write this, we're in an era of rapid change. Without going into too much detail on all the JS drama, I'll leave you with five short anecdotes to help you choose what's best for you.
First, the React docs position these frameworks as "full-stack," which is misleading, as it suggests those not in the category can't cover traditional full-stack behavior, which is wrong. (I'd argue you could consider 11ty a full-stack framework if you use it that way, but I digress.)
This category of frameworks leads with JavaScript. They differ from one another in the APIs they introduce to aid the development experience, along with the patterns they provide for distinguishing between which JS code is server-rendered and which is client-rendered (or both).
I continue to use Astro whenever possible. I'm actively building out a series of business applications with Astro. While it presents challenges (as any framework does at some point), I prefer the more apparent distinction between server-rendered HTML and client-rendered JavaScript.
If that appeals to you, try it. For many folks in the space, Astro isn't the right pattern for highly dynamic applications, and that's totally fine. It's not optimized for that. It doesn't work for everyone.
Prior to Astro, I used Next.js for many years, but it has some significant problems today, and I don't suspect I'll ever use it again. Part of this has to do with governance and vendor lock-in. But from a tech side, once the app router was introduced, we started to see more APIs come on the scene that have been unstable and/or unnecessary. I don't like working with it.
But it is still, hands down, the go-to React meta framework. You may find yourself leaning toward it, and that's fine. Just make sure you know what you're getting into.
(Disclaimer: I work at Netlify. Next.js is owned and maintained by Vercel, a direct competitor of Netlify. I loved Next.js long before I worked at Netlify, but I had already started to move away from it before I started at Netlify in 2023. However, only recently have better options been appearing on the scene. If I'd written this last year, I would have probably said, "Next.js is going to cause you problems, but it's probably your best choice.")
Fortunately, we're evolving!
Over the last year, I have seen teams struggling with Next.js begin to move to Remix. While it seems that those who have moved to Remix seemed happy with the experience, I can't confidently comment much more than that, as I have minimal experience with Remix.
And now there's a new player in town! TanStack is a suite of developer tools that can be stitched together to build dynamic and complex React applications. It is currently in beta, but it's getting a lot of love from developers.
Even over the last few weeks, I've seen notes of folks considering the Next-to-TanStack move over the Next-to-Remix move. This could be the corner of the community I'm viewing. But it could also be because TanStack has taken a framework-agnostic approach where it can. Or that TanStack is more developer-community focused and not owned by a corporation.
I'll leave you by repeating the mantra to summarize these recommendations: Astro for sites, TanStack for apps, Vite under the hood.
These three projects lead with what's best for the developer and the developer community. Together, they cover the majority of use cases for any new project, even outside the context of React.
You'll spend a lot of time with your frameworks of choice. So don't just take my word about them. Learn more about them. Test them out. See what feels right. Then, get back to building!