Week 7 Assignment
Overview
This week's assignment builds upon the Shopping List application we developed in Week 6. In Week 6, we created a simple shopping list application. The application allowed users to add items to a list, specifying details such as the item's name, quantity, and category.
In the Week 7 assignment, we extend this functionality by integrating third-party APIs to suggest meal ideas based on the items in the user's shopping list. This new feature will not only make our application more interactive and useful, but it will also give us a chance to practice working with external APIs.
Prerequisites
Before starting this assignment, you should have completed the Week 6 Assignment and have a working shopping list application. If you have not completed the Week 6 assignment, you have three options:
-
Use another student's Week 6 Assignment as a starting point for this week's assignment. Refer to the list of student repositories.
-
Use your assignment solution from Week 3 as a starting point for this week's assignment. Refer to the Week 3 Assignment for instructions.
-
Start from scratch. Instead of selecting an item from the list, the user can type in the item name and click a button to display the various meal ideas.
Optional Challenge 1: Minimal Instructions
If you're up for a bit of a challenge, try tackling this assignment without the comfort of detailed steps. Instead, use just the example output below as a guide. This will give you a chance to practice your problem-solving skills.
Optional Challenge 2: Display Ingredients for Each Meal
If you're up for an even bigger challenge, try displaying the ingredients for each meal. You will need to make a second API call to retrieve the ingredients for each meal.
Detailed Steps
Setup Files
Let's first create a new folder for this assignment and populating it with the files from previous assignments to use as a starting point.
- Create a new folder called
week-7
inside yourapp
folder. - Copy
new-item.js
,item-list.js
,item.js
,items.json
, andpage.js
from /week-6 to /week-7.
Create meal-ideas.js
The MealIdeas component is responsible for fetching and displaying meal ideas based on a selected shopping list item. Here are a few things to keep in mind when creating this component.
- Import Necessary Hooks and Begin Component Definition
- Since this component uses
useState
anduseEffect
, start with the"use client"
directive. - Import the
useEffect
anduseState
hooks fromreact
. - This component should receive a single prop: ingredient.
- Define State Variables
- Inside your component, define a state variable using the useState hook:
meals
.meals
will hold the list of meal ideas fetched from the API. Initialize it to an empty array.
- Define API Fetching Function
- Next, outside your component, define a function called
fetchMealIdeas
, which fetch data from the API.fetchMealIdeas
should take an ingredient as a parameter, make a fetch request to the TheMealDB API (opens in a new tab), and return the meals that include that ingredient. - The API endpoint for fetching meal ideas is:
https://www.themealdb.com/api/json/v1/1/filter.php?i=${ingredient}
. For example, if the ingredient is "chicken", the API endpoint would be: https://www.themealdb.com/api/json/v1/1/filter.php?i=chicken (opens in a new tab). The API returns a list of meals that include the specified ingredient. Each meal has three properties:idMeal
: id of the mealstrMeal
: name of the mealstrMealThumb
: URL of an image of the meal
- Define Load Function
- Next, inside your component, define a function called
loadMealIdeas
. This function should callfetchMealIdeas
with theingredient
prop and store the result in themeals
state variable.
- Use the
useEffect
Hook
- Use the
useEffect
hook to callloadMealIdeas
whenever theingredient
prop changes.
- Render the Component
- Finally, define your component's render method. This should return a
div
that includes a header and a list of meals. Each meal should be a list item that displays the meal's name.
Modify item.js
The changes in the item.js
file from Week 6 to Week 7 are relatively straightforward with just one main change. The goal is to make each item in the list clickable so that when a user clicks on an item, it can trigger an action in the parent component.
-
Update Component Definition: Update your Item functional component's definition to accept an additional prop called
onSelect
. -
Add onClick Handler to List Item: In the returned JSX, add the
onClick
prop. This makes the entire list item clickable and the providedonSelect
function will be triggered when a user clicks on the item.
Modify item-list.js
-
Import Necessary Hooks: Start by importing the necessary React hooks at the top of your file. You'll be using the
useState
hook. -
Update Component Definition: Begin defining your
ItemList
functional component. In addition to theitems
prop from Week 6, this component should now also receive anonItemSelect
prop. -
Update Render Method: The rendering of sorting buttons remains the same. The change is in rendering the list items. Now, each Item component should be clickable and trigger the
onItemSelect
function with the respective item as an argument. This is achieved by passing theonSelect
prop to each Item component.
Modify page.js
In the page.js file, the modifications you'll need to make are as follows:
-
Import New Components: You'll need to import the new MealIdeas component at the top of the file. This component is responsible for fetching and displaying meal ideas based on a selected ingredient.
-
Additional State Variable: You'll add a new state variable
selectedItemName
using the useState hook. This state will hold the name of the selected item from the shopping list. -
New Event Handler: You'll create a new event handler function
handleItemSelect
. This function extracts the name of the selected item, cleans it up, and updates the selectedItemName state. This event handler is passed down to theItemList
component and is called when an item in the list is clicked.
The item names in the shopping list are not always clean. For example, the item name "chicken breast" is stored as "chicken breast, 1 kg 🍗". For the MealDB API to work properly, we need to clean up the item name by removing the size and emoji. split
, trim
, and replace
functions can be used to clean up the item name. For example, to remove the emojis you could use something like this (opens in a new tab): text.replace(/([\u2700-\u27BF]|[\uE000-\uF8FF]|�[�-�]|�[�-�]|[\u2011-\u26FF]|�[�-�])/g, '');
-
Layout Changes: The layout of the page has been modified to accommodate the new features. The NewItem and ItemList components are grouped together on one side, and the MealIdeas component is placed on the other side. This is achieved by wrapping them in a div with a flex property.
-
Passing Props: The
ItemList
component now receives theonItemSelect
prop, which it didn't previously. The MealIdeas component receivesselectedItemName
as a prop namedingredient
.
Example Output
The following is an example of what your application might look like. Without the optional challenge, you only have to display the meal ideas list for a shopping list item. You do not have to display the ingredients for each meal.
https://cprg306-assignments.vercel.app/week-7 (opens in a new tab)
Assignment Submission
The instructor will be able to find your assignment in your GitHub repository. Make sure you have committed and pushed your changes to GitHub before the assignment deadline.
Check your GitHub account to see if your assignment has been correctly pushed.