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.