A live version of this project can be found here
The following accounts can be used for user and admin access:
Admin User
User: asdub
Pass: hashb7790
Standard User
User: laryta
Pass: hashb7795
I have created an app to promote the services of a freelance designer. Allowing users to purchase custom design services. Specifically illustrations/ artwork in the form of Icons, Logos, Posters and general Custom Artwork.
Users can select from a variety of design services, select and enter their desired customisations. The user has the option of creating a profile to keep all their orders in one place and view or download their artwork. In addition, users receive email notifications regarding the status of their order.
Site admins can add, edit or delete services. And also complete user orders by uploading completed artwork.
My intention was to create a simple app that permits the user easy and considered access to the design services being presented. Users can easily navigate and purchase design services and check on an orders status. While administrators can easily complete user orders, via an upload feature.
Having previously designed my own layouts and styles, I wanted to use a front-end framework with my own custom styling (and layouts in parts) for this project. The app has been built using the Bootstrap front-end framework.
Owing to the services, the user base is likely to be very broad. With this in mind the design and layout has been considered to appeal to all age groups and demographics.
- Be simple and easy to use (as always!)
- Visually pleasing.
- Provide users with a structured and easy to navigate freelance services.
- Allow users to purchase a service (Stripe).
- Allow users to customise their service.
- Allow users to manage their address data (CRUD).
- Provide users with email & on screen notifications regarding their ordered services.
- Provide defensive design in terms of data editing or deletion.
- Provide seamless login/ registration functionality.
- Allow administrators to manage services.
- Allow administrators to complete user orders.
- The app provides an ecomm platform. An unlimited number of products or services can be added to the database. Managed by the administrator.
- I want the app purpose to be obvious or easy to figure out.
- I want to view recent artworks purchased by other users.
- I want to know about the service provider before purchase.
- I want to view all available services.
- I want detailed information on each service.
- I want the ability to search for services by keyword.
- I want to customise each service.
- I want to update my order quantities.
- I want to remove items from my order.
- I want to see the total cost of each order item.
- I want to see a breakdown of customisation costs.
- I want a simple payment process.
- I want to know of any issues with payment.
- I want to see payment confirmation and a summary of my order.
- I want to know when my order is available.
- I want to easily access my order.
- I want to view my order history.
- I want to view the status of my order.
- I want to view or download my order artwork.
- I want to update my delivery address.
- I would like to add a service.
- I would like to edit a service.
- I would like to delete a service.
- I would like to see pending user orders.
- I would like to complete a user order:
- with the ability to upload user ordered artwork.
- and notify a customer when artwork is available.
The Bootstrap front-end framework was used as the base app. With additional custom css.
The app uses the following Bootstrap content, components & utilities:
A limited palette of colours was used throughout the app.
There are four colours used in total throughout the app.
Wireframes were completed for desktop, tablet & mobile devices.
Google Fonts is in use on this app.
To keep on the simplicity brief,
Only one font is in use:
Roboto in weights of 400 & 600.
All primary icons in this project are from Noun Project.
With secondary icons from Font Awesome.
The following SVG icons have been used throughout the app:
All illustrations/ artwork featured on this project are from unDraw. An amazing open source project by Katerina Limpitsouni.
Example SVG
The app uses the Bootstrap front-end framework. Which is fully responsive. Bootstrap uses a 12 column grid system for layout.
The app renders in three layouts for mobile, tablet and desktop. Additional css media queries have been included to ensure a consistent experience across device types.
Users will initially land at /home, a single page scroller containing a work/ portfolio section display the 8th most recent completed customer artworks. And also an about and social contact section. A large header (in all views) with dropdown navigation allows the user to quickly search, access account facilities and order bag. Dropdown menus allow the users to navigate to individual services, all services, or can sort the services by price, rating and type. Home contains a hero banner with a large CTA. Taking the user to an all services view.
Selecting a service takes the user to a detailed view. And features the customisation options available to the user. The user can add a service to their order from here. A toast notification will prompt the user to 'View their orders' where they can access their order bag. Or from the shopping card icon within the header.
In the order bag view, the user can review their orders and customisations. See their item and total pricing. And also a breakdown of their customisation costs. A user can amend order quantities and remove items. A user can proceed to checkout/ payment from here.
The checkout view displays an order summary, a customer form and payment facilities. Once the user completes their transaction, they are directed to an order success view, which details the order just made. A user will also receive an automated email confirming their order.
A site administrator has access to a management view where they can add new products and check pending orders. An admin can access the complete customer order functionality from here. Where an admin can see an overview of pending customer orders. They can upload the customer artwork, once the admin selects complete - the customer will receive an email confirming their artwork is available to download and has been shipped. The email contains a link to view and download the artwork. If the user has an account setup they can view their historical orders and artwork (if complete).
The apps functionality is explored in further detail in the Features section below.
As this app uses the Django web framework , the included default object-relational mapping layer (ORM) is used. Django uses SQLite by default Heroku Postgres in production.
Data is organised around 5 core apps, the models associated with these apps are detailed below. The diagram above displays the relationships between these models.
App | Model |
---|---|
Home | Images |
Checkout | UserProfile, Product, Order, OrderLineItem |
Bag | Product |
Products | Order, Image, Product, Category |
Profile | Order, Image, UserProfile |
Improved Design is an ecommerce app providing users with the following functionality:
Navigation
- The app has a dynamic menu structure. Providing a standard horizontal text navbar on medium and large screens, and a hamburger menu with sidebar when used via a small/ mobile screen. Both menus have dropdown submenus.
- Under /account an unregistered user has the navigation options of: 'Login' and 'Register'.
- Under /account an authorised user has the following options: 'Profile' and 'Log Out'
- In addition to the above, an admin user also has access to 'Management'
Authentication - AllAuth
- Users have the ability to register for a user profile.
- The registration process features email verification.
- AllAuth also provides Login and logout views for the user.
- A user can reset their password via the 'Forgot Password' hyperlink on the login form.
Home - (/)
- The home app contains a single view detailing a hero banner, work portfolio, about and contact sections.
- The hero banner contains a CTA to the services view.
- Artwork for the work portfolio is generated from the 8 most recent admin completed artworks which have been uploaded.
- A further CTA is located under the work/ portfolio section.
- The contact section contains font awesome social media icon links with transition effects
- An app wide footer detailing copyright and links to terms & Conditions/ privacy policy. Improved design branding is also included.
Services - (services/)
- The services view lists individual services in the form of service cards.
- Service cards contain representational artwork, title, cost and rating.
- If authenticated as an admin, the options to edit and delete the service will also appear on the card.
- A user can also filter & sort the services based on Price, Name & Category. In both ascending or descending order.
Services Detail - (services/<product_id>)
- Upon selecting a service on the Services View, the service detail view is rendered.
- This view contains additional information in the form of a description and responsive image of the representational artwork.
- The user can also specify how they wish to customise their artwork, they can:
- Choose the quantity
- Detail their requirements
- Choose a size
- Provide text content
- Choose a colour profile
- An 'Add to Order' button allows the user to add the order to their order bag.
- Or return to the all services view.
- When an item is added to an order, a notification will appear inviting the user to 'View all Orders'
Orders/ Bag - (orders/)
- This view lists all the current orders the user has added.
- Each order item is listed with the following data:
- Service Image
- Title
- SKU
- Size
- Colour Profile
- User customisations:
- Requirements / Request
- Text Content
- Item Cost
- Item Quantity
- Subtotal
- The user can amend quantities or remove an item in this view.
- An order total, delivery cost and Grand Total is displayed under the list of order items.
- A cost breakdown is also provided where the price has changed due to user customisations.
- The user can proceed to checkout/ payment or return to the all services view via the buttons under the order view.
Checkout - (checkout/)
- The checkout view provide the user with a form requiring their,
- Full Name, Email Address, Phone Number, Address and Credit Card Information.
- If authenticated, an option to store the address information is made available.
- A brief order summary of the users order items is display containing:
- Service Image, Title, Size, Quantity and Item Subtotal.
- The Order Total, Delivery Costs and Grand Total.
- Payment processing is done with stripe - to test, use '4242 4242 4242 4242', an expiry date in the future, any 3-digit CCV code and a five digit postcode.
- A user can complete their order and make payment by selecting the 'Complete Order' button.
- If there are any issues with payment a toast notification will appear.
- Stripe webhooks are used to complete the order and create the database entry should the user close the page or technical issues.
Checkout Success - (checkout_success/) If the payment is successful the user will be directed to the checkout success view.
- This view confirms the successful order and order number.
- Advises user of email confirmation sent.
- Saves the user's delivery information, if selected.
- Attaches the order to the UserProfile, if created/ exists.
- A toast notification will also appear confirming the order, and presenting the user with the order number.
- A CTA is located at the bottom of this view, directing the user back to the all services view.
- If accessed from the 'View Order' button within profie views (detailed below), this CTA return the user to their profile.
Account/ Profile - (account/) Once authenticated a user will have a 'Profile' option within the account dropdown menu.
- The profile view displays the user's saved address.
- A user can also update this information from this view.
- The profile view displays the dusers orders and status.
- Completed orders are listed first, the user has the option to view their artwork.
- The user can also view their original order confirmation.
Artwork - (artwork/)
- Accessed from the profile view above and also via URL sent to the user when an administrator completes their order.
- Displays any artwork images associated with the order.
- And two options to download and view the artwork.
Management - (add/) If a user is authenticated as an administrator a management option is available in the account dropdown.
- In this view an admin can add additional services/ products.
- View pending orders.
- Access the customer order completion view.
Customer Order - (customer_order/<order_id>) An administrator can review a customer's order and upload artwork to the order. Upon selecting complete, the user will receive an email containing a link to view and download their artwork. And the administrator will be redirected back to the add service/ pending orders view.
In a future version I would like to add more interactive customisation fields. Giving a user the ability to select colours. Draw shapes. And provide a mood board they could add images, colour swatches etc to.
I would like to improve security on the app. And setup sign-in with social media/ gmail functionality to streamline the registration process.
I would like to expand on the current offering to include: - Daily, Weekly, Monthly sales data. - Apply the same breakdown to the number of services sold. - Display metrics regarding registered users, active users.
In conjunction with the streamlined registration process, I would like to collect data and present metrics to the admin detailing order abandonment.
Django is an open source framework for web development in Python. It is a flexible web development tool that can be used to create almost any type of website or app that is needed. Thanks to it's high usage, and extensive packages - there is often no need 'to reinvent the wheel' when deploying apps. And trusted and proven code can be used to deploy your project.
Django's most powerful feature is its object-relational mapper (ORM), allowing the interaction with a database in a pythonic way - with no need need for SQL queries.
Check out the database schema above to see how this apps database is structured.
The following Django dependencies apply to this app:
- asgiref
- boto3
- botocor
- dj-database-url
- Django
- django-allauth
- django-countries
- django-crispy-forms
- django-storages
- gunicorn
- jmespath
- oauthlib
- Pillow
- psycopg2-binary
- PyJWT
- python3-openid
- requests-oauthlib
- s3transfer
- sqlparse
- stripe
- Login to your GitHub account(or join).
- Go to the repo by clicking here.
- Click fork in the top right corner of the screen.
- Login to your GitHub account(or join).
- Go to the repo by clicking here.
- On the main page of the repository click on 'Code'.
- Click on the 'Clipboard'/ copy the clone URL (Clone with HTTPs).
- In your local environment open your terminal, navigate to or create a directory.
- Paste the URL into your terminal and enter. The repo should be successfully cloned.
Deploying the app on Heroku is very straightforward.
Initial Heroku Setup
Go to Heroku and login or create an account.
- Click on 'new' in the top right corner and then create app.
- Choose a name and select your region
- Your app overview will load.
- Select Resources > Add-ons, and choose Heroku Postgres - the 'Hobby Dev or Free' option.
- Copy the the Postgres URI into a custom variable in settings called 'DATABASE_URL'
Database Migration
Back in the apps settings.py, the database settings need to be updated.
Locate the default 'DATABASE' settings, and replace with:
if 'DATABASE_URL' in os.environ:
DATABASES = {
'default': dj_database_url.parse(os.environ.get('DATABASE_URL'))
}
else:
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.sqlite3',
'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
}
}
This piece of code checks the env for 'DATABASE_URL'. If false the default sqlite db is loaded. If true, the heroku postgres db is loaded via the dj_database_url dependency.
Next you will need to run migrations.
-
You can achieve this by running the following commands:
python3 manage.py showmigrations
Check the output of the above contains migrations for each of the apps, once happy run:
python3 manage.py migrate
-
Now that we have migrated the database, we need to load the data. The initial data for this app came from a JSON fixture. Run the following commands to load the date:
python3 manage.py loaddata categories
Followed by:
python3 manage.py loaddata products
*Data must be loaded in order based on the models relationship.
-
Install gunicorn webserver with:
Pip3 install gunicorn
-
Generate an up to date requirements.txt
Make sure you are in the apps main folder and enter the following command:
pip freeze --local > requirements.txt
-
Generate a Procfile, this tells Heroku to start your app.
You can create one by typing the following into your terminal:
echo "web: gunicorn improved_design.wsgi:application" > Procfile
-
In the main apps settings.py, add the Heroku apps URL to 'Allowed Hosts', also include 'localhost'.
Heroku Setup Continued
- Login into Heroku CLI with:
Enter your login credentials when prompted.
heroku login -i
- Temporarily disable collectstatic with:
heroku config:set DISABLE_COLLECTSTATIC=1 --app {app name}
- Initialise Heroku git remote with:
heroku git:remote -a {app name}
- Now we are ready for the initial commit to Heroku:
git push heroku master
- Back on the Heroku dashboard, we will setup automatic deploys. Navigate to Deploy > Deployment Method > Select GitHub. Search for your repo and then click connect. Then click enable automatic deploys.
Initial Setup
- Go to https://aws.amazon.com/ > Create or Login to your AWS account.
- On the management console select or search and select 'S3'
- Select 'Create Bucket' and populate the name field.
- Deselect 'block all public access' and select the related warning.
- Click 'Create Bucket'
Bucket Setup
-
Under Properties
- Enable 'Static Website Hosting' in order to get an endpoint.
- Set the defaults to index.html and error.html - these won't be used.
-
Under Permissions:
- Add the following CORS configuration:
[ { "AllowedHeaders": [ "Authorization" ], "AllowedMethods": [ "GET" ], "AllowedOrigins": [ "*" ], "ExposeHeaders": [] } ]
- Under Bucket Policy:
- Select Policy Generator:
- Select: 'S3 Bucket' Policy from the type dropdown.
- But ' * ' in the principal field.
- Select ‘get_object’ from the actions dropdown.
- Copy your ARN from Bucket Policy (on the previous page).
- Click add statement
- Copy the policy config into the Bucket Policy editor and save.
- Under Access Control List
- Select ‘list’ under Everyone/ Public access.
- Accept the public warning and save.
- Select Policy Generator:
-
Creating a User:
- From the service menu select 'IAM'.
- First, create a group for the user by selecting ‘User groups’ > ‘Create User group’.
- Enter group name (anything you like).
- Click ‘Create Group’.
- Next navigate to ‘Policies’.
- Click ‘Create policy’.
- Click the JSON tab then ‘Import managed policy’.
- Select AmazonS3FullAccess and import this policy.
- In the JSON tab, add your bucket ARN from above as a value for the ‘Resource’ key.
- Click Next - ignoring tags - And then ‘Review Policy’.
- Enter a name and description for your new policy.
- Then click ‘Create policy’.
- Navigate back to ‘User groups’.
- Click on the group created earlier > click permissions > Add permissions > Attach policy.
- Select your policy from the list and then ‘Add Permissions’.
- Now you can create a user.
- Navigate to ‘Users’ > ‘Add user’.
- Enter your desired user name.
- Select - ‘Access key - Programmatic access’
- Select ‘Next Permissions’
- Select the group made above.
- Select ‘Next Tags’, again ignoring the tags page. Click ‘Next Review’
- And finally! click ‘Create User’
- Download the user keys via the CSV.
You can find out more about connecting Django to AWS S3 here
W3C HTML Validator (Results)
The W3C validator found 3 initial issues:
- Error: 'http-equiv' attribute whose value is 'X-UA-Compatible' must have a content attribute with the value 'IE=edge' [x]
- Error: Element div not allowed as child of element button in this context. [x]
- Error Duplicate ID category-link. [x]
- Error: Start tag a seen but an element of the same type was already open. [x]
JS Hint found 1 initial issues:
- Warning Missing semicolon. (line 111) [x]
Found 50+ initial PEP8 issues, all related to 'line too long'. I corrected the errors highlighted. And omitted altering any files generated by Django, e.g. migrations.
WC3 CSS Validator (Results)
WC3 CSS Validator found 1 error & 38 warnings:
- Error & Warnings missing bracket (line 673) [x]
Performance
Issues reported,
- Image elements do not have explicit width and height
Best Practices
Issues reported,
- Links to cross-origin destinations are unsafe
Accessibility
Issues reported,
- Background and foreground colors do not have a sufficient contrast ratio.
- Alt attrs missing from social media links
SEO
Issues reported,
- Document does not have a meta description
- [Performance] The images in question are SVG's which do have heights and widths set!? [ ]
- [Best Practices] Links to cross-origin destinations are unsafe - added rel="noreferrer" to external links [x]
- [Best Practices] Vulnerabilities due to jQuery version - updated to latest [x]
- [Accessibility] Left colours as they are - I feel the contrast is perfectly sufficient [ ]
- [Accessibility] Alt attrs missing from social media links - Adding alt attributes caused W3C errors [ ]
- [SEO] Added meta tags [x]
Several other errors are present relating to external scripts (Stripe & Jquery)
Results now seem to be at an acceptable level. In furture versions SEO would be the priority to score highter in.
Testing documentation can be found here
I want the app purpose to be obvious or easy to figure out.
A large hero banned indicates the purpose of the app on the home page.
I want to view recent artworks purchased by other users.
Completed user artwokr is displayed in a responsive grid on the home page
I want to know about the service provider before purchase.
An about section is located on the home page, also accessed from the nav menu.
I want to view all available services
A 'View all Services' option is available in the menu - ALl home CTA's also direct the user to this view
I want detailed information on each service
A detailed view is available for each service/ product.
I want the ability to search for services by keyword.
Search functionality is available in the sites header, accessible from anywhere in the site - on any screen size.
I want to customise each service.
User customisation fields are available on each product.
- I want to update my order quantities
- I want to remove items from my order.
- I want to see the total cost of each order item.
- I want to see a breakdown of customisation costs.
These user stories are achieved in the order/ bag view. A user can manage their order from here.
- I want a simple payment process.
- I want to know of any issues with payment.
- I want to see payment confirmation and a summary of my order.
The payment process is very straightforward, and takes place all within the app. A toast notification will appear if there is a payment issue and also when an order is successfully processed.
I want to know when my order is available. & I want to easily access my order. A user will receive an email confirmation when the place their order. Once the order is completed by an admin, they will receive an additional email confirming this containing a link to their completed artwork.
- I want to view my order history
- I want to view the status of my order.
- I want to view or download my order artwork.
- I want to update my delivery address.
These stories are achieved in the Profile and artwork views.
- I would like to add a service.
- I would like to edit a service.
- I would like to delete a service.
- I would like to see pending user orders.
- I would like to complete a user order:
- with the ability to upload user ordered artwork.
- and notify a customer when artwork is available.
These are achived in the Management & Complete Order views. A customer is notified by email when an order is completed, which contains a link to view their artowrk.
** There is an issue with changing order quantities in the bag view with multiple items, only the first item is affected.
** The sort dropdown on the all services/ product view is currecly not working. However, this functionality is working if accessed via the nav menu dropdown.
https://docs.djangoproject.com/en/3.2/
I referred to the Code Institutes course material for the functionality aspects of this app. Full credit: https://github.com/Code-Institute-Solutions/boutique_ado_v1/ This credit applies app wide.