Real time App with React and WebSocket - Part 1
Learn the basics to make a Real Time web application with React and the WebSocket API. We will bootstrap a small Client-Server app.
This project is in 2 parts. Part 1 explains the goal and setup the front-end app. Part 2 explains how to setup the server and use Zustand as a state management tool.
Table of Content
Part 1: Building the frontend
- Technologies Involved
- Project Scope
- Implementation
- Step 1: Create a new Project
- Step 2: Create the Front End
- A Real Time Web Server
- The HTTP Server
- The Socket.IO Server
- State Management with Zustand
- Creating a Zustand store
- Using the Store
- Conclusion
Technologies involved
- React - We'll be using the latest React version (16.13.1) as our web UI library.
- Next.js - a modern solution to build React web applications.
- Zustand - as a small yet efficient state management library. It will make it simple for us to subscribe to both state change and WebSocket (WS) events.
- Socket.IO - as our WS implementation for the server and client.
- Yarn Berry (yarn version 2)
Project scope
We will do simple tic-tac-toe game:
The server will provide the game state and an AI that you can play against.
WebSocket is an event based protocol, we will send event back and forth between our client and server. So let's start defining the rules and events for our game:
1/ The user can request to play a new game. This will emit a "new-game" event. The server will create a new game and return its state with the event "state".
2/ The user can emit "play" event with the box id as payload (0 to 8). The server will check if the play is valid and update the game by emitting an event "state".
3/ After the player plays a valid move, the server will play for the Bot and emit an event "state".
4/ After each move, the server checks if the current player won. If the winner information will be stored in the game state. If the game ends (player wins, loses or ties) the server will emit a "state" event.
6/ The player can start a new game at any moment by requesting a new game (emit "new-game")
Implementation
At any time you can see the full code in this GitHub repository.
First we will start with the project structure. Then continue with implementing a static front-end.
In Part 2 we will create our server and finally connect it with our front-end application.
Step 1: Create a new Project
Let's start by creating a new project with git and yarn 2:
$ mkdir tic-tac-toe && cd $_
$ git init
$ yarn set version berry
$ yarn init
$ git add . && git commit -m "initial commit"
These commands will:
- create a new project directory and navigate into it
- initialize a new git repository
- install yarn berry locally into the project
- generate a package.json file
- add and commit the changes
By default zero install should be configured (check the .gitignore
file). If you wish, you can change this behavior before committing.
If you already have yarn berry installed globally, you don't have to install it locally and you could use yarn create next-app
instead of most of the previous commands.
Step 2: Create the Front End
We will use Next.js for that. The motivation is because Next.js has become a popular Framework for React Apps which includes a lot of features and optimizations out of the box. One feature that will be useful for us in this project is the ability to extend the Next's server. We will leverage this feature to create our WebSocket server so we don't have to setup our own server.
Let's start with installing the dependencies. We can follow Next.js documentation to get the latest instructions:
$ yarn add next react react-dom
If Yarn Berry is not installed globally then yarn create next-app
will use Yarn 1.x and You'll need to set it up and reinstall the dependencies.
The documentation then asks us to add the following scripts in package.json
:
Now let's create our first page.
First, Next.js will search for a folder called pages
and load each file as different pages. Next.js will search for the folder in the root directory or the src
directory. I prefer having the source in the src
directory, so I'll create the folder like this:
$ mkdir -p src/pages
$ ls -R
.:
package.json README.md src yarn.lock
./src:
pages
./src/pages:
Then Finally, we can create our first index page:
I'm using Typescript for this repository so I'm creating a .tsx
file instead of a .jsx
one.
If we run yarn dev
now, we will get the following error:
ready - started server on http://localhost:3000
It looks like you're trying to use TypeScript but do not have the required package(s) installed.
Please install typescript, @types/react, and @types/node by running:
yarn add --dev typescript @types/react @types/node
If you are not trying to use TypeScript, please remove the tsconfig.json file from your package root (and any TypeScript files in your pages directory).
Next.js will notice that you're trying to use Typescript but we haven't installed the dependencies yet. Add them as asked: yarn add --dev typescript @types/react @types/node
Now we can run yarn dev
again and navigate to localhost:3000/ :
$ yarn dev
ready - started server on http://localhost:3000
We detected TypeScript in your project and created a tsconfig.json file for you.
event - compiled successfully
event - build page: /next/dist/pages/_error
wait - compiling...
event - compiled successfully
Our Next.js app should now be working fine!
If your IDE start throwing "module not found" errors, you might need to do some special configuration for it to work with Yarn Plug&Play system. The easiest way is to follow Yarn's documentation
For example : yarn dlx @yarnpkg/pnpify --sdk vscode
Now let's work a bit on the UI:
That's it for the UI part ! If we run the app again, we should be able to see a TicTacToe game and see the actions logged in the console.
Next we will create the server and connect them both using WebSocket and Zustand as a state manager: Real time App with React and web-sockets - Part 2