Continue working in the 06-jobs-api repository that you used for lessons 9 and 10. Before you start, create a new branch as usual, with a branch name of week12. (Week 11 was a catch up week, so you create the week12 branch when the week10 branch is active.)
Create a public directory under starter. This is for the HTML and JavaScript files for the front end. The HTML file is already created, and the code is below. Put that in the public directory with a file name of index.html. (Reminder: You do have to change the form, which below is for a job, to match your data model.) This lesson does not involve the creation of any new Node or Express function. It is all client side HTML and JavaScript.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Jobs List</title>
</head>
<body>
<h1>Jobs List</h1>
<hr>
<p id="message"></p>
<hr>
<div id="logon-register" style="display:none;">
<button type="button" id="logon">logon</button>
<button type="button" id="register">register</button>
</div>
<div id="logon-div" style="display:none;">
<form>
<div>
<label for="email">email:</label>
<input type="email" id="email">
</div>
<div>
<label for="password">password:</label>
<input type="password" id="password">
</div>
<button type="button" id="logon-button">logon</button>
<button type="button" id="logon-cancel">cancel</button>
</form>
</div>
<div id="register-div" style="display:none;">
<form>
<div>
<label for="name">name:</label>
<input type="text" id="name">
</div>
<div>
<label for="email1">email:</label>
<input type="email" id="email1">
</div>
<div>
<label for="password1">password:</label>
<input type="password" id="password1">
</div>
<div>
<label for="password2">verify password:</label>
<input type="password" id="password2">
</div>
<button type="button" id="register-button">register</button>
<button type="button" id="register-cancel">cancel</button>
</form>
</div>
<div id="jobs" style="display:none;">
<p id="jobs-message"></p>
<table id="jobs-table">
<tr id="jobs-table-header">
<th>Company</th>
<th>Position</th>
<th>Status</th>
<th colspan="2"></th>
</tr>
</table>
<button type="button" id="add-job">add job</button>
</div>
<div id="edit-job" style="display:none;">
<form>
<div>
<label for="company">company:</label>
<input type="text" id="company">
</div>
<div>
<label for="position">position:</label>
<input type="text" id="position">
</div>
<div>
<label for="status">status:</label>
<select id="status">
<option value="pending">pending</option>
<option value="interview">interview</option>
<option value="declined">declined</option>
</select>
</div>
<button type="button" id="adding-job">add</button>
<button type="button" id="edit-cancel">cancel</button>
</form>
</div>
</body>
</html>
This front end uses a single page style. There are multiple views in the page, in different DIV entries, but which of these is visible is controlled by the JavaScript you write.
Edit app.js. Comment out the following lines:
app.get('/', (req, res) => {
res.send('<h1>Jobs API</h1><a href="/api-docs">Documentation</a>');
});
Add the following line below these commented out lines:
app.use(express.static('public'))
Start the server and go to localhost:3000 in your browser. You see the page, but it does not do anything. This is because there is no JavaScript to go with it. You create that now. This is the JavaScript for the front end. Front end JavaScript does not run in Node. Instead it is delivered to the browser and runs in the browser context, with full access to the document, window, and DOM.
There are various divs to be manipulated by this JavaScript. (Again, you have to change this code to match your data model.) The code is segmented into various modules. There is one module, index.js, to initialize window handling. There are five divs, only one of which will show at a time: one to select logon or register, one to do logon, one to do register, one to display the jobs, and one to add or edit a job. For each of these divs, there is a separate module controlling its operation.
To begin, add the following line to index.html, right above the close of the body tag:
<script src="./index.js" type="module"></script>
These modules call one another using the exports that each provides. For this to work, you must declare it as type “module”. Create index.js in the public directory.The index.js module should read as follows:
let activeDiv = null;
export let enabled = true;
export const enable = (state) => {
enabled = state
}
export const setDiv = (div) => {
if (div != activeDiv) {
if (activeDiv) {
activeDiv.style.display = "none";
}
div.style.display = "block";
activeDiv=div
}
};
export let token = null;
export const setToken = (value) => {
token=value;
}
export let message = null;
import { showJobs, handleJobs } from "./jobs.js";
import { showLoginRegister, handleLoginRegister } from "./loginRegister.js";
import { handleLogin } from "./login.js";
import { handleAddEdit } from "./addEdit.js";
import { handleRegister } from "./register.js";
document.addEventListener("DOMContentLoaded", () => {
token = localStorage.getItem("token");
message = document.getElementById("message");
handleLoginRegister();
handleLogin();
handleJobs();
handleRegister();
handleAddEdit();
if (token) {
showJobs();
} else {
showLoginRegister();
}
});
Remember that this code is running in the browser, and not in Node. That means you must use import and not require. We need to keep track of the active div, and that is stored in activeDiv. We export a function that sets the active div, making it visible and hiding the previous active div. We also have to have a means of disabling input. This is because we will use asynchronous functions, and the application can get confused if input comes in while these are in progress. We also have to have a global place to display messages. We also have to keep track of whether the user is logged in. We do that in a token in local storage (although this creates security risks as previously described.) When local storage is used, the user remains logged in even if the page is refreshed. Once the DOM is loaded, we initialize the handlers for each of the divs. Then, if the user is logged in, we display the list of jobs. If the user is not logged in, we display the initial panel with a button for logon and a button for register. Note that we need to provide and to export functions to set the enabled flag and the token. This is because one can’t write directly to these variables from other modules.
You will need to create loginRegister.js, register.js, login.js, jobs.js, and addEdit.js, all in the public directory. The loginRegister.js module is as follows:
let loginRegisterDiv = null
import { enabled, setDiv } from "./index.js"
import { showLogin } from "./login.js"
import { showRegister } from "./register.js"
export const handleLoginRegister = () => {
loginRegisterDiv = document.getElementById("logon-register")
const login = document.getElementById("logon")
const register = document.getElementById("register")
loginRegisterDiv.addEventListener("click", (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === login) {
showLogin()
} else if (e.target === register) {
showRegister()
}
}
})
}
export const showLoginRegister = () => {
setDiv(loginRegisterDiv)
}
Each of the div handling modules follow this pattern. Required imports (used when one div handler calls another) are resolved up front. Then, within the handler function, the div and its controls are resolved. Also within the handler function, event handler is declared to handle mouse clicks within the div. A separate function handles display of the div. (React works in similar fashion, if you know that framework — but this lesson does not use React.)
The register.js module is as follows:
let registerDiv = null;
import { enabled, setDiv, message, token, enable, setToken } from "./index.js";
import { showLoginRegister } from "./loginRegister.js";
import { showJobs } from "./jobs.js";
let name = null;
let email1 = null;
let password1 = null;
let password2 = null;
export const handleRegister = () => {
registerDiv = document.getElementById("register-div");
email1 = document.getElementById("email1");
password1 = document.getElementById("password1");
password2 = document.getElementById("password2");
name = document.getElementById("name");
const registerButton = document.getElementById("register-button");
const registerCancel = document.getElementById("register-cancel");
registerDiv.addEventListener("click", (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === registerButton) {
showJobs();
} else if (e.target === registerCancel) {
showLoginRegister();
}
}
});
};
export const showRegister = () => {
email1.value = null;
password1.value = null;
password2.value = null;
setDiv(registerDiv);
};
The login.js module is as follows:
let loginDiv = null;
import { enabled, setDiv, token, message, enable, setToken } from "./index.js";
import { showLoginRegister } from "./loginRegister.js";
import { showJobs } from "./jobs.js";
let email = null;
let password = null;
export const handleLogin = () => {
loginDiv = document.getElementById("logon-div");
email = document.getElementById("email");
password = document.getElementById("password");
const logonButton = document.getElementById("logon-button");
const logonCancel = document.getElementById("logon-cancel");
loginDiv.addEventListener("click", (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === logonButton) {
showJobs();
} else if (e.target === logonCancel) {
email.value = "";
password.value = "";
showLoginRegister();
}
}
});
};
export const showLogin = () => {
email.value = null;
password.value = null;
setDiv(loginDiv);
};
The jobs.js module is as follows:
let jobsDiv = null;
import { enabled, setDiv, message, setToken, token, enable } from "./index.js";
import { showLoginRegister } from "./loginRegister.js";
import { showAddEdit } from "./addEdit.js";
let jobsTable = null;
let jobsTableHeader = null;
export const handleJobs = () => {
jobsDiv = document.getElementById("jobs");
const logoff = document.getElementById("logoff");
const addJob = document.getElementById("add-job");
jobsTable = document.getElementById("jobs-table");
jobsTableHeader = document.getElementById("jobs-table-header");
jobsDiv.addEventListener("click", (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === addJob) {
message.textContent = "";
showAddEdit(null);
} else if (e.target === logoff) {
showLoginRegister();
}
});
};
export const showJobs = async () => {
setDiv(jobsDiv);
};
The addExport.js module is as follows:
let addEditDiv = null;
import { enable, enabled, message, setDiv, token } from "./index.js";
import { showJobs } from "./jobs.js";
let company = null;
let position = null;
let status = null;
let addingJob = null;
export const handleAddEdit = () => {
addEditDiv = document.getElementById("edit-job");
company = document.getElementById("company");
position = document.getElementById("position");
status = document.getElementById("status");
addingJob = document.getElementById("adding-job");
const editCancel = document.getElementById("edit-cancel");
addEditDiv.addEventListener("click", (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === addingJob) {
showJobs();
}
}
});
};
export const showAddEdit = (job) => {
setDiv(addEditDiv)
}
Create all these files and then try the application out. You will find that the application is now responsive, and you can navigate between the active divs. However, the application still does not do much, because, of course, there is no code to communicate with the back end. This is to be added using the fetch() function. Fetch is asynchronous. In the code that follows, we use the async/await pattern, so of course whenever that is used, the surrounding function must be declared as async. Also, we need to disable input for the period in which the async operation is in progress. We do this by setting the enabled flag. Our click handlers ignore clicks if they occur while the enabled flag is false. We may get an error, so the async operations must be surrounded with a try/catch. If an error occurs, we notify the user, but we also log the error to the console. You may not want to log all errors to the console in a production application, but it is wise to do it when you are developing, so that you can find your own errors.
First, we’ll make register and logon work. For either register or logon, if the step is successful, the back end returns a JWT token. This is stored for use in accessing jobs records.
Adding these capabilities to register.js gives the following:
let registerDiv = null;
import { enabled, setDiv, message, token, enable, setToken } from "./index.js";
import { showLoginRegister } from "./loginRegister.js";
import { showJobs } from "./jobs.js";
let name = null;
let email1 = null;
let password1 = null;
let password2 = null;
export const handleRegister = () => {
registerDiv = document.getElementById("register-div");
email1 = document.getElementById("email1");
password1 = document.getElementById("password1");
password2 = document.getElementById("password2");
name = document.getElementById("name");
const registerButton = document.getElementById("register-button");
const registerCancel = document.getElementById("register-cancel");
registerDiv.addEventListener("click", async (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === registerButton) {
if (password1.value != password2.value) {
message.textContent = "The passwords entered do not match.";
} else {
enable(false);
try {
const response = await fetch("/api/v1/auth/register", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
name: name.value,
email: email1.value,
password: password1.value,
}),
});
const data = await response.json();
if (response.status === 201) {
message.textContent = `Registration successful. Welcome ${data.user.name}`;
setToken(data.token);
localStorage.setItem("token", token);
name.value = "";
email1.value = "";
password1.value = "";
password2.value = "";
showJobs();
} else {
message.textContent = data.msg;
}
} catch (err) {
console.log(err);
message.textContent = "A communications error occurred.";
}
enable(true);
}
} else if (e.target === registerCancel) {
name.value = "";
email1.value = "";
password1.value = "";
password2.value = "";
showLoginRegister();
}
}
});
};
export const showRegister = () => {
email1.value = null;
password1.value = null;
password2.value = null;
setDiv(registerDiv);
};
Note that the values the user types are cleared out after processing. We do not want those to live on in memory.
The login.js module becomes:
let loginDiv = null;
import { enabled, setDiv, token, message, enable, setToken } from "./index.js";
import { showLoginRegister } from "./loginRegister.js";
import { showJobs } from "./jobs.js";
let email = null;
let password = null;
export const handleLogin = () => {
loginDiv = document.getElementById("logon-div");
email = document.getElementById("email");
password = document.getElementById("password");
const logonButton = document.getElementById("logon-button");
const logonCancel = document.getElementById("logon-cancel");
loginDiv.addEventListener("click", async (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === logonButton) {
enable(false);
try {
const response = await fetch("/api/v1/auth/login", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
email: email.value,
password: password.value,
}),
});
const data = await response.json();
if (response.status === 200) {
message.textContent = `Logon successful. Welcome ${data.user.name}`;
setToken(data.token);
localStorage.setItem("token", token);
email.value = "";
password.value = "";
showJobs();
} else {
message.textContent = data.msg;
}
} catch (err) {
console.log(err);
message.textContent = "A communications error occurred.";
}
enable(true);
} else if (e.target === logonCancel) {
email.value = "";
password.value = "";
showLoginRegister();
}
}
});
};
export const showLogin = () => {
email.value = null;
password.value = null;
setDiv(loginDiv);
};
Make these changes and test the application again. You should find that you can register and logon. Logoff doesn’t work right at present, but this can be corrected in jobs.js with the following change:
} else if (e.target === logoff) {
setToken(null);
localStorage.removeItem("token");
message.textContent = "You have been logged off.";
jobsTable.replaceChildren([jobsTableHeader]);
showLoginRegister();
}
Note that logoff involves no communication with the back end. The user is logged off by deleting the JWT from memory. We also have to clear the jobs data from memory, for security reasons. That’s what the replaceChildren does.
Next we need to make the changes so that we can create job entries. The addEdit.js module is changed as follows:
addEditDiv.addEventListener("click", async (e) => {
if (enabled && e.target.nodeName === "BUTTON") {
if (e.target === addingJob) {
enable(false);
let method = "POST";
let url = "/api/v1/jobs";
try {
const response = await fetch(url, {
method: method,
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${token}`,
},
body: JSON.stringify({
company: company.value,
position: position.value,
status: status.value,
}),
});
const data = await response.json();
if (response.status === 200 || response.status === 201) {
//successful create
if (response.status === 200) {
message.textContent = "The job entry was updated.";
} else {
message.textContent = "The job entry was created.";
}
company.value = "";
position.value = "";
status.value = "pending";
showJobs();
} else {
message.textContent = data.msg;
}
} catch (err) {
console.log(err);
message.textContent = "A communication error occurred.";
}
enable(true);
} else if (e.target === editCancel) {
message.textContent = "";
showJobs();
}
}
}
In this case, we have to pass the JWT in the header for the call to work. Once you have added this code, try out the application again. You are now able to add entries, but you can’t actually see them. Next you add the code to populate the table of jobs entries. Of course, this involves another fetch operation, passing the JWT as for the add. Then the results are used to populate a table.
There is a somewhat tricky part to this. We want to have edit and delete buttons for each row of the table. But, how do we associate an edit button with the edit operation, and when it is clicked, how do we know which entry is to be edited? This is done as follows. The edit buttons are given a class of “editButton”, and similarly, the delete buttons are given a class of “deleteButton”. We can also associate a hash called dataset with each button, and we store in dataset.id the id of the entry, as returned from the database.
It looks like this:
export const showJobs = async () => {
try {
enable(false);
const response = await fetch("/api/v1/jobs", {
method: "GET",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${token}`,
},
});
const data = await response.json();
let children = [jobsTableHeader];
if (response.status === 200) {
if (data.count === 0) {
jobsTable.replaceChildren(...children); // clear this for safety
} else {
for (let i = 0; i < data.jobs.length; i++) {
let editButton = `<td><button type="button" class="editButton" data-id=${data.jobs[i]._id}>edit</button></td>`;
let deleteButton = `<td><button type="button" class="deleteButton" data-id=${data.jobs[i]._id}>delete</button></td>`;
let rowHTML = `<td>${data.jobs[i].company}</td><td>${data.jobs[i].position}</td><td>${data.jobs[i].status}</td>${editButton}${deleteButton}`;
let rowEntry = document.createElement("tr");
rowEntry.innerHTML = rowHTML;
children.push(rowEntry);
}
jobsTable.replaceChildren(...children);
}
} else {
message.textContent = data.msg;
}
} catch (err) {
console.log(err);
message.textContent = "A communication error occurred.";
}
enable(true);
setDiv(jobsDiv);
};
So, plug this code into jobs.js at the appropriate point, and then try the application again. You should now be able to see the entries for each job.
However, the edit and delete buttons don’t actually work. This is because the click handler in jobs.js is ignoring them. We can add a section to the click handler to remedy this.
} else if (e.target.classList.contains("editButton")) {
message.textContent = "";
showAddEdit(e.target.dataset.id);
}
The dataset.id contains the id of the entry to be edited. That is then passed on to the showAddEdit function. So we need to change that function to do something with this parameter. This function is in addEdit.js, and should be changed as follows:
export const showAddEdit = async (job) => {
if (!job) {
company.value = "";
position.value = "";
status.value = "pending";
addingJob.textContent = "add";
message.textContent = "";
setDiv(addEditDiv);
} else {
enable(false);
try {
const response = await fetch(`/api/v1/jobs/${job}`, {
method: "GET",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${token}`,
},
});
const data = await response.json();
if (response.status === 200) {
company.value = data.job.company;
position.value = data.job.position;
status.value = data.job.status;
addingJob.textContent = "update";
message.textContent = "";
addEditDiv.dataset.id = job;
setDiv(addEditDiv);
} else {
// might happen if the list has been updated since last display
message.textContent = "The jobs entry was not found";
showJobs();
}
} catch (err) {
console.log(err);
message.textContent = "A communications error has occurred.";
showJobs();
}
enable(true);
}
};
With this change, the add/edit window will be shown with the right values. If an add is being done, the function is called with a null parameter, and the form comes up blank with an add button. If an edit is being done, the function is called with the id of the entry to edit. That is retrieved from the database and the fields are populated, and the button is changed to say update. We also store the id of the entry in the dataset.id of the addEdit div, so we keep track of which entry is to be updated.
So far, so good, but what happens when the user clicks on the update button? In this case, we need to do a PATCH instead of a POST, and we need to include the id of the entry to be updated in the URL. So we need the following additional changes to addEdit.js:
if (e.target === addingJob) {
enable(false);
let method = "POST";
let url = "/api/v1/jobs";
if (addingJob.textContent === "update") {
method = "PATCH";
url = `/api/v1/jobs/${addEditDiv.dataset.id}`;
}
try {
const response = await fetch(url, {
method: method,
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${token}`,
},
body: JSON.stringify({
company: company.value,
position: position.value,
status: status.value,
}),
});
const data = await response.json();
if (response.status === 200 || response.status === 201) {
//successful create
if (response.status === 200) {
message.textContent = "The job entry was updated.";
} else {
message.textContent = "The job entry was created.";
}
company.value = "";
position.value = "";
status.value = "pending";
showJobs();
} else {
message.textContent = data.msg;
}
} catch (err) {
console.log(err);
message.textContent = "A communication error occurred.";
}
enable(true);
}
Make these changes, and edits should work. Make sure that adds still work correctly.
This completes all CRUD operations except for delete. You will notice that the delete buttons don’t work. FIx this. Implement the delete function following the pattern used for the edit button. You call the jobs delete API, and in the URL you include the ID of the entry to be deleted. Then the home page is displayed again. Note: There is an error in the implementation of the delete operation in the jobs controller. The instructor’s guidance is to use this line:
res.status(StatusCodes.OK).send()
This is incorrect, because an empty body is not valid JSON. Change it to:
res.status(StatusCodes.OK).json({msg: "The entry was deleted."})
If you do not make this change, an exception is thrown on the front end when you attempt to parse the response body with response.json().
Once you have everything working, do a git, add, and commit to the week12 branch, and push it to your github repository. Then modify the Render.com deployment you have to point to the new branch. This will cause your new code to be deployed to Render.com. Verify that your application front end is working on Render.com.
This is a little cheat sheet so that you don’t find it too hard.