Why not use create-react-app?

If you’re looking to get a quick, modern version of a React website up, you’re often going to be hit with the advice to use create-react-app. I don’t disagree that it’s easy – especially if you know the site is going to be thrown away, or is extremely temporary. I would highly recommend it for those situations.

However, the luster of create-react-app starts to fade when you start to build real sites with it. At some point you’re likely to want to npm run eject to remove the black-box that is your build process and get at the underlying webpack config for your own custom needs. While working with Webpack isn’t impossible, I find it often gives you a new problem – configuring your build tool, not to mention maintaining your configuration file with latest webpack syntax changes, updating plugins to match, etc. In short though, it’s a piece of the toolchain I don’t think you should have to think about. Because you do, it simply adds more complexity to your project and often distracts from doing more app-specific tasks with your time. I’d venture to say that most of the time, people do not need webpack’s lower level approach to build tool configuration. They simply want to give some entry points to a compiler and have it do all the expected compilations from typescript (or modern javascript) to a more compatible, minified form. They want code-splitting, cache-busting resource names, and minified code/images. There are a few other things, but generally speaking this is what most people want. If a build tool does that out of the box, you shouldn’t have to think about it.

So, what’s NOT to like about webpack?

  • configuration beyond the basics is complex
  • requires maintenance of build config file
  • poor error messages when debugging failures (imho)
  • slow build times due to transpilation using babel, which is written in Javascript

What’s to like about webpack?

  • ubiquitous; you do have resources to help with your problems
  • better than what came before it?

An alternative build tool

There are a number of native build tools that have recently been developed that are a lot easier and more practical for projects. Most of these native transpilers are written in either Go or Rust, but both languages offer a significant performance boost over the javascript-based Babel transpiler (used by Webpack).

A lot of performant build frameworks will use one of the three dominant native transpilers out there, which are Typescript’s tsc, esbuild or swc. Of these, esbuild and swc seem to be geared towards a broader compilation of resources rather than strictly typescript. Going through the various frameworks out there and what underlying build tool they use is beyond this article. I’ve found that Parcel, which originally used esbuild and has more recently transitioned to using swc, is one of the cleanest, most developer-friendly build tools currently available.

So, without further discussion, let’s jump into a minimal web site that uses Parcel.

Initial Repo Definition

We’re shooting for a simple website that uses typescript, Parcel (obviously), React, and TailwindCSS. The use of TailwindCSS is a personal choice. Others may prefer Styled Components, or more traditional sass or less. To demonstrate the

styling approach of TailwindCSS, we will be styling the same example page that create-react-app uses.

You can find all of the code discussed here on the corresponding Github repo: Minimal Web Repo.

Create the basic repo and add placeholder files

If you’re familiar with the terminal, open your favorite and let’s get started. Here are the commands to get the npm repo setup. We’ll create a src directory and touch a few files which is just a quick way to make empty files that we will be filling in later.

$ mkdir example
$ cd example
$ npm init -y
$ mkdir src
$ touch src/index.html src/index.tsx src/global.css \
    ./types.d.ts ./tsconfig \
    ./tailwind.config.js

Now we’ll install the basic dependencies for the the main frameworks we discussed.

$ npm install tailwindcss react react-dom
$ npm install --dev postcss @types/react @types/react-dom @types/tailwindcss parcel @parcel/transformer-svg-react @parcel/transformer-typescript-tsc

At this point we need to edit the package.json file to remove the main: parameter and add in two new scripts, build and start. Build will be for generating production assets and Start will be for hot-reloading development work.

Your final package.json, after these edits should look like this. Package versions will likely be different, but should be okay. Don’t forget to remove main entry in your package.json!

package.json

{
    "name": "example",
    "version": "1.0.0",
    "description": "",
    "scripts": {
        "test": "echo \"Error: no test specified\" && exit 1",
        "build": "npx parcel build src/index.html",
        "start": "npx parcel src/index.html"
    },
    "keywords": [],
    "author": "",
    "license": "ISC",
    "dependencies": {
        "@parcel/transformer-svg-react": "^2.5.0",
        "@parcel/transformer-typescript-tsc": "^2.5.0",
        "@types/react": "^18.0.8",
        "@types/react-dom": "^18.0.0",
        "@types/tailwindcss": "^3.0.10",
        "parcel": "^2.5.0",
        "postcss": "^8.4.12",
        "react": "^18.1.0",
        "react-dom": "^18.1.0",
        "tailwindcss": "^3.0.24"
    }
}

Adding Typescript Support

With that out of the way, let’s configure the repo for typescript. We’ve already installed the packages for this support, but we need to add a small configuration file to specify what files should be considered for typescript and when they are, what compiler options we prefer. We do this by creating a tsconfig.json file in the project root.

tsconfig.json

{
    "include": ["src/*.tsx?", "types.d.ts"],
    "exclude": ["node_modules", ".parcel-cache", ".vscode", "dist", "build"],
    "compilerOptions": {
        "experimentalDecorators": true,
        "esModuleInterop": true,
        "jsxImportSource": "react"
    }
}

Most of the entries in the tsconfig.json file should be rather obvious, but the reference to types.d.ts needs a bit of explanation. Any .d.ts files are Typescript definition files that can be named/organized with various arbitrary conventions, but all essentially define types or interfaces for your code. In these files, you’ll find type definitions for complex objects, or method signatures, or anything else that helps specify typing information for Typescript as it pertains to your source code.

Now, in this case, we are actually not adding a type for something explicitly with our code, but rather patching a common type of import used with React – SVGs. We can import SVG files with an import statement, but that type is not provided by default with React, or any other libraries so we manually define it within our types.d.ts file and include that within our tsconfig.json configuration for inclusion. Now, typescript is able to reconcile the svg import appropriately.

types.d.ts

declare module '*.svg' {
    const content: any;
    export default content;
}

Adding Tailwind CSS Support

Just as a reminder – Tailwind CSS is just one of many approaches to managing stylesheet styling within your app. Within the React eco system there are a number of approaches to this, but Tailwind has been very intuitive to me as I’ve used it. I’m also fond of Styled Components too, but feel like Tailwind gets closer to helping me distill the CSS down to the relevant parts and if I want to abstract these styles to a string value or a shared component, I can still do that with Tailwind. I’m also not forced to use React to use Tailwind and that part appeals to me a lot. There are times when you would like to have access to the same fundamental styling but it might be in a context where there is no need for anything more than a simple html page. Tailwind can support that all the way up to a full blown React app.

First we’ll need to define our main configuration file for Tailwind.

tailwind.config.js

module.exports = {
    content: ['src/**/*.{html,ts,tsx,js,jsx}'],
    theme: {
        extend: {
            animation: {
                'spin-slow': 'spin 20s linear infinite',
            },
            keyframes: {},
        },
    },
    plugins: [],
};

There are a few things to point out in this configuration. First, we’ve specified the files that will be aware of Tailwind CSS styles. That part should be fairly self explanatory. The second, is the inclusion of a theme animation. This section here is optional, but for the sake of copying the spinning React logo found in the create-react-app demo, we’ve included a spin animation definition here, such that we can style our svg with animate-spin-slow later, and it will apply the typical CSS definition for a spin animation with rate of 20s. We could have just defined this style directly in the React component using CSS and bypassing use of Tailwind altogether but this is a good example of how you can define your own custom Tailwind styles. We’ll see more of this later when we build the component.

In addition to the main config file, we also need to define a PostCSS config file to alert it to the fact that Tailwind styles are part of this ecosystem. Just fill it in and move on; not much more to say about it.

.postcssrc

{
    "plugins": {
        "tailwindcss": {}
    }
}

Additionally, within our ./src directory, we’ll define a global.css file that will import Tailwind definitions for us when we create our React components. We’ll fill it in and refer to it later when we define our React component.

src/global.css

@tailwind base;
@tailwind components;
@tailwind utilities;

Adding our Application Files

Here’s where we get to the fun part; adding our application code. Again, we are simply trying to recreate the initial demo page from create-react-app. For that, we need a simple component, with some centered text and logo. There is a bit of CSS animation used to spin a SVG logo as well.

Here are the four files we’ll define within our ./src directory:

  • favicon.ico – site icon; refer to Github repo for contents or create-react-app
  • index.html – our main html page used to load our app
  • index.tsx – the main app code; we use .tsx to specify it’s a Typescript file containing JSX.
  • logo.svg – a standard SVG file for the React logo; this was taken directly from create-react-app

For brevity, we are going to skip the two images within the ./src directory and assume you can add those yourself or copy them from either our Github repo or the original create-react-app repo if you prefer.

First up is our src/index.html file. This will be a standard HTML5 page with the local file imports for our App code and styles. Parcel will parse this and replace the various file references with cache-busting references to the files it will generate. Of of those files we are referring to is the src/global.css file we created earlier for supporting Tailwind.

NOTE: Be careful that your script tag for the index.tsx is of type module, otherwise Parcel will not consider it for inclusion as an input src file.

You will also see that we begin to use a little Tailwind styling on the body tag with our class definitions for m-0 p-0 which set both margin and padding to zero. We’ll use more styling within our React component, but you’ll notice that it’s the same style directives in use within this generic html file as it will be within our React code.

src/index.html

<!DOCTYPE html>
<html lang="en">
    <head>
        <meta charset="utf-8" />
        <title>Example Site</title>
        <link rel="shortcut icon" href="favicon.ico" />
        <meta
            name="description"
            content="Example of a web site using typescript, react, parcel and tailwindcss"
        />
        <link rel="stylesheet" href="./global.css" />
    </head>
    <body className="m-0 p-0">
        <div id="app"></div>
        <script type="module" src="./index.tsx"></script>
    </body>
</html>

And now we can define our React component to display the spinning logo with centered text.

src/index.tsx

import React from 'react';
import Logo from './logo.svg';

import { createRoot } from 'react-dom/client';

// This is non-standard, but given this App is our root, it's convenient.
const container = document.getElementById('app');
const root = createRoot(container);
root.render(<App />);

// Begin our App-specific code. I've copied some of the layout of the
// create-react-app but using Tailwind CSS as an example of its use.
type Props = {};

function App({}: Props) {
    return (
        <main className="grid place-items-center h-screen bg-[#282c34] m-0 p-0">
            <div>
                <img
                    src={Logo}
                    className="m-0 p-0 animate-spin-slow w-full"
                    alt="logo"
                />
                <div className="m-0 p-0 text-center text-white text-2xl">
                    <p>
                        Edit <code>src/index.tsx</code> and save to reload.
                    </p>
                    <a
                        className="text-blue-700 underline text-2xl text-center"
                        href="https://reactjs.org"
                        target="_blank"
                        rel="noopener noreferrer"
                    >
                        Learn React
                    </a>
                </div>
            </div>
        </main>
    );
}

export default App;

If you’re familiar with typical React app conventions, you’ll notice we’ve done a slightly non standard thing by including the code to inject our component into the DOM as part of this component file. We can get away with that here because we only ever import this component once, within our HTML file, but this would need to be split out into a separate loader file if we intended to import this component within another context, like testing, for example.

You’ll also notice that we refer to our custom animate-spin-slow Tailwind style here that we previously defined in our tailwind.config.js file. Every other Tailwind style in use here is standard except this one.

Conclusion

That’s it! If all went well, you should be able to run the following, open the reported url in your browser and see my clone of the create-react-app entry page.

$ yarn start

If you remember, the yarn start create a hot-reloading Parcel server for us to dev with, but when it comes time for a production build, we can issue a yarn build and take the contents of the dist folder to deploy however we choose.

Further considerations

This is a minimal app setup. If this were to be used for a real site, a server would have to be added to serve up the initial HTML file. That could be something a simple file-serving http server, or something a bit more complex like Express.js or any other Javascript (or non-Javascript) web server software.

Additionally, if this were being used for a real site, you’d likely want to include a testing framework for your frontend code. I would highly recomment react-testing-library or jest for testing the React components.

References