Blog
Juri Strumpflohner
January 12, 2023

React, Vite and TypeScript: Get started in under 2 minutes

React, Vite and TypeScript: Get started in under 2 minutes

Let’s be honest. Dealing with tooling is not something enjoyable if you have to deliver code. It should just work and not be in the way. So let’s explore how to kickstart your next React project using Vite, in under 2 minutes, without worrying about the setup.

Table of Contents

· How do I create a new project setup?
· Running, building and testing the app
· Building the app
· Testing the app
· Running integration tests with Cypress
· Linting
· Customize Vite and Vitest
· Hidden gem: Caching
· Hidden gem: Easily modularize your app
· Hidden gem: Visualize your architecture
· Hidden gem: Guard your boundaries
· Hidden gem: Just run what changed
· Hidden gem: A dedicated Editor extension
· Hidden gem: Automated Upgrades
· Using CRA? Automatically migrate to Vite + Nx
· Conclusion

Traditionally, you might lean towards Create-React-App (CRA) started to do precisely that. But what if I told you there’s a better alternative, providing

  • not just scaffolding for the initial setup but helping you along the way to generate components, routing, etc
  • automatically sets you up with best practices tools for e2e testing, unit testing, code formatting, and linting
  • has built-in support for Vite and Vitest (alternatively Webpack & Jest)
  • caches your scripts to speed up things
  • helps you modularize your application
  • comes with automated upgrade features to keep your tooling evergreen

I’m talking about Nx. Nx comes with a set of plugins that come with code generation abilities and help abstract some of the lower-level tooling setups. And this can be really interesting for the use case we wanna tackle today.

Reader: “Wait a minute, I heard about Nx. Isn’t that for monorepos?”Me: _“Yeah you’re right. But in 15.3 they introduced something called ‘standalone apps’”
Reader: “Standalone?”
Me: “Yeah, a fancy term for a setting up a single app and allows for some cool modularization. There’s a video introducing that feature here: https://youtu.be/qEaVzh-oBBc"
_Reader: “ha, interesting 🤔”

So let’s go and set up our React + Vite + TypeScript project.

How do I create a new project setup?

To set up a new project, just invoke the following command:

npx create-nx-workspace@latest awesomereactapp --preset=react-standalone

Note awesomereactapp is the name of the app and folder being created, and --preset=react-standalone tells Nx which template to use when scaffolding the initial setup. You can also invoke it like:

npx create-nx-workspace@latest awesomereactapp

And then choose the option you prefer in the terminal prompt:

In the end, what you’ll get is the following structure:

Running, building and testing the app

First off, let’s run our new, shiny application. Just invoke

1npm start 2

And in a matter of milliseconds, your app should be served at http://localhost:4200/.

npm start just invokes the script defined in the package.json:

package.json
1{ 2 "name": "awesomereactapp", 3 ... 4 "scripts": { 5 "start": "nx serve", 6 "build": "nx build", 7 "test": "nx test" 8 } 9 ... 10} 11

Internally this delegates to nx serve, where serve is the Nx target to be invoked. You can find those in the project.json:

project.json
1{ 2 "name": "awesomereactapp", 3 "$schema": "node_modules/nx/schemas/project-schema.json", 4 "sourceRoot": "./src", 5 "projectType": "application", 6 "targets": { 7 "build": {...}, 8 "serve": { 9 "executor": "@nrwl/vite:dev-server", 10 "defaultConfiguration": "development", 11 "options": { 12 "buildTarget": "awesomereactapp:build" 13 }, 14 "configurations": { 15 "development": { 16 "buildTarget": "awesomereactapp:build:development", 17 "hmr": true 18 }, 19 "production": { 20 "buildTarget": "awesomereactapp:build:production", 21 "hmr": false 22 } 23 } 24 }, 25 "test": {...}, 26 "lint": {...} 27 }, 28 "tags": [] 29} 30

This is where you can see all the targets available for a given project, and you can add your own! In a Nutshell, an Nx target contains

  • executor - a function (here dev-server) exposed by the plugin (here @nrwl/vite) to run the task at hand. Think of it as the wrapper of your usual Npm scripts
  • options - this is where you can pass options to the executor
  • configurations - allows you to create different versions of the options . You control which configuration is used by passing it via the --configuration=production flag to your commands. Also, note the defaultConfiguration.

You can go more in-depth if you want on the Nx docs.

Building the app

Just in the same way as serving our web app, we can build it with

npx nx build

This places an output into a dist folder. Now that we've seen the project.json targets, you probably guessed that you could customize that output folder directly in those settings.

Testing the app

This setup also comes with testing baked in using Vitest. And you probably guessed it, you can just run tests as follows:

npx nx test

Running integration tests with Cypress

You might have noticed the e2e folder. That's a fully-functioning setup of Cypress for doing integration-level or even full end-to-end tests.

This is excellent because you don’t have to configure anything at all. No need to

  • Cypress configured to use Vite (instead of Webpack)
  • set up linting for the e2e project (yes writing good quality test code is just as important)
  • spinning up our development server manually first that serves our React app such that we can load it in our Cypress tests environment

All we need to do is to use

npx nx e2e e2e

This might look weird initially, but basically, we run the e2e target (see e2e/project.json) on the e2e project.

1{ 2 "name": "e2e", 3 "$schema": "../node_modules/nx/schemas/project-schema.json", 4 "sourceRoot": "e2e/src", 5 "projectType": "application", 6 "targets": { 7 "e2e": { 8 "executor": "@nrwl/cypress:cypress", 9 "options": { 10 "cypressConfig": "e2e/cypress.config.ts", 11 "devServerTarget": "awesomereactapp:serve:development", 12 "testingType": "e2e" 13 }, 14 "configurations": { 15 "production": { 16 "devServerTarget": "awesomereactapp:serve:production" 17 } 18 } 19 }, 20 ... 21 } 22} 23

By default, these tests run in headless mode, but you can pass --watch to run it interactively with the Cypress test runner such that the tests get re-executed whenever we change our source.

Want Cypress Component testing? There’s an Nx generator that can help set that up. Check out the docs: /nx-api/react/generators/cypress-component-configuration

Linting

And similarly, linting can be triggered by running the following command:

npx nx lint

There’s a .eslintrc.json file already at the workspace's root that contains some best practices rules.

Customize Vite and Vitest

The project setup is made in a way that you can easily customize your Vite and Vitest setup. Just open the pre-generated vite.config.ts at the root of your workspace and add custom Vite plugins or fine-tune Vitest.

1/// <reference types="vitest" /> 2import { defineConfig } from 'vite'; 3import react from '@vitejs/plugin-react'; 4import viteTsConfigPaths from 'vite-tsconfig-paths'; 5 6export default defineConfig({ 7 server: { 8 port: 4200, 9 host: 'localhost', 10 }, 11 plugins: [ 12 react(), 13 viteTsConfigPaths({ 14 root: './', 15 }), 16 ], 17 // vitest config 18 test: { 19 globals: true, 20 cache: { 21 dir: './node_modules/.vitest', 22 }, 23 environment: 'jsdom', 24 include: ['src/**/*.{test,spec}.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], 25 }, 26}); 27

Hidden gem: Caching

Nx is known for its caching that helps optimize the speed in monorepos. Caching takes the inputs (the command, source files, environment variables…) and computes a hash.

On every run, Nx compares that hash against a local cache folder. If the hash exists, Nx restores the command line output and potential artifacts (JS, CSS,… files) produced by a previous run. This helps speed up computation because you don’t run it if you don’t need to.

See Nx the docs for more info: /concepts/how-caching-works

While this obviously makes a lot of sense in a monorepo, it can also help speed up single-project workspaces. Most projects have multiple targets, such as build, test, lint. These can be cached! Imagine you have a PR where you change some *.spec.ts files because you add a test or fix some. Your CI script probably runs all the targets (build, test, lint) all the time. And it should totally do that. But you could avoid the build step because your spec file should not influence that outcome. As such, it could be restored from the cache. test needs to run and potentially also lint if you run linting also for spec files.

You can fine-tune what goes into the cache for each command. More on the Nx docs

Hidden gem: Easily modularize your app

Imagine a storefront application. You will probably have domain areas like

  • Product list — which would have facilities for listing all currently available products, their ratings, user reviews etc
  • Orders — for viewing your currently open orders as a user or browsing past orders. Things like placing a new order or triggering a refund on a previously acquired product
  • Payments — for handling the payment flow, asking for the credit card, triggering the payment, and starting the order placement process once the payment is successful
  • Authentication — which handles the whole signup/login flow and provides lower-level utilities for other domain areas in the app, like getting access to the current user.
  • User Profile — which manages everything user related. Think of it when you access Amazon and go to your account. Things like managing your addresses

We’re just scratching the surface here. This can become big quickly. The only way to manage such a structure with the current tooling (including CRA) is to organize these domains in folders. So you’d have something like this in a CRA setup:

1cra-app 2 ├─ public/ 3 ├─ src/ 4 │ ├─ authentication/ 5 │ │ ├─ current-user/ 6 │ │ │ ├─ ... 7 │ │ │ └─ index.ts 8 │ │ ├─ login/ 9 │ │ └─ signup/ 10 │ ├─ orders/ 11 │ │ ├─ checkout/ 12 │ │ ├─ place-order/ 13 │ │ ├─ refund/ 14 │ │ └─ order-list/ 15 │ ├─ payments/ 16 │ ├─ products/ 17 │ ├─ user-profile/ 18 │ │ ├─ addresses/ 19 │ │ └─ credit-cards/ 20 │ ├─ App.css 21 │ ├─ App.tsx 22 │ ... 23 ├─ package-lock.json 24 ├─ package.json 25 └─ README.md 26

Most devtools (including CRA) force you into a monolithic structure, where you divide your features into folders. Folders are limited in terms of isolation, though; as your application grows, this might quickly go out of hand.

We can impose a different, stronger structure with Nx by extracting these areas into dedicated libraries or modules. These live side-by-side with your application. Let’s say we have a folder named “domains” which contains these domain areas. Then you can easily generate a new library with the following command:

npx nx g @nrwl/react:lib checkout --directory=domains/orders/checkout --bundler=none

The above command creates a new “ checkout “ library in the domains/orders/ folder. Here's what it looks like:

1awesomereactapp 2├─ public 3│ └─ favicon.ico 4├─ src 5│ ├─ app 6│ ├─ ... 7├─ domains 8│ └─ orders 9│ └─ checkout 10│ ├─ src 11│ │ ├─ index.ts 12│ │ └─ lib 13│ │ ├─ domains-orders-checkout.module.css 14│ │ ├─ domains-orders-checkout.spec.tsx 15│ │ └─ domains-orders-checkout.tsx 16│ ├─ tsconfig.json 17│ ├─ tsconfig.lib.json 18│ ├─ tsconfig.spec.json 19│ └─ vite.config.ts 20├─ ... 21├─ index.html 22├─ package-lock.json 23├─ package.json 24├─ ... 25├─ tsconfig.app.json 26├─ tsconfig.base.json 27├─ tsconfig.json 28├─ tsconfig.spec.json 29└─ vite.config.ts 30

Notice the domains/orders/checkout/src/index.ts: this is the public API of the checkout library where you can decide what to export and what should remain private within the library. This conscious process of selecting what to expose and what not leads to a much stronger encapsulation than just a folder structure. It also greatly helps with the maintainability aspect as your app grows.

When generating the library, a TypeScript path mapping is automatically created in the root-level tsconfig.base.json:

1{ 2 "compileOnSave": false, 3 "compilerOptions": { 4 ... 5 "paths": { 6 "@awesomereactapp/domains/orders/checkout": [ 7 "domains/orders/checkout/src/index.ts" 8 ] 9 } 10 }, 11 "exclude": ["node_modules", "tmp"] 12} 13

In this way, anything that’s being exported from the checkout library can be consumed like

1import { SomeComponent } from '@awesomereactapp/domains/orders/checkout'; 2

You can also just run linting or testing in isolation for these new libraries:

npx nx test domains-orders-checkout

And obviously, caching (as seen previously) would work on these new libraries as well.

Note, _domains-orders-checkout_ is the unique name of the project, composed by its file structure. You can change the name in the _domains/orders/checkout/project.json_ if you'd like.

Hidden gem: Visualize your architecture

Another side-effect of splitting up your codebase into libraries is that your code structure and architecture emerge and becomes visible. Nx comes with a graph command built-in, so you can even visualize it:

npx nx graph

It becomes even more interesting if you select the “Group by folder” checkbox as the domains become visible at that point:

Note this is a hypothetical app to demo some of the features of the Nx graph visualization. Some of the connections might only make a little sense.

Hidden gem: Guard your boundaries

Scaling a software product is more than just the initial structuring and modularization. It consists of a constant ongoing process of ensuring modules stay in shape and don’t contain any undesired cross-references or circular dependencies. You could leverage the Nx graph to verify that visually, but that doesn’t scale.

To help with that, Nx has a built-in module boundary lint rule. Projects can be assigned “tags”, like type:domain, type:utils, type:shared and domain:products, domain:orders, domain:auth. These tags can be assigned in the project.json, like

1{ 2 // ... more project configuration here 3 "tags": ["domain:products", "type:domain"] 4} 5

Note that _type:domain_ or _domain:products_ are really just strings. You can define them however you want.

In the .eslintrc.base.json you can then define the rules. Here for instance we're stating that a library of type:utils can only depend on other utility libraries, while a type:domain can depend on both, other domain libraries as well as utility libraries.

.eslintrc.base.json
1{ 2 "overrides": [ 3 { 4 "rules": { 5 "@nrwl/nx/enforce-module-boundaries": [ 6 "error", 7 { 8 "depConstraints": [ 9 { 10 "sourceTag": "type:utils", 11 "onlyDependOnLibsWithTags": ["type:utils"] 12 }, 13 { 14 "sourceTag": "type:domain", 15 "onlyDependOnLibsWithTags": ["type: domain", "type:utils"] 16 }, 17 { 18 "sourceTag": "domain:products", 19 "onlyDependOnLibsWithTags": ["domain:products", "domain:orders"] 20 } 21 ] 22 } 23 ] 24 } 25 } 26 ] 27} 28

If some of these lint rules need to be followed, your editor will show it right in your code, and you can also run lint checks for each PR on CI.

If you’re curious, you can read more here.

Hidden gem: Just run what changed

In such a modular structure (as shown above) where your code is organized in smaller modules/libraries, it is very common that a given team member just works within a single domain area. Hence, very often PRs just touch a subset of the entire set of libraries. Nx comes with a backed-in command that allows you to take advantage of that on CI, using the so-called “affected commands”.

Let’s say we make a change in the product-detail library of our application. This would affect all other libraries that depend on it. You can also visualize it by running

npx nx affected:graph

To run tasks only for the affected areas, use:

npx nx affected:<target>

To make a concrete example, running just tests for those projects:

npx nx affected:test

Hidden gem: A dedicated Editor extension

If you are not the “command line interface type” developer and you’d rather prefer something integrated within your IDE, then there’s good news. The Nx core team also ships a dedicated VSCode extension: Nx Console.

It has a dedicated view within VSCode to trigger common commands, browse the workspace structure and even inline render the graph.

It also comes with contextual menus to quickly access most of the commonly used functionality:

Here’s a walkthrough video showing some of the powerful capabilities of Nx Console:

Hidden gem: Automated Upgrades

To keep workspaces evergreen, Nx comes with automated code migrations that

  • upgrade your package.json packages to the next version
  • can automatically update your configuration files if necessary
  • can automatically update your source files if necessary (in case of breaking changes in the API)

This allows for smooth transitions even if there are breaking changes. Just run

npx nx migrate latest

Nx gathers the currently installed packages and updates them to the latest version. If a package comes with Nx migration scripts, Nx collects them in a migrations.json file. You can inspect and then run them. This dramatically helps to keep your project tooling up to date.

Read more about how Nx migrations work on the docs.

Using CRA? Automatically migrate to Vite + Nx

If you’re currently on a CRA setup, you can easily migrate to an Nx + React + Vite-based setup by running the following command in your CRA project:

npx nx init

Read more on the Nx docs: /recipes/adopting-nx/adding-to-existing-project

If for some reason you cannot migrate to Vite just yet, you can pass _--vite=false_ to keep a Webpack-based setup for now.

Conclusion

Ready? Give it a try:

npx create-nx-workspace mycoolapp --preset=react-standalone

And let us know what you think :)

Learn more