"Family Recipe" is a recipe sharing website made for my Portfolio Project 4 for code institute, where users can upload and share recipes as well as reading recipes from other users.
I thought about this as I have a family recipe book from South Africa and I think it would be great to have a website where family members could share their family recipes with eachother and the world. I also wanted a rating system based on the amount of likes and a comment section for recipes.
Click here to view the live site.
The project uses ElephantSQL as PostgreSQL relational database for storing the data. I used lucidcharts to display the relationships to the different database tables. The main relationships were based on the likes model as these were called both in the recipe model to show the total likes and related back to the unique users, and then also the users liked recipes.
- This is the main model for the website to function. I based this off the walkthrough I think therefore I blog. Adjusted the layout a bit to fit the website function as a worldwide sharing platform for recipes in mind.
- The comment model allows logged in users to comment on recipes.
- The custom model.
- Enable users to like recipes.
To create a website with good UI and UX I interviewed my sister in France who is a chef and asked her what her thoughts were on a recipe sharing website and also got insight to what she would want to use the site for.
There are many future features that will later be introduced and enhancements that I did not get around to implementing just yet but these will be reflected in the user stories section.
I employed the agile methodology, starting from the planning stage and continuing until the final product was built. To ensure that I stayed organized and on track, I utilized a GitHub project and a Kanban board. Information about User Stories can be found in the subsection below⇩
Five planes of User Experience:
My User Stories can be found here. All User Stories include:
- Acceptance Criteria
- Tasks
- Labels (MoSCoW Priotarization)
Some of the User Stories are part of an EPIC.
Please, go to TESTING.md if you want to come to the section where I test my User Stories.
The key goal of the website is to create a platform where users can create an account and share recipes as well as viewing recipes shared by other users.
The target audience is primarily people who have family recipes that they are proud of and want to share with the world but also anyone that likes to cook and who wants new and different recipes to try out from around the world. That is why I added country as an option to the recipe model as well as the like function so that users could see how the recipes were rated.
-
Easy navigation throughout the site.
-
Uncomplicated user experience to login and handle recipe creation and modification.
-
Responsive and informative feedback on the site (comments and likes).
-
Capability to access the site across various electronic devices.
-
Fully working links and functionality.
-
Easily accessible navigation bar with the links that have easily understandable names.
-
Responsive design.
-
A brief and simple description of the site’s purpose.
-
A complete collection of recipes.
-
A thorough list of ingredients and step-by-step instructions to follow.
-
A possibility to see what country the recipe is from.
-
A possibility to read comments.
-
Add registration/login features that give the user access to extra functionality.
-
Add Logout functionality for security reasons.
-
CRUD functionality:
- Implement feature that allows user to Create recipes.
- Implement feature that allows user to Read recipes.
- Implement feature that allows user to Update recipes.
- Implement feature that allows user to Delete recipes.
-
Comment and like feature:
- Enable logged-in users to post comments on any of the published recipes.
- Enable logged-in users to like/unlike published recipes.
The blog is divided into different pages. Some of the pages are accessible only for logged in users. The blog structure allows users to access recipes via the recipes page or by sorting them through different categories. All users can access detailed information about each recipe by clicking on the recipe card. Users who are logged in can publish, edit, and delete their own recipes, and interact with other recipes by liking and commenting on them.
For detailed information about all existing features see the section Existing Features.
I created wireframes using Balsamiq. Please note this was only a concept in the beginning so the end result was quite different.
The choice of colors depends on the background image chosen for home, sign up, login, and logout pages.
-
313131
is the primary color used throughout the pages:- Welcome message.
- Category boxes on the home page.
- Add | Edit recipe forms.
- Sign Up | Login | Logout forms.
- Confirmation message when choosing to delete a recipe.
-
850000
is used for links and hover styling on navbar and footer. The color was chosen with the help of Color Contrast Analyzer (provided by Lighthouse testing) in order for background and foreground colors to have a sufficient contrast ratio. -
FFFFFF
is used as the background color for all recipe-related pages including recipes, recipe details, add recipe, edit recipe and categories.
All buttons throughout the pages have the same styling, ensuring uniformity and providing a seamless user experience.
I went with following colors:
-
GREEN
that usually is associated with "YES" | "SAVE" | "OK". -
#7e180d
that usually is associated with "NO" | "CANCEL" | "GO BACK". -
WHITE
for the text in order to get a sufficient contrast ratio.
Users have the option to upload images in various sizes. To ensure consistency, I made sure that all recipe cards on the same line had the same height. The size of the images doesn't affect the layout of the recipe detail page. All sections are uploaded correctly as they should be.
- The colors used were based on the the colors in the logo and the hero image tomato color that stood out to me. I chose the primary color to invoke a feeling of warmth of sharing recipes with community and it matched the heart color as part of the logo.
- #7E180D color1
- #FFFFFF color2
- #313131 color3
Google Fonts was used to import the chosen fonts in use for the site.
- HTML
- CSS
- Python
- JavaScript
-
- Python HTTP server for WSGI applications.
-
- Database platform used by the deployed project on Heroku.
-
- The cloud platform used to store static media files.
-
- used for version controll.
-
- The IDE used to create the site.
-
- The code hosting platform used to save and store the files for the website.
-
- The cloud platform used to deploy the project into live environment.
-
- The front-end development framework used for styling along with custom CSS.
-
- The diagramming application used to create ERD diagrams.
-
- used to see how the site looks on a range of devices.
-
The navbar is bootstrap and is at the top right of every page.
-
If the user is logged in or is a guest the menu content changes appropriately.
-
The hamburger menu is applied on media breakpoint of 979px. This was when the text from the navbar started going into the family recipe logo and looked cluttered.
-
I kept the menu clean with no additional styling to focus attention on the recipes and content instead.
-
Guests:
- The navigation bar contains links for the Logo, Home, Register and Login pages.
- Clicking "Register" directs users to the create account form.
- Clicking "Login" directs users to the login form.
-
Logged-in users:
- The navigation bar contains links for the Logo, Home, Recipes, Add Recipe, and Logout pages.
- Clicking "Add Recipe" takes the user to a page where they can fill in a form to publish a recipe.
- Clicking "Logout" directs users to the confirmation page.
-
-
On smaller devices, the navigation bar is displayed using a hamburger menu:
Mobile logo (shown above)
Desktop logo (shown above)
- This section welcomes users to the blog, once the user is logged in this message changes, this was achieved using the django check to see if the user is authenticated.
The short welcome explains:
- The concept of the site as a platform to share recipes.
- That the user will need to create an account to do this.
The message then changes as seen below. I put a link after the welcome to the user to allow them to view their recipes. From this page they are also able to comment, like, edit and delete their recipes.
- When the user clicks on "view your recipes" they will be taken to the recipes that they have submited. On this page they will be able to also edit and delete their recipes.
- The footer is quite unobtrusive to not take away from the main content, the social media icons open up in a new page when clicked.
- The footer remains consistent across all pages.
- The form enables users to register and create an account.
- The form includes following fields:
- Username
- Password
- Password (again)
-
The form enables users to log in.
-
When a user logs in, they gain the ability to:
- like recipes.
- comment on existing recipes.
- create new recipes.
- edit/delete their own recipes.
- When the user clicks on Logout in the navbar, they are redirected to a page displaying a confirmation message above.
-
This page shows a list of all the published recipes.
-
Information displayed:
- recipe image;
- created date;
- create date;
- "Read more" button.
-
By clicking the "View Recipe" button, the user redirects to a page containing detailed information about that specific recipe.
The recipe detail page includes the following information:
-
recipe title;
-
country;
-
number of servings;
-
create date
-
likes
-
Ingredients section.
-
Instructions section.
-
Comments section:
-
There are some comments on the recipe:
![image]()
- This page includes a form that allows users who are logged in to publish their own recipes.
- Available fields:
-
Title
-
Country
-
Image (uploaded to cloudinary)
-
Description
-
Ingredients
-
Instructions
-
Servings
-
Buttons
-
SAVE:
-
After correctly submitting the form, the user will be redirected to the recipes page upon clicking the save button.
-
A success message is displayed to the user.
-
-
"CANCEL" button:
- When the user clicks on the 'go back' button, they will be redirected to the recipes page.
-
-
- Categories to be added such as vegan, desert, main course etc
- Enable users to log in using their social media accounts.
Detailed testing of the site can be found at TESTING.md.
Testing includes following:
- Validator testing
- Responsivness & Browser Compability Testing
- Manual Testing
- Automated Testing
- Testing of User Stories
- Lighthouse
-
Eror: I could not runserver or make migrations at some point. The error below was shown. Migration folder 0004 was missing.
-
Fix: Moved all the migration files to a subfolder and then tried again.
-
Problem: I could not get css to load.
-
Fix: I had set debug mode to true which seemed to affect loading of files.
-
Problem: Could not get am i responsive to work
-
Fix: post found on slack to ignore x-frame in settings. I just had to install an extension to do this in chrome
None
To style the forms I watched the following tutorials
-
Favicon Generator was used to generate a favicon from the image.
-
Font Awesome was used to add the icons for the social media links in the footer.
-
The hero image was taken from Pexels
-
The recipes published by me are taken from Jamie Oliver.
-
The main code of this project is based on the Code Institute tutorial "I Think Therefore I Blog" with changes made to suit my project. Django Documentation was used throughout the project.
-
The readme skeleton was taken from Kattis91
-
Secret Key Generator was used to generate Django Secret Key.
I would like to aknowledge the following people:
-
Spencer my mentor who helped me throughout this project.
-
My sister Claire in France who helped with the concept via her feedback of what she would like in a recipe sharing site.
-
Tutor Assistance Team for helping me out with diverse things throughout the project.
-
Install Gunicorn(the server that is used to run Django on Heroku):
pip3 install django gunicorn
-
Install dj_database_url and pyscopg2(connect to PostegreSQL):
pip 3 install dj_database_url pyscopg2
-
Install Cloudinary (The cloud platform used to store static media files):
pip3 install dj3-cloudinary-storage
-
Create Project.
-
Create App.
-
Add App to installed apps in settings.py:
INSTALLED_APPS = [ ... 'APP_NAME', ]
-
Navigate to ElephantSQL.com and click “Get a managed database today”.
-
Click Create New Instance.
-
Set up your plan:
- give your plan a Name;
- select the Tiny Turtle (Free) plan.
-
Click “Select Region” and select a data center near you.
-
Click "Review", check that your details are correct and click “Create instance”.
-
Return to the dashboard and click on the database instance name for this project.
-
Copy the database URL.
-
Sign up for Heroku and accept terms of service.
-
Click the "Create a new app" button.
-
Give your app a name and select the region closest to you. A name must be unique.
-
Create env.py file and check that the file is included in the .gitignore file.
-
Import os library:
import os
. -
Set environment variables:
- DATABASE_URL with the value you just copied from ElephantSQL:
os.environ["DATABASE_URL"]="<copiedURL>
- SECRET_KEY:
os.environ["SECRET_KEY"] = "randomSecretKey"
- DATABASE_URL with the value you just copied from ElephantSQL:
-
Add the following code:
import os import dj_database_url if os.path.isfile('env.py'): import env
-
Remove the insecure secret key provided by Django. Change your SECRET_KEY variable to the following:
SECRET_KEY = os.environ.get('SECRET_KEY')
-
Comment out the original DATABASES variable and add the code below:
DATABASES = { 'default': dj_database_url.parse(os.environ.get("DATABASE_URL")) }
-
Save all files and make migrations:
python3 manage.py migrate
-
Go back to the Heroku dashboard and open the Settings tab:
-
Create Config Vars:
- KEY: PORT | VALUE: 8000.
- KEY: SECRET_KEY | VALUE: randomSecretKey(the value that is in env.py)
- KEY: DATABASE_URL | VALUE: ElephantSQL database url(no quotation marks needed)
- KEY: DISABLE_COLLECTSTATIC | VALUE: 1 (Temporary to be able to deploy the project as we do not have any static files yet)
-
Create a Cloudinary account (steps can be found in the Code Institutes tutorial in LMS).
-
Copy API Environment Variable in the Cloudinary dashboard.
-
Go back to env.py and add a new environment vriable:
- CLOUDINARY_URL with the value just copied from the dashboard ⇧(remove CLOUDINARY_URL in the beginning).
-
HEROKU: Add a new Config Var with the KEY CLOUDINARY_URL, and the same value(URL) as in the step above.
-
settings.py:
-
Add Cloudinary Libraries to installed apps (the order is important):
INSTALLED_APPS = [ ..., 'cloudinary_storage', 'django.contrib.staticfiles', 'cloudinary', ..., ]
-
Tell Django to use Cloudinary to store media and static files:
STATICFILES_STORAGE = 'cloudinary_storage.storage.StaticHashedCloudinaryStorage' STATICFILES_DIRS = [os.path.join(BASE_DIR, 'static'), ] STATIC_ROOT = os.path.join(BASE_DIR, 'staticfiles') MEDIA_URL = '/media/' DEFAULT_FILE_STORAGE = 'cloudinary_storage.storage.MediaCloudinaryStorage'
-
-
Link file to the templates directory in Heroku. Place under the BASE_DIR line:
TEMPLATES_DIR = os.path.join(BASE_DIR, 'templates')
-
Change the templates directory to TEMPLATES_DIR:
TEMPLATES = [ { ..., 'DIRS': [TEMPLATES_DIR], ..., ], }, }, ]
ALLOWED_HOSTS = ['app-name.herokuapp.com', 'localhost']
web: gunicorn family_recipe.wsgi
-
Click on the "Deploy" section on the top of the page.
-
Select GitHub as deployment method and click the "Connect to GitHub" button.
-
Search for the repository for this project, adrianproject4.
-
Click "Connect" to link up Heroku app to the GitHub repository.
-
Click on "Deploy Branch".
-
Click the "Enable Automatic Deploys" button to make it possible for Heroku to rebuild the app a new change is pushed to GitHub repository.
-
Log in to GitHub and ind the repository for this project, adrianskelton/adrianproject4.
-
In the top-right corner of the page, click Fork.
-
Type some new name into the "Repository name" field to distinguish your fork from the upstream repository.
-
Click Create Fork.
-
The fork is now in your personal account and can be changed in the way you want.
-
On GitHub, navigate to your fork of the adrianproject4 repository.
-
Above the list of files, click <>Code.
-
Copy the URL for the repository. Repository can be cloned in three different ways:
- HTTPS;
- SSH;
- GitHub CLI.
-
Open Terminal and change the current working directory to the location where you want the cloned directory.
-
Type
git clone
, and paste the URL you copied earlier. Press Enter