All of the greatest punk/hardcore music - all under one roof. Whether you are looking for collectable, vintage vinyls, or the newest releases from your favourite artists, you are in the right place.
Table of contents generated with markdown-toc
The target audience for this site is primarily fans of punk/hardcore music, particularly:
- People who collect vinyl albums; including rare editions.
- Bargain hunters looking for a good deal on albums/merchandise.
- People looking to keep up to date with the latest releases in the music scene.
- Younger people/students looking for discounts on their favourite music.
- People looking for both current, as well as vintage/collectable, band merchandise.
No. | Bug | Link | Completed |
---|---|---|---|
Site users | |||
#1 | As a user, I want the site's navigation to be simple and intuitive. | Link to issue #1 | ✅ |
#2 | As a user, I want to be able to be able to checkout without an account. | Link to issue #2 | ✅ |
#3 | As a user, I want to be able to be able to checkout without an account. | Link to issue #3 | ✅ |
#4 | As a user, I want to be able to sort/filter products based on the following criteria: Price, Alphabetically, Product type, Artist. | Link to issue #4 | ✅ |
#5 | As a user, I would like to be able to create an account to save/update my delivery information. | Link to issue #5 | ✅ |
#6 | As a user, I would like to be directed to an error page that is inkeeping with the sites design/aesthetic if there is any error. | Link to issue #6 | ✅ |
#7 | As a user, I would like to be able to add products to a 'wishlist' to view them later. | Link to issue #7 | ✅ |
#8 | As a user, I want clear feedback on what is in my shopping bag, including when it is empty. | Link to issue #8 | ✅ |
#9 | As a user, I want to be able to easily update my shopping bag, including removing items from it. | Link to issue #9 | ✅ |
#10 | As a user, I want clear confirmation when an order has been completed. | Link to issue #10 | ✅ |
#11 | As a user, I want feedback that my order is being processed. | Link to issue #11 | ✅ |
#12 | As a user, I would like a page that shows a list of all the artists, so I am able to find relevant products easily. | Link to issue #12 | ✅ |
#13 | As a user, I want to be able to reset my passowrd if I forget it. | Link to issue #13 | ✅ |
#14 | As a user, I want the ability to easily log in and out of my account. | Link to issue #14 | ✅ |
#15 | As a user, I want to be able to contact the store owners to provide feedback or query an order. | Link to issue #15 | ✅ |
Site owners | |||
#1 | As a user, I want the ability to add products to the store. | Link to issue #1 | ✅ |
#2 | As a user, I want the ability to edit products easily. | Link to issue #2 | ✅ |
#3 | As a user, I want the ability to delete products from the store. | Link to issue #3 | ✅ |
#4 | As a user, I want the ability to add new artists to the store. | Link to issue #4 | ✅ |
#5 | As a user, I want the ability to edit information about artists. | Link to issue #5 | ✅ |
#6 | As a user, I want the ability to delete artists from the store. | Link to issue #6 | ✅ |
#7 | As a user, I want to see orders that have been placed. | Link to issue #7 | ✅ |
#8 | As a user, I'd like to be able to see all messages sent in one place. | Link to issue #8 | ✅ |
For this project, I used the pre-made Red at Night colour palette from Daniel Cranney's 'ColorHub'. I decided that to use this colour palette as it is largely monochromatic (using black, grey, and white), but also has a striking red; I believe this colour will attract users' attention toward any calls to action on the site.
The colours used were:
#EF4444
: This colour was used as the brand colour, used mostly on call to action buttons/links, or anything else that I wanted to focus the user's attention to.#18181B
: This colour was used as the primary colour for text (on a white background), and to give a contrasting background to some sections. Also, an opaque version of this colour is used on image overlays.#D4D4D8
: This colour was used sparingly either as text decoration or borders - I chose this colour as I felt it is fairly subtle/understated.#F8FAFC
: This colour is used as the primary background colour, and text colour on dark backgrounds.
I used two Google fonts for this project: DM Sans for the headings, and Montserrat for main body text.
Page | Desktop | Tablet | Mobile |
---|---|---|---|
Landing | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Artists | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Product list | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Product single | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Add/Edit product | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Basket | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Checkout | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Checkout success | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Profile | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Blog list | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Blog single | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Edit blog | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Login/Register | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Contact | Desktop wireframe | Tablet wireframe | Mobile wireframe |
Error pages (400, 500, etc) | Desktop wireframe | Tablet wireframe | Mobile wireframe |
NOTE: To meet the demands of the site, more models were required since the creation of the physical model above - all models can be found below:
The individual database models can be found in this separate markdown file.
Allows the user to navigate between the following sections:
- Home
- Shop
- Blog
- Contact
- Login/Register
- Basket
The "Shop", "Login/Register" and "Basket" also have submenus that display on hover:
The basket dropdown also contains:
- A scroll icon which displays if the dropdown is scrollable.
feature-nav-basket-scroll.mp4
- A small quantity icon to show how many items are current in the user's basket.
This page displays when the user visits the home directory of the site - meaning that it will be the page that most users see for the first time when visiting. It consists of:
- The navigation bar
- A title
- A lead paragraph
- Two buttons - that lead to the shop and blog
Social icons The social icons, upon the page loading, have a staggered animation in which each item slides up into view - one after another. I achieved this using a SCSS for loop.
@for $i from 1 to 5 {
.homepage-icons a:nth-child(#{$i}) {
animation-delay: calc(#{$i * 0.25}s);
}
}
It looks like this in the final version of the site:
social-icons-animation.mp4
When the user visits the blog area of the site, they are initially directed to the blog listing page - which shows the blogs sorted in order from newest to oldest. Also, from the admin, anyone with the correct permissions can set a 'featured' blog, which will be the first blog to show on this page:
Add blog When a staff member navigates to the main blog page, they are able to create a new blog by simply clicking the pencil button - which will take them to the following form:
NOTE: this form also includes the ckeditor rich text field - this gives staff more flexibility to format their content.
Once a user clicks the blog of their choice, they are redirected to the page for that individual blog post. This page includes a breadcrumb which allows you to return to the main blog listing page.
Staff members are also able to set the featured blog via the frontend, by simply navigating to the blog they wish to feature, and clicking the star icon in the header section.
I also installed an additional pip package - ckeditor - to allow any admin who is writing a blog to do so using a rich text field. I decided to take this approach as I found this way ensured that any formatting present in the blog post was present on the page; it also allowed for authors to include relevant links, images, tables, etc.
Edit Blog
When on an individual blog page, staff members are also able to edit blogs by clicking the pencil in the header section.
When the user visits the shop area of the site, they are directed to a list of all products - although it is also possible to navigate to album/merchandise-only pages through the navigation.
Sale tag
If an item is on sale (meaning on_sale=True
in the backend), a red flag displays on the item to reflect this. Also, the new price is shown next to the original price to show the user the deal they are getting.
Digital download tag
If an album is also available for digital download, this is reflected in the products list by the 'digital download available' tag at the bottom of the item - next to the product type.
Filter
On the main shop page, there are options to sort the products based upon certain criteria (e.g. price, name, etc).
Example
shop-filter-video.mp4
Search
The user is also given the ability to search for items within the shop by clicking the floating search icon in the bottom-right corner of the page. However, on mobile - to free up screen real estate - the search bar is included within the main mobile menu.
Examples
shop-search-expand.mp4
shop-search-example.mp4
Once a user has found the product they wish to view on the main shop page, they can click through into the individual pages, which have the following features:
Product information
Merch version
Album version
This section shows the following information about the product:
- Name
- Image
- Artist
- Description
- Sizes (if product type is merch)
- Digital download (if product type is album)
- Quantity selector
This section also includes the 'Add to wishlist button'
Example of quantity buttons - including minus button being disabled when quantity reaches 1
shop-qty-btns.mp4
Track listing
On the album pages, the user is also shown the track listing of the album they are currently viewing - ordered by track number. However, if there is no tracklist for this album, the following displays:
Related items
If there are any other products available from the same artist the user is viewing, they will be shown as related products at the bottom of the page. However, this list is limited to four items.
Example
shop-related-items.mp4
Edit and delete products
From these pages, staff members also have the option to edit and delete the product they are currently viewing:
Edit album
Edit merch
Similarly to the blog pages, the user is also warned before deleting any product
From the "Shop" menu in the navigation, the user can also navigate to the Artists page. This page shows a list of all artists on the site, arranged alphabetically. Each artist name is a link that leads the user to a filtered version of the shop page that shows products by the artists they have selected only.
Add/Edit/Delete Artists From this page, the user also has the option to add, edit and or delete artists.
Edit artist form - same as the add artist form, but pre-populated:
Delete artist - before the artist is deleted, the user is directed to a warning page to ensure they really want to delete the artist in question
I have implemented a simple contact page for customers to contact the store owners. Once this form is filled out, the messages are sent to the inbox (see below).
These pages are modified versions of the templates provided by AllAuth, which seemlessly handle all authorisation.
Once a user has logged in, they are able to see their profile page - this page shows them:
- Their name
- Their delivery information (which is pre-populated if they have saved their information in the past)
- Their order history
- Their wishlist
Staff members also have access to a page that shows all profile pages, to allow them to know how many users the site has.
When a staff member is logged in, they also have access to the following pages:
- Inbox
- All users
- All orders
Inbox
Here, staff members can see all messages submitted by the contact form.
All orders
This view shows staff members all orders, ordered from newest to oldest.
All users
Similarly to all orders, this page shows all the site's users - allowing staff to see how popular the site is
Once a user has added an item to their basket, they will be able to navigate to it via the dropdown menu (or by clicking the basket on mobile). Here, they are able to see all of the items currently in their basket, the cost of these items, the delivery cost, how much more they need to spend to qualify for free delivery (where applicable), and their grand total.
From the basket, users will be able to continue through into the checkout page. Once they have filled out the checkout form, if they have been successful, they will be redirected to a success page, and an email confirmation will be sent to them.
- Tours: Add a section to the site which allows user's to search/buy tickets for an artist of their choice's tour.
- Membership: Add a means for users to sign up for a paid membership, which gives them access to perks such as pre-release sales and free shipping.
- Featured item on homepage: Add a section to the homepage in which admins can set a featured artist/product/blog.
- Double albums: Add a double album product type that is similar to an album, but has room for two tracklists.
- Limited edition tag: Similar to the 'sale' and 'featured' tag - add in another, different colour, tag to show that an album is limited edition/collectable.
- Individual song pages: allow users to purchase, and see information about, individual songs.
- Product comments: allow users to add comments on product pages, to leave their feedback.
- Low stock tag: Add a tag - similar to the 'sale' and 'featured' tags - that let the user know that an item has low stock; this may encourage more sales if the user doesn't wish to miss the chance to purchase the item.
Tests can be found here in a separate file.
Bugs for this project can be found within the project's 'Issues' tab - for ease, I have added links to each bug's issue log below.
No. | Bug | Link | Completed |
---|---|---|---|
#1 | Basket dropdown scroll icon added on every hover, but not removed | Link to issue #1 | ✅ |
#2 | Blog listing page errors when no featured blog is selected | Link to issue #2 | ✅ |
#3 | Album track listing not sorting by track number | Link to issue #3 | ✅ |
#4 | Album info table not ordering items properly when there are 10 or more items | Link to issue #4 | ✅ |
#5 | 'Sale' tag displays above dropdown menus | Link to issue #5 | ✅ |
#6 | Quantity buttons' disabled state | Link to issue #6 | ✅ |
Issues 7 and 8 removed as they are no longer relevant | |||
#9 | Sales prices displaying incorrectly when more than one size added to basket | Link to issue #9 | ✅ |
#10 | Basket remove button removing all sizes of same item | Link to issue #10 | ✅ |
#11 | Inbox search returning empty inbox page if no queries are matched | Link to issue #11 | ✅ |
#12 | Pages only showing for superusers. | Link to issue #12 | ✅ |
#13 | Non-superusers cannot add items to wishlist | Link to issue #13 | ✅ |
- Log in to your GitHub account and navigate to this repo
- Copy the HTTPS code (highlighted in red below)
- Open your terminal and navigate to the directory you would like to clone this project into
- Run the following command in your terminal
git clone
followed by the HTTPS link from step two - Once the project has cloned, open it in your code editor and add an env.py file to the main directory, this file will require the following environment variables:
- CLOUDINARY_CLOUD_NAME: This can be found on your Cloudinary account.
- CLOUDINARY_API_KEY: This can be found on your Cloudinary account.
- CLOUDINARY_API_SECRET: This can be found on your Cloudinary account.
- DATABASE_URL: Leave this one blank for now, we'll add this during Heroku setup
- EMAIL_HOST: Your smtp email host - for this project, I used Gmail
- EMAIL_HOST_USER: Your email
- EMAIL_HOST_PASS: A password generated by your email provider, to allow access into your account
- EMAIL_PORT: Your email port
- SECRET_KEY: A randomly generated key, which can be created using online services, or by running this command in your terminal
- STRIPE_PUBLIC_KEY: Leave blank - we'll add this in during the following 'Stripe Setup' step.
- STRIPE_SECRET_KEY: Leave blank - we'll add this in during the following 'Stripe Setup' step.
- STRIPE_WH_SECRET: Leave blank - we'll add this in during the following 'Stripe Setup' step.
- AWS_ACCESS_KEY_ID: Leave blank - we'll add this in during the following 'AWS Setup' step.
- AWS_SECRET_ACCESS_KEY: Leave blank - we'll add this in during the following 'AWS Setup' step.
- DEVELOPMENT: If add this variable in with a variable of 1, it will activate debug mode in Django.
- Go to the Stripe website and login (create an account if you haven't already).
- From the 'Home' tab, copy both the 'Publishable key' and 'Secret key' and add them to your config variables from the previous section.
- Navigate to the 'Developers' tab, and then the 'Webhooks' in the side menu.
- Click '+ Add endpoint', add the url of your site, with /checkout/wh added to the end - i.e.
https://www.yourwebsite.com/checkout/wh
- NOTE: You will also need to take note of your webhook key here - we will need this for the environment variables.
- (Optional, but recommended): Trigger a payment to test your webhook.
- NOTE: If you set up a
DEVELOPMENT
variable earlier, you need to get rid of it, or your webhooks will fail.
- Go to the AWS website and login (create a personal account if you haven't already - don't be intimidated about having to enter debit card information, you will not be charged).
- Navigate to AWS Management Console, under My Account.
- Search for service S3, and create a new bucket; giving it the same name as your website, and selecting the nearest region to you.
- NOTE: Uncheck block all public access and acknowledge that the bucket will be public access.
- Also, be sure to set 'Object Ownership' to 'ACLs enabled' and 'Bucket owner preferred'
- On the following page, click on your newly created bucket.
- Navigate to the properties tab, and select 'Static Website Hosting', and use it to host a website.
- The index and error documents can be filled with default values, as they won't be used.
- Next, navigate to the permissions tab, and update the following sections:
- In the CORS section, paste the following:
[
{
"AllowedHeaders": [
"Authorization"
],
"AllowedMethods": [
"GET"
],
"AllowedOrigins": [
"*"
],
"ExposeHeaders": []
}
]
- In the bucket policy, select a type of 'S3 bucket policy', allow all principles by filling in the field with a star ('*'), select the action will be 'GetObject', and then paste in your arn key from the bucket policy section on the other tab, and paste it into the ARN field at the bottom of the form (keep a note of this ARN value for later). Once this is done, click 'Add Statement', then 'Generate policy', then copy this policy into the bucket policy editor.
- NOTE: Add a '/*' onto the end of the "Resource" key here, to allow it access to everything.
- Then, click save.
- In the ACL list tab, click edit and enable List for Everyone (public access) and accept the warning box. If the edit button is disabled you need to change the Object Ownership section above to ACLs enabled.
- Search for 'IAM' via the services search - or locate it in the services menu.
- Select 'User Group' from the sidebar, and create a group (it's a good idea to name it 'manage-' followed by your project name).
- You will need to click through some extra steps to see the 'Create Group' button.
- Next, select 'Policies' from the sidebar, followed by 'Create policy'.
- From here, click 'Import Managed Policy', search for 'S3', and import 'AmazonS3FullAccess'.
- Copy your ARN key into the "Resource" key twice using a list, the first version should be your ARN key, and the second should be your ARN key followed by '/*', i.e.
"Resource": [
'your-arn-key',
'your-arn-key/*'
]
- Click through the next pages until you find the 'Review Policy' page. From here you can give it a name and description - related to your project - and click 'Create Policy'
- Next, we need to attach the policy to the group we made earlier. To attach the policy, on the sidebar click User Groups. Select your group, go to the permissions tab, open the Add permissions dropdown, and click Attach policies. Select the policy and click Add permissions at the bottom.
- Finally, navigate to 'Users' and create a new user - named after your project name. Give them programmatic access (in the checkboxes), and continue to the next page.
- Select your group, which has the policy attached, and click through until you can 'Create User'
- IMPORTANT: Download the CSV file, which contains essential information, and cannot be downloaded again.
- Install the following packages:
pip3 install boto3
andpip3 install django-storages
(and run freeze > requirements.txt) - Add storages to installed apps.
- Create an if statement to check if there is an environment variable called 'USE_AWS' in os.environ
- Define AWS_STORAGE_BUCKET_NAME (your bucket's name), AWS_S3_REGION_NAME (that you selected earlier on), AWS_ACCESS_KEY (which we can get from the environment - the value of which we can find from the CSV file downloaded earlier), and AWS_SECRET_ACCESS_KEY (again, from the environment, the same as AWS_ACCESS_KEY).
- Set up a variable called AWS_S3_CUSTOM_DOMAIN, which should be set up using an f-string as follows:
f'{AWS_STORAGE_BUCKET_NAME}.s3.amazonaws.com'
- Create a new file called 'custom_storages.py' in the root directory, and import settings and storages (
from storages.backends.s3boto3 import S3Boto3Storage
) - Create a custom class called StaticStorage, and another called MediaStorage, the final result will look like this:
from django.conf import settings
from storages.backends.s3boto3 import S3Boto3Storage
class StaticStorage(S3Boto3Storage):
location = settings.STATICFILES_LOCATION
class MediaStorage(S3Boto3Storage):
location = settings.MEDIAFILES_LOCATION
- Finally, go back to settings.py and add the following to set the locations of our static files:
# Static and media files
STATICFILES_STORAGE = 'custom_storages.StaticStorage'
STATICFILES_LOCATION = 'static'
DEFAULT_FILE_STORAGE = 'custom_storages.MediaStorage'
MEDIAFILES_LOCATION = 'media'
# Override static and media URLs in production
STATIC_URL = f'https://{AWS_S3_CUSTOM_DOMAIN}/{STATICFILES_LOCATION}'
MEDIA_URL = f'https://{AWS_S3_CUSTOM_DOMAIN}/{MEDIAFILES_LOCATION}'
- Navigate to the Heroku Website and login (sign up if you haven't already)
- Create a new app, name it, and select the region nearest to you.
- From here, in the 'Overview' section, navigate to 'Add-ons' and select Postgres, under the 'Hobby Dev' plan.
- Click into your newly created Heroku Postgres database, and take a note of your database URL, and add it to your config variables as
DATABASE_URL
.
- NOTE: At this stage, you should also makemigrations and migrate to your new Postgres database, to ensure you have all your models set up.
- Next, under the 'Settings' tab, scroll down and select 'Reveal Config Vars' - here we are going to set all of the required config variables set up earlier:
- NOTE: The 'USE_AWS' variable should only be added to the Heroku config variables, so you can add this in now:
USE_AWS
with a value ofTrue
- Navigate to the 'Deploy' tab, and under 'Deployment Method' select 'GitHub'.
- Connect your account, select your project, and the branch you wish to deploy from.
- Enable automatic deploys - as this will deploy your site every time you commit changes to GitHub.
- Finally, click 'Deploy Branch'.
- Sit back, relax, and let Heroku do the work for you!
FINAL NOTE
IF you encounter any problems while deploying - it is a good idea to use the Heroku build logs to find the problem. Also, temporarily enabling the DEVELOPMENT environment variable, thus turning on Django's DEBUG mode, can be very useful; but make sure you delete the DEVELOPMENT variable when you are finished.
- Codemy for demonstrating how to upload files through Django forms.
- Elf Sternberg for the code to regroup the artists queryset used on the /artists page.
- Code Artisan Lab for the code to set up the related items list on product pages
- Laffuste & Phoenix for the syntax to iterate through RelatedManager objects.
- Wouter J showed me how to use for loops in SCSS, which allowed me to add the homepage social icons animation in simply.
- Frost.baka, for the code to stop admin adding another model in the admin area for the homepage.
- Geeks For Geeks, for showing me how to add a rich text field to the Django admin area.
- MaximeK, for showing me how to extend Django models - which I used to extend the Product model into the Album/Merch models.
- Thanks to Unsplash for all the product images.
- Thanks to Freepik for the vinyl favicon
- Thanks to Natalie Parham for the homepage image
- I would like the thank the tech Twitter community for helping me brainstorm artist names for testing.
- I would like to thank my parter, Laura Gilbert, for helping me test the site from a user's perspective.
- I would like to thank my mentor Simen for all of his feedback and support with this project.