React Components
Because they keep getting props.
— From "1000 programming dad-jokes"
Why React Components?
The idea behind React is that you can think about a web page in terms of isolated components, where every component is an independent and reusable piece of the UI.
Consider a potential first version of our easy-opus app, which would need to display a bunch of tasks.
Every task could be its own component containing the task title, description and a button to delete the task. The title, description and the button could in turn be components themselves.
The tasks might then be grouped in a task list component, which in addition to the tasks, should have a button to add a new task.
Here is a simple mockup of the structure of our new web page:
Now comes the interesting part about React. In terms of implementation, a component is just a regular JavaScript function that takes the data it should render as input and returns the UI representation of the data as output.
Here is an example of how the Task
component could look like:
function Task({ title, description }: { title: string, description: string }) {
return (
<div>
<p>{title}</p>
<p>{description}</p>
<button>Delete</button>
</div>
);
}
Currently, our components have no styling. We'll fix this in another chapter, for now we just focus on the logic.
Let's have a look at this function.
Looking at its inputs, we see that it takes an object containing the data it should render.
Here the data to be rendered are the title
and the description
.
Looking at its outputs, we see that it returns the UI representation of the data.
Here the UI representation is a div
containing the title
, description
and a button.
Note that the button currently doesn't do much. We will change this later.
To see components in action, let's actually create a simple React project.
Create a React Project
To create a React project, we will use a tool called Vite, which provides various frontend tooling (including a development server and optimized builds).
First, let's create a new vite
project:
pnpm create vite
You will be asked to select various options.
Select React
for the framework and TypeScript + SWC
for the variant.
After you've created the new project, the tool will even output the commands you should execute afterwards (which is quite helpful).
Now navigate to the newly created directory, where your project resides:
cd example
You will see a couple of familiar files, including a package.json
which—among other things—includes the dependencies of the project.
Let's install those dependencies:
pnpm install
Finally, you can run a development server:
pnpm dev
If you go to http://localhost:5173
and look at the result, you will see a web page with some demo content.
Let's replace this demo content with our own components!
First, you can remove App.tsx
, App.css
, index.css
and the assets/
folder.
We will not need these right now.
Replace the code in main.tsx
with the following very simple task list:
import ReactDOM from 'react-dom/client';
function App() {
return (
<div>
<h1>Tasks</h1>
<ul>
<li>Read the Next.js book</li>
<li>Write a website</li>
</ul>
</div>
);
}
ReactDOM.createRoot(document.getElementById('root')!).render(<App />);
Save the file and navigate to http://localhost:5173
once more.
You will now see our component.
This is one of the greatest features of the development server—it will automatically recreate the page when you edit your code.
If you've paid attention so far, there should be a giant question mark in your head now: Why are we allowed to write HTML in our JavaScript?
The answer is simple: The syntax is not actually HTML (cue ominous music here).
Introducing JSX
Consider our App
function:
function App() {
return (
<div>
<h1>Tasks</h1>
<ul>
<li>Read the Next.js book</li>
<li>Write a website</li>
</ul>
</div>
);
}
While this might look like HTML, it is important to emphasize that this is absolutely not HTML. Instead this syntax is something called JSX, which is a syntax extension to JavaScript that can be transpiled to normal JavaScript.
To be precise, the JSX will be transpiled to calls to the React.createElement
function which will return regular JavaScript objects.
The root object will be dynamically added to the DOM using this line of code:
ReactDOM.createRoot(document.getElementById('root')!).render(<App />);
The difference between JSX and HTML is not just a nitpick. For example, in JSX we can (and will) use and nest our own components exactly like we would use and nest HTML elements.
Here is an example of a TaskList
component rendered inside the App
component:
import ReactDOM from 'react-dom/client';
function App() {
return <TaskList />;
}
function TaskList() {
return (
<div>
<h1>Tasks</h1>
<ul>
<li>Read the Next.js book</li>
<li>Write a website</li>
</ul>
</div>
);
}
ReactDOM.createRoot(document.getElementById('root')!).render(<App />);
While possible, it's much more complicated (and relatively rare) to add your own custom tags in HTML.
Additionally, you can include JavaScript expressions in JSX by wrapping them in curly braces {}
.
For example, this is a totally valid React component:
function Example() {
const x = 2;
const y = 2;
return <p>{x + y}</p>;
}
You can't do this in regular HTML.
Add Props to a React Component
JSX is great, but our current TaskList
component is not.
The main problem is that the data it represents is hardcoded into the component.
We fix that by passing properties (props
) to the component.
This is simply a JavaScript object containing the data the component should render.
In the case of the TaskList
we simply want to pass an array of strings named tasks
containing our—well—tasks.
We can then use the map
function to create a list item li
for each element of that array:
type TaskListProps = {
tasks: string[],
};
function TaskList(props: TaskListProps) {
return (
<ul>
{props.tasks.map((task) => (
<li>{task}</li>
))}
</ul>
);
}
This is already not too bad, but of course we want to take advantage of the latest and greatest JavaScript syntax there is.
Writing props.tasks
is annoying and will become even more annoying when we have a lot of props.
We can use object destructuring to alleviate this:
function TaskList({ tasks }: TaskListProps) {
return (
<ul>
{tasks.map((item) => (
<li>{item}</li>
))}
</ul>
);
}
You can see how all the concepts from the JavaScript and TypeScript chapters are coming together quite nicely.
Excellent! We can use the new component like this:
function App() {
return (
<div>
<h1>Tasks</h1>
<TaskList tasks={['Read the Next.js book', 'Write a website']} />
</div>
);
}
Note that there is a problem with the current implementation of our component. If you open your browser console, you will see an error:
Warning: Each child in a list should have a unique 'key' prop.
The reason for that is that React needs a way to identify which items in a list have changed or have been added or removed. It does that by looking at the keys of the items. These basically give the elements a stable identity.
Let's add IDs to the tasks and use the task IDs as keys for the component:
function App() {
const tasks = [
{
id: 'TSK-1',
title: 'Read the Next.js book',
},
{
id: 'TSK-2',
title: 'Write a website',
},
];
return (
<div>
<h1>Tasks</h1>
<TaskList tasks={tasks} />
</div>
);
}
function TaskList({ tasks }: TaskListProps) {
return (
<ul>
{tasks.map((item) => (
<li key={item.id}>{item.title}</li>
))}
</ul>
);
}
There is one final improvement we can make:
The div
element in App
is pretty useless, since it only serves the purpose of wrapping the h1
and TaskList
(since a component can only return a single JSX element).
Luckily, there is a special component called a React fragment which allows us to wrap multiple JSX elements without showing up in the DOM later.
You can create a fragment using the <></>
syntax:
function App() {
const tasks = [
{
id: 'TSK-1',
title: 'Read the Next.js book',
},
{
id: 'TSK-2',
title: 'Write a website',
},
];
return (
<>
<h1>Tasks</h1>
<TaskList tasks={tasks} />
</>
);
}
Improving our Project Structure
We should note that it's common practice to put components into separate files (unless two components are heavily related to each other).
In this example, we might put the TaskList
component into a file task-list.tsx
:
type Task = {
id: string,
title: string,
};
type TaskListProps = {
tasks: Task[],
};
export function TaskList({ tasks }: TaskListProps) {
return (
<ul>
{tasks.map((item) => (
<li key={item.id}>{item.title}</li>
))}
</ul>
);
}
Note that we have to export
the TaskList
component so that we can import
it in our index.tsx
file:
import { TaskList } from 'task-list';
function App() {
const tasks = [
{
id: 'TSK-1',
title: 'Read the Next.js book',
},
{
id: 'TSK-2',
title: 'Write a website',
},
];
return (
<div>
<h1>Tasks</h1>
<TaskList tasks={tasks} />
</div>
);
}
ReactDOM.createRoot(document.getElementById('root')!).render(<App />);
Now, we have two separate files.
The index.tsx
file contains our main App
component and the task-list.tsx
file contains our TaskList
component.
Additional components will go into additional files, allowing us to keep our codebase well structured and maintainable.