4-6 Hours
To build a UI for a portfolio-ready client-side application.
You'll build a web UI for the Eventonica app that will allow a user to interact with the code you already wrote in Part 1. This will be the basis of the subsequent parts as well.
In this tutorial, we will create:
- a form for creating a user with an email
- a form for deleting a user ID
- a list to display all users
You'll use React and JavaScript to build functionality for all the features listed in the main project README.
By the end of this lesson, your project will have:
- A user list
- A form to create a user and delete a user
By the end of this project, your project will also:
- Have a README
- Have CSS to customize the look
- All the features listed in the main Eventonica README
Remember to commit your code and push to GitHub frequently while working on this project. A good rule of thumb is whenever a new piece of functionality is working, commit with a descriptive message.
Before we create some components from scratch, your first challenge is to abstract an existing element into a new file, then export it as a component.
It's typical for a React developer to start in one file, then begin to divide it into new files and components as things grow. This keeps things organized and easy to read. There seems to be a general agreement that files should not be over 200 or 300 lines long, but of course there are plenty of opinions.
- Create a new folder in
client/src
namedcomponents
. Create a file calledfooter.jsx
withincomponents
.
Having a folder called components
will be useful because in the future, we can have other folders such as apis
or utils
. Separating concerns in a project is a good practice because it makes the files and folders easier to read and navigate.
-
Copy all the code from
<footer>
to</footer>
inApp.js
and paste it intocomponents/footer.jsx
. -
Use your knowledge of React to convert this to a component named
Footer
that is exported fromcomponents/footer.jsx
back to its original position inApp.js
. You can do it!
Before this stage, ensure you have a commit in place with the working app.
All your data added via the UI will be gone when you refresh the page, because all the JS files will be reloaded. In later weeks you'll learn how to save your data in databases instead of browser memory. Because of this, it's much easier to have some mock users and events at the top of your files. This way, every time you refresh, some data already exists.
- Copy the code in
App.js
in the<section className="user-management">
section. Create a new file in thecomponents
file calledUsers.jsx
, and paste the section code here. This section should be deleted fromApp.js
once it is inUsers.jsx
. Make sure your file looks like this:
import React from 'react';
const Users = () => {
return <section className="user-management">...</section>;
};
export default Users;
Use this Users
component in App.js
, and check that the section is rendering correctly.
- Try adding some mock users at the top of your
Users
file. For example,
const marlin = { name: 'Marlin', email: '[email protected]', id: '1' };
const nemo = { name: 'Nemo', email: '[email protected]', id: '2' };
const dory = { name: 'Dory', email: '[email protected]', id: '3' };
Later on you will add more fields to this form, and eventually store these users in a database.
-
Use
useState
to createusers
andsetUsers
. The default value forusers
can be a list of your mock users. For example,const [users, setUsers] = useState([marlin, nemo, dory])
-
Iterate through your user list and display their name and email in the list. Remember to have a key for each list item
With JS DOM, you would attach event handlers to the DOM. In React, you should not be doing any direct DOM manipulation (ie document.getElementById
, setting innerHTML
, etc)
See the example code below to compare Vanilla JS vs React (this example code may not be exactly how your code works but hopefully you can see how to apply the idea to yours)
let users = [];
document
.querySelector('#add-user-action')
.addEventListener('click', (event) => {
event.preventDefault();
const newId = parseInt(document.querySelector('#new-user-id').value);
const newName = document.querySelector('#new-user-name').value;
const newUser = { id: newId, name: newName };
users.push(newUser);
displayEvents(); // calls another function to refresh the DOM after users is updated
});
// id, name, and email are states that store what values the user types in those fields
// users is an array of user objects
// All of these states can be defined in the component
const handleSubmit = (e) => {
e.preventDefault();
const newUser = { id: id, name: name, email: email };
setUsers([...users, newUser]);
};
JS Syntax Checks:
Take a look at some of the object and array syntax in the code snippet above. Do you understand what [...users, newUser]
represents?
const newUser = {id: id, name: name, email: email}
can also be written as
const newUser = {id, name, email}
This is a ES2015 feature called Object property shorthand
-
Update the HTML form under "Add User" to have an ID field and an email field.
-
As the user types in the input fields, we will need to store what value is in each field. For now, we will have a different state for each field. For example, for storing what the user types in the name field, there can be a new state
const [name, setName] = useState('')
Every time the user types a name in the name field, the
name
state is updated. How can we achieve this?Hint: Input elements have a
value
property that contains the current input, see example at w3schools.
Your input field could look something like this:
<input
type="text"
id="add-user-name"
value={name}
onChange={(e) => setName(e.target.value)}
/>
Do the same thing for the ID and email fields.
Check : try console logging your name
, id
, and email
states. Do you see them changing as the user types?
-
When the user clicks submit, it should:
- create a new user with the ID, email, and name that the user entered
- add that new user to the list of users
Notice
preventDefault
used in the sample code -- Comment it out and see what happens. Learn more about it from the preventDefault MDN docs -
After creating a new user, you should see it appear in the list. How does this happen "automatically"?
-
Bonus: after creating a new user, you might see that the input fields still have the values filled in. How can you update the submit function so that the input values are reset after pressing "submit"?
The form should allow a user to be deleted from your list of users. For this functionality, we will also practice sending props.
-
Create a
DeleteUser
component. Move the delete user div into this file. (thediv
with thedelete-user
form and "Delete User"h3
). -
To delete a user, you'll need a way to uniquely identify what user should be deleted. We will ask the user for an ID, and delete the user with that ID. Create a state to store what
deleteId
the user has typed. -
When a user is deleted, we want the user object with that ID to be removed from the
users
list. How can you use thesetUsers
function to do that? Create a function in theUsers
component:
const deleteUser = (deleteId) => {
const newUsers = users.filter((i) => i.id !== deleteId);
setUsers(newUsers);
};
-
Pass this function as a prop to your
DeleteUser
component. -
Clicking submit in the delete form should call this function with the ID that the user entered. Don't forget
preventDefault()
. AfterdeleteUser
is called, a user should be removed from theusers
list inUsers.jsx
. Check this by looking at your<ul>
list of users, or by console loggingusers
state.
Check for understanding:
- Why do we define
deleteUser
inUsers.jsx
instead ofDeleteUser.jsx
? - Discuss with a partner how props and state work together for the delete functionality.
Note: You can find the code on Github