As we turn towards the internet to search for the perfect recipe, we end up flooding our browser with large numbers of tabs, and filling up our bookmarks with many unsorted saved pages of the recipes that we like. This, while completely functional, is a very crude way of organising our recipes and oftentimes leads to a lot of confusion.
Furthermore, when we finally decide on the recipes we want to prepare and head out to purchase the required ingredients, we would have to fill in a grocery list ingredient by ingredient. Don’t we wish we had something that can automatically import a recipe into a grocery list?
Our team aims to create an app that targets these problems as mentioned, by streamlining the searching and saving process into a single app, allowing users to search, sort and save recipes found on the web. Grocery lists are also made much easier with features that allow users to add all ingredients into a compiled Grocery list with one tap.
Aspiring home cooks who want to cook at home.
The mobile app will be the platform on which the user interacts to search for and save recipes, as well as the users grocery list which are saved locally on the users device.
The backend will store the profiles of the user (including usernames and passwords), as well as perform web-scraping and sorting functions.
- User Login
- Allow users to login and store their preferences
- Display personalized recommendations on the homepage
- Grocery list: Phase 1
- User-inputted grocery list
- Can be combined into an overall list or split according to dishes/cuisine
- Conversion between non-standard units (e.g. cups/tablespoon), volume (ml), and mass (g)
- Web-scraped recipes: Phase 1
- Pull recipes from a few popular recipe websites
- Display the picture and title in a card format
- Ingredients and preparation methods are shown when user clicks on a recipe
- Web-scraped recipes: Phase 2
- Enable users to search all aggregated sites or individual sites for recipes
- Users can save recipes to their device
- Grocery list: Phase 2
- Users can add the ingredients from online recipes (web-scraped) to their grocery list
- Each ingredient in the online recipe is identified and classified. Users can combine ingredients that are similar into one entry, or leave them as separate items but under the same category (e.g. skim milk vs full cream milk)
As of now, while there exist many recipe websites that include their own search and sorting system, these sites usually feature their own or user submitted recipes. Our app is looking to combine the available information from multiple sites into one search platform allowing users to be able to access multiple sources in a single instance.
Users will first register with their email and password, after which they can login with the same email and password to retrieve user data on a separate device. Guest login without registration and without data syncing is also available.
Firebase Authentication is used to authenticate users, with corresponding user data stored in the Cloud Firestore database. The Firestore security rules are configured such that the currently authenticated user is only able to modify their own data, and not other users’ data.
Upon login or registration, users are first brought to the home screen in the app, where they are shown popular recipes as well as recommendations based on time of day (e.g. breakfast recipes during morning, dinner recipes during evening).
Recommendations are scraped from the recommended section in the recipe sites. For example, breakfast recipes are scraped from the breakfast section of the websites, and so on. Current implementation only scrapes from AllRecipes.com, however other recipe sites will be included in the final product.
The search page allows users to search for recipes based on both recipe name and ingredients. For instance, searching “chicken” will bring up both recipes with “chicken” in the name as well as recipes which include “chicken” as one of the ingredients. The matching recipes are displayed in a card format with the image, name, source website, as well as ratings info.
Search terms are run through the web scraper and recipe data is returned to construct the search cards in the app. Additional information not shown on the cards, such as list of ingredients and preparation methods, are asynchronously loaded in the background.
Users can tap on any of the recipe cards, either on the home screen or on the search screen, to view more detailed information of the recipe, including preparation time, serving portions and preparation methods.
This information is presented in a visually consistent format regardless of the source website formatting. Unnecessary details such as long preambles and backstories are omitted.
Information fetched from the individual recipe webpages are processed and sorted into categories which allows for a consistent and clean UI. Details can be found in the “Web scraper” section below.
Found in the detailed recipe view, users can tap on the “Add To Grocery List” button to add all ingredients in the recipe into the grocery list. Here, the ingredients are split into name/amount/unit and added into the list.
Ingredients from recipes are added here. Users have the option to add their own ingredients into the recipe list; the name, amount and units are manually specified. They can tap on the individual ingredients to check them off, to signify that the ingredient has been purchased. In the final product, ingredients can be added to individual recipes and custom recipes can be created, however this is not supported currently.
On the top right, users can press the toggle button to switch between the recipe categorised list, and the combined list which combines all items with the same name, allowing for convenient shopping.
To delete items from the grocery list, users can either delete items individually, by swiping left on the ingredient and tapping the delete icon, or delete the entire recipe at once by tapping the recipe header (To be changed in the future to long press to trigger). All amounts are automatically updated in the combined list.
The grocery list is currently stored locally in a .js file - this will be migrated to the cloud in the future and stored under user data in Firestore. As such, the user will have access to the grocery list across devices.
The combined list is dynamically generated from the list of recipes - a change in the recipe list triggers an update in the combined list. Upon deletion of a single ingredient from the recipe list, the corresponding amount will be subtracted from the combined list. Similarly, when a recipe is added to the grocery list, the ingredient will be inserted into the combined list (if not already present) or the amounts incremented (if present in other recipes)
Users can tap on the bowl icon on the recipe section headers to bring up a pop-up. Here, users can scale the recipe portions, which scales the ingredients proportionally. For instance, selecting 2x scales the ingredients by a factor of 2, and a subsequent selection of 1x will bring the ingredient quantities back down to the original amount.
One of the core components of our app, the web scraper is based on Node.js. The library “Got” is utilized to load webpages, after which the library “Cheerio” parses the webpages and provides the relevant text. The text is then run through a site-specific cleanup algorithm, which formats the raw text and places them into the appropriate categories (e.g. ingredients, preparation method, serving size etc).
As the app features the ability to add ingredients from a recipe into the grocery list, the text cleanup algorithm also extracts the name, quantity and units of the ingredient, from a raw string. For example, “3 tablespoons of olive oil” will be processed and written into an object formatted as such:
recipeObject = {name: “Olive oil”, amount: 3, unit: “tablespoon”};
The web scraper script is stored remotely on Firebase, which acts as a Node.js server. When a user enters a search term, it is passed to the web scraper via Firebase. On script completion, Firebase returns the processed results back to the app to be displayed on the search screen.
(Note: The actual web scraper script, including text cleanup and processing, is entirely coded by us. Firebase only hosts the script remotely and does not scrape or process any data.)
To facilitate faster loading times, the web scraper returns only the basic essential info to build the recipe search cards (Name, URL, Image, Ratings), before returning all the other additional info such as preparation methods, serving portions and ingredients.
This bifurcation of “main” data fetching (to be shown on the cards) and “ancillary” data fetching (not shown on cards) allows for improved performance as the list of search results can be rendered without needing the “ancillary” data to be loaded (which can take up to 10-20s).
In order to build the combined list efficiently when many ingredients are added, we used a hash function to allow for amortized O(1) lookup times. When a new item is added into the grocery list, the ingredient names are processed to ensure consistent capitalisation and formatting.
The names are then fed into our hashing function, which takes the unicode index of each character in the name and multiplies it by a large prime number to generate hash keys with a minimal chance of duplicates. The ingredient data, which consists of the amount and unit, is then inserted into the hash table according to its hash key.