Another developer started a product list page and has requested your help to complete it.
Here are the incomplete features.
All the changes you need to make will be in src/components/ProductView/ProductView.js.
Before starting to change code on a new-to-you project, it is helpful to become familiar with what’s already been done. In this case, the original developer used create-react-app as the starting point, then added a few of components to display a product list. They determined that the product data will come into the view through a prop named products
. (It does not matter where that data comes from as that will be managed outside the component.) Additionally, they finished the layouts in the UI with a box, image and button for each product as well as a panel on the right side.
When you run the application (npm install
followed by npm start
), you’ll see these UI elements display, but clicking does not do anything. That’s where you come in!
When you look at the code in ProductView
, you’ll find console.log
statements in the onClick
event handlers. That’s a clue some work is needed there. Additionally, you’ll find a comment that starts with “TODO”. While that might not be the only work required, these will be elements you need to modify.
An astute developer may also notice there are some props on other components that aren’t in use yet. The extra effort to find them while you’re coming up to speed may save you some time later since you may be able to leverage them to accomplish your tasks.
Next, you need to pick somewhere to start. The panel toggle seems good since it has a TODO
comment.
To replace a constant with a state variable, you’ll want to employ the useState
hook. Perhaps, something like this
Remember add its import at the top. It will come from the react
library.
The application should still run after this change, although VS Code and/or the JavaScript console will probably warn you that setSideOpen
is defined but not used.
Go ahead and use it to replace the console.log
associated with the toggle UI on the side panel. Remember
true
to false
(or false
to true
) is the purpose of the NOT operator (!
).All this means you could choose to toggle the value with a function call like setSideOpen(!sideOpen)
.
After replacing the console.log
you will likely have code that looks something like this.
If all the changes are correct, you will be able to click the “>” tab on the side panel to close it. Study the code and see if you can find where it switches to “<”. It is common practice in React to use a state variable in multiple UI updates.
Now, you will add a second state variable on your own. This one will track the selectedProduct
.
Hint: The previous developer left a
console.log
in the handler foronClick
of eachProductListItem
to show you the data that is needs to be stored in state for the selected item.
In order for the info to show in the side panel, the new state variable will need to be passed through a prop to the ProductDetails component. Go ahead and add that as well.
Hint: The
visible
prop is already receiving the value of theshowOpen
state variable. If you look at the ProductDetails component, you will find the other prop that the previous developer already coded.
When all is connected correctly, you can run the application, click each button below the images, and see associated details in the right side panel.
What would help the user experience is to also highlight the box in the list for the item whose details are showing. Again, the previous developer thought of this and created a prop to change the style, isSelected
.
Think about the condition for showing the selection and put that into the JSX.
HINT: Both
item
andselectedProduct
are objects that have “id” attributes.
Finally, you can run the application again to see both the highlight on list and the side panel both update.
Congratulations getting two state variables working. You’ll notice as you click around the application, these variables are completely independent. This can leave the user with a disconnnect where they see a highlight in the list without also seeing the details.
For example, if you close the side panel then click the buttons, you will not be able to see the details immediately. You have to take an extra step to open the side panel. A nicer user experience would be to automatically open the side panel when the details show.
Likewise, if you close the side panel while looking at an item’s details, the highlight still shows in the list. Users may appreciate it more if the selection automatically clears when the side panel is closed.
Fortunately, React hooks can help in both situations. Specifically, the useEffect
hook. This hook works by calling a function when one or more variables change. These variables can be either passed in through props or managed with state. In this component, you have two state variables to work with.
First, consider which variable change will trigger the effect (selectedProduct
), and which setter function will need to be called (setSideOpen(true)
).
Second, code the framework of the useEffect
hook. Each time you utilize useEffect
, it can start this same way.
Finally, put the trigger into the array, and the call to the setter in the function body. Consider any conditionals you might need (or there may be none).
You probably ended up with something like this
Try it out in the browser by refreshing, closing the side panel, clicking one of the buttons in the list. If all is well, the side panel will open while showing that item’s details.
Again, consider which variable change will trigger the effect (sideOpen
), and which setter function will need to be called (setSelection
).
Second, put in the framework for useEffect
. (You can find it above, if you’d like to look at it again.)
Finally, put the trigger into the array, and the call to the setter in the function body. Again, consider what conditionals are needed, if any.
You probably wrote something like this.
Refresh your browser, try to close the side panel. One of three things will happen
The first step is understanding when the values of state variables have changed. This means combining console.log
with useEffect
.
The next step is to look at the components themselves - both the one with useState
hooks, as well as its children.
Begin by added console.log
to both useEffect
functions, so you can see when they are being triggered. Many developers like to start with the logging as the first statement, before any conditionals they might have.
For example, if had the bug with side panel resetting but not closing, you now have useEffect
hooks that look something like this.
// Open side panel
useEffect(() => {
console.log(`selectedProduct CHANGED TO`, selectedProduct);
setSideOpen(true);
}, [selectedProduct]);
// Deselect product
useEffect(() => {
console.log(`sideOpen CHANGED TO`, sideOpen);
setSelectedProduct();
}, [sideOpen]);
When you watch the console in the browser, you’ll see output like this.
// After refresh
selectedProduct CHANGED TO undefined
sideOpen CHANGED TO true
// Select first item (abbreviated output)
selectedProduct CHANGED TO {id: 1, code: "abc", name: "Bistro Table ...
// Close side panel
sideOpen CHANGED TO false
ProductView.js:20 selectedProduct CHANGED TO undefined
ProductView.js:27 sideOpen CHANGED TO true
The reason for the side panel getting stuck open is evident in the last 3 statements. First, notice the side panel was closed (sideOpen CHANGED false
), which the useEffect
noticed to write out the log followed by setting selectedProduct
to undefined
. That means the other useEffect
picked up the change, wrote out the log statement, then updated sideOpen
again (back to true
).
The fix is to include conditionals so that the setters are called only as necessary.
// Open side panel when product is selected
useEffect(() => {
console.log(`selectedProduct CHANGED TO`, selectedProduct);
if (selectedProduct)
setSideOpen(true);
}, [selectedProduct]);
// Deselect product when side panel is closed
useEffect(() => {
console.log(`sideOpen CHANGED TO`, sideOpen);
if (!sideOpen)
setSelectedProduct();
}, [sideOpen]);
Congratulations on getting the bugs worked out of your implementation! A key skill for developers is trying each path a user can take to ensure the experiences are both correct and pleasing.
Add a console.log
before the return statement in ProductView
. Refresh the application in the browser and click through several scenarios for opening and closing the panel and selecting items in the list. Notice how often the output displays in the console.
Add a console.log
before the return statement in ProductListItem
and/or ProductDetails
. Refresh the application in the browser and click through several scenarios for opening and closing the panel and selecting items in the list. Notice how often the output displays in the console.
There are times when it would benefit your users if the application remembered some of the settings on refresh or between sessions. The useEffect
hook is one way (probably the best way) to put the value of a state variable into localStorage
. Likewise, the useState
definition is often the best time to get the value back out.
Remember that there will be no value in localStorage
the first time a user visits your application, so you’ll need to check for that and provide an appropriate default.
In this bonus phase, store the state of the side panel (whether opened or closed) in local storage. You can view (and edit or delete) local storage in the JavaScript tools in your browser by going to the “Application” tab and looking for “Local Storage” under “Storage” on the left.
HINT: If you get stuck trying to figure out how to determine if the boolean
false
has been put intolocalStorage
, consider storing it as a string because any non-empty string (including “false”) is considered truthy.