Giter VIP home page Giter VIP logo

ms3-coffeedb's Introduction

MS3: CoffeeDB

Data-Centric Development Project Submission

Access

View the deployed project: here

View the Github repo: here

The 'All Reviews' page is only visible when more than 3 reviews exist in the database per submission.
Click here to see an example of a submission with more than 3 reviews in order to experience the 'All Reviews' page.

Please note that any issues relating to the speed of CoffeeDB is currently due to a temporary throttling issue with MongoDB.
Click here for a more detailed explanation.
Click here to see the actual performance scores from Lighthouse.

Example input

If you do not wish to search for oyur own coffee to submit, feel free to use the below example input data.

Click here for the image. (Download to your device and upload to CoffeeDB)

Brand: Eat Your Hat

Name: Wholebean Espresso

Roast: Medium

Origin: Blend

Organic: Yes

Website: https://www.eatyourhat.com/products/whole-bean-espresso/

Notes: Milk Chocolate, Peanut

Table of Contents

Strategy

User Stories & Project Objectives

"As a user..."

I want to be able to discover new coffee based on attributes that I like, without having to look around dozens of different coffee roaster websites. CoffeeDB provides a single source solution providing users with a database of coffee beans from around the UK. Users are able to filter the results based on their criteria, such as roast type, origin, whether it is organic and the flavour notes.

I want to be able to read independent reviews and see user ratings in order to help me with buying decisions on what my next coffee purchase will be. Users are able to review and rate coffee on CoffeeDB, and a convenient 'average rating' is provided for fair comparisons.

I want to add coffee to the website if I've recently discovered a new bag. Signing up is quick and easy, anyone with an account can submit new coffee to the database and we provide lists of previously-entered attributes like origins, brands and flavour notes to help prevent duplicate values. Submissions are also automatically checked for duplicates based on the coffee name entered. Users can easily see the submissions they've entered by visiting their profile page, which is publicly accessible so you can share your submissions with your friends! We also provide you with the ability to edit your submissions in case you entered incorrect information or have a better photo you'd like to have instead.

Scope

Current Features

CRUD

  • Create

    • Visitors to CoffeeDB can signup and create an account
    • Users can add coffee submissions to the database
    • Users can rate coffee submissions which are added to the submissions database entry
    • Users can write reviews which are added to the submissisions database entry
  • Read

    • Anyone can browse through all coffee submissions. Filters assist with narrowing the results down to what the user is trying to find and a search input is available so the user can type in a brand name of coffee product name.
    • Anyone can see user profiles, this allows the user to see what submisisons other users have made.
    • Anyone can see the average rating for each coffee submission, including the number of ratings that have been made.
    • Anyone can see the reviews for each coffee submission, and they are able to sort through by rating, brand/coffee name or date written.
  • Update

    • Account information is easily updatable from a users profile page. This allows the first name, last name, email and username to be updated. Duplicates are detected so a user cannot choose an email or username that already exists.
    • Submissions can be edited if the corresponding user is logged in. All attributes are editable and if custom values are added or removed then the options within the Browse page filter are adjusted accordingly.
    • Ratings can be changed if the corresponding user is logged in. The user can simply click on the star icon they wish to change their rating to, and it will update the value in the database.
    • Reviews can be edited in case the user spots a spelling mistake or has changed their opinion. A submission's view page will automatically enter the pre-existing review in the text input field and submitting a new review will replace the original and update the review timestamp.
  • Delete

    • Users are able to delete their accounts. They must enter their username and password in order to make this action, and 2 clicks are needed to facilitate the deletion of an account. Users are notified that their account details become available for use if another user signs up with the same information. Users are also notified that all of their submissions are deleted when they delete their account.
    • Submissions can be deleted if the corresponding user is logged in. The user can simply click on the EDIT button in the top right corner of any of their submissions which will present a DELETE option at the bottom of the edit fields. 2 clicks are needed to facilitate the deletion of a submission.

Features

  • Index

    • Search input available for users to quickly enter a brand or coffee product name straight from the homepage. Autocomplete has been added to this search field, so if the user begins to write an already existing brand name or coffee name they are presented with it as a suggested input.
      Click here to read about how the autocomplete code works.
    • A down arrow has been animated (moving up and down) to grab the users attention to indicate there is more content below the initial hero splash page they see when they first visit the homepage.
    • Clicking the down arrow scrolls the user down to the container below the title container, but scrolls smoothly so it is clear that the content is further down and not actually on a different page.
    • I have included a 'latest addition' section. One of the purposes of this is to motivate users to add more content to the database as their input is highlighted on the homepage of CoffeeDB for however long until the next user adds a submission.
    • Top rated submissions, showing the 5 highest average reviews. This helps users who want to try the coffee that the userbase of CoffeeDB enjoys the most. If multiple submissions have the same rating, eg. 5.0, then it is ordered by number of ratings, which is also visible to the user.
    • Recent reviews, showing the most recent 3 reviews submitted to CoffeeDB. Similar to the 'latest addition' section, the goal is to highlight users input in order to motivate more reviews to be written as it is highlighted on the main page of the website. This also helps users hoping to discover new coffee as this section will always be different based on new reviews written.
    • A video has been included to draw attention to a link to the browse page, arguably the second most important page on CoffeeDB after the homepage. This video is 985KB in size, which will mean it will load quickly on mobile devices and it won't eat up a lot of data for those on limited or expensive data plans.
    • To hide any compression artefacts on the video, it has been darkened, which serves a secondary purposes of making the overlayed text easier to read.
    • The video is made up of two videos which loop in sequence, this was to avoid the same clip looping back to its starting position by having it cut to another shot.
  • Browse

    • Search input
      available for users to enter a brand or coffee product name, in addition to another additional filter such as roast type, origin, organic, flavour notes. Autocomplete has been added to this search field, so if the user begins to write an already existing brand name or coffee name they are presented with it as a suggested input.
      Click here to read about how the autocomplete code works.

    • Filter options:

      • Roast: Dark / Medium / Light
      • Origin default values: Brazil / Ethiopia / Blend / Colombia, other values are taken from any custom origin inputs users have added to the database. 4 are shown and then a button is presented that will present all origins to be shown.
      • Organic: A toggle switch which when set to 'Required' will only show results that are marked as being organic. The default value is 'Not required'.
      • Popular tasting notes: This is presented in the form of a 'word cloud'. The font size of a 'note' is representative of its frequency of occurance in the database. The top 10 are shown by default and then a 'Show more' button is presented for the user to see all notes.
        Click here to read about how the word cloud code was written.
      • All/any toggle: This toggle allows the user to specify if their results should contain ALL of the tasting notes that they have checked, or if their results can contain any of the notes that they have checked.
      • The filter allows for multiple selections, for example the user can select to see results that are either Dark or Medium roast, from Brazil or Colombia but that must have a flavour note of 'Dark chocolate'.
      • The filter submission will generate a GET request which means the URL of the search results reflect the users criteria. This allows the user to bookmark the page for future reference or send to a friend. A potentially common use-case of this could be a family member wanting to buy coffee as a gift, the user can send their preferences as the URL so the family member can choose a coffee that the user will likely enjoy.
        Click here to read about how the dynamic filter queries are generated.
    • Sorting:

      • Date (Most Recent) [default]
      • Date (Oldest)
      • Name (A > Z)
      • Name (Z > A)
      • Brand (A > Z)
      • Brand (Z > A)
    • Pagination
      allows the results to be split over multiple pages. CoffeeDB will only present 6 submissions per page as this occupies a reasonable amount of screen space on all devices from mobile to desktop without introducing a huge area for the user to have to scroll through. Displaying only 6 results per page also reduces the amount of time needed to load a page, and reduces the amount of data downloaded if the user didn't want to search through all results. The user is also informed about how many results were returned from their filter criteria, along with the page that they are on and how many pages their criteria is split across.
      Click here to read about how the pagination code was written.

    • Validation
      When a user makes a filter selection, they are presented with the filters listed above the results so that if they wish to remove one of the filters they are able to. This also validates their selection so they know if they've made a mistake.
      Click here to read about how this validation code was written.

    • Navigation
      As the homepage has an immediate hero container presenting the CoffeeDB logo and a dedicated navigation, a secondary nav was needed for when the user scrolled down past this container. The mobile nav appears when the user begins to scroll and its positioning becomes 'sticky' when the secondary nav hits the top of the users viewport. If the navigation is collapsed and the user scrolls back to the top, the navigation is closed automatically.

  • Submission card

    • The top left corner displays the average rating for each submission, allowing the users to see the consensus of the quality of the coffee immediately from the browse page.
      If the submission being viewed was submitted by the user viewing it, then the top right corner displays an EDIT button which takes the user through to the editing page. The user-submitted image is displayed prominently and along with the brand and coffee name, which are linked to the View Submission page if clicked.
    • Attributes are then shown, which include roast type, origin, whether it is organic and an optional link to the coffee roasters website.
    • A maximum of 4 flavour notes are permitted and shown on each submission card, and each one is linked to a filter criteria showing all submissions containing the clicked note.
    • Finally the username of the user who made the submission is shown, so that others can see what other submissions the user has made.
  • Add

    • Image uploader which will only accept image files and is sent to an external image hosting service.
      Click here to read more about how the images are kept in the database.
      Click here to read about how the image hosting API code was written.
    • Brand and Origin dropdown 'select' inputs. This presents a list of brands/origins that they user can select. The values have some default values, and the other values come from whatever users have previously entered into the database. Another option is to select 'Other...' which will then disable the dropdown selector and dynamically create a text input field below. This allows the user to type in a new brand/origin that doesn't exist as an option already. Autocomplete has been added to this text input to assist users in case they're entering a value that already exists.
    • Notes only permits the user to 'check' 4 values, the other notes become disabled when 4 have been selected by the user, until one is unchecked. The user can type in a new flavour note if one does not exist as an optional already, Autocomplete has been added to this text input. If 3 ntoes have been checked and the user types in a 4th manually then the disabled state still gets trigerred.
    • Submitting data to the databse will add any new attributes like custom brand/origin/notes to the options in the filter section on the Browse page. They also become options if any user then goes to add a new entry.
    • For desktop users, a live preview of their input is available on the right half of the screen. This dynamically updates with the users input while they're typing, including the image they have uploaded. This is primary so that the user can see what their submission will look like before they submit. In particular, this provides validaiton that the image they have uploaded is proportioned correctly, so the image live preview is also available for mobile users.
  • Profile (submissions)

    • This page is publicly accessible by anyone. However if the user logged in is the user viewing the profile page, they also have access to a sub-navigation menu that allows them to go to two additional pages, one to update their account details and another to delete their account.
    • Displayed at the top informs users of the first name of the user, the date they signed up to CoffeeDB and how many submissions they have contributed.
    • Pagination has been included to split their submissions if they exceed 6.
    • Sort options are available.
  • Profile (update account) / Signup

    • If updating their account, users can make changes to their first name, last name, email and username without having to enter their password.
    • Whenever signing up or updating an account, entering a first name and last name will automatically fill in the username with the two names concatenated as a suggestion.
    • When updating an account, changing the password requires the user to enter their existing password.
    • Both signing up and updating an account share a UI that validates if the users password meets the criteria.
    • Sign Up / Change Password submit buttons are disabled by default and become active when the password criteria is met.
    • Updating the username updates the session token.
  • Profile (delete account)

    • 'Delete my account permanently' is disabled until the user has entered at least 3 characters into both text input fields.
    • Clicking 'Delete my account permanently' will prompt a second button to appear, labelled 'Permanently Delete' to ensure the user understands the action they're committing to.
  • View Submission

    • The UI is made up of two sections, the submission card and options to rate or review the submission.
    • If a user attempts to rate a submission when they're not logged in, they are prompted with a link to login.
    • The review text input field is disabled if a user is not logged in.
    • When the user hovers over the rating icons it will fill in the icons up to the one they are hovering over.
    • Submitting a rating will update the 'Your rating' section, the 'Average rating' will reflect the changes in numbers and the submission card average in the top left corner will update.
    • Writing a review will count down the character counter to inform the user of how many characters they have left.
    • If the user has already submitted a review, the review text input field will already contain their review so they can edit it or replace it with another.
    • The character counter will reflect the number of characters if the review already exists and is in the text input field on page load.
    • Reviews will show the review text, username of the reviewer, timestamp of when the review was written and their rating if it exists.
    • The review timestamp is recorded in UTC but displayed to the user in their local timezone.
    • If more than 3 reviews exist, a link to 'View all reviews' is triggered.
  • All reviews

    • Click here to see an example of a submission containing more than 3 reviews which triggers a link to appear to visit an 'All Reviews' page.
    • All reviews for a submission are listed on one page.
    • The ability to sort reviews by date or number of ratings is available.
    • As the 'all reviews' page is 2 pages deep, meaning the previous page is not accessible via the navigation menu, a back button is presented which returns the user to the 'View Submission' page.
  • 404

    • If a user tries to access a page that doesn't fit the correct state, eg. a logged in user trying to access a signup page... then they are redirected to a relevant page, in this example they are redirected to their profile page.
    • If the user tries to access a page where the endpoint does not exist, they are presented with a 404 error page with a link to take them back to the homepage.
  • Login

    • The user is able to login with their email and password.
    • Their username is used as the session token, which is updated if they ever change their username.
    • A link is provided to signup if the user hasn't an account already.

Long-term vision

  • Currently when inputting a value that cannot be a duplicate, eg. an email, username, coffee name etc. it does not tell you in real time that is already exists, it requires the form to be submitted to then provide validation. In a future build I would introduce AJAX to be able to provide live validation for a better user experience.
  • A bug exists where if a user deletes their account their reviews still exist. For GDPR reasons it would be better to remove these, but it will also impact the user experience in that the reviews will still link to the users profile which no longer exists since deletion.
  • Currently a user is not able to delete ratings or reviews, only edit them. This should be introduced in a future build.
  • While CoffeeDB demonstrates usage of a non-relational database, it also makes use of MongoDB's ability to include limited relationships between sets of data. While this is functional in this current build, for CoffeeDB to expand and provide features such as recommending coffee to users based on their previous ratings/reviews, a transition to a dedicated relationship database management system such as MySQL would be beneficial.
  • In a future build I would like to include the ability to share profile pages over social media. This would be beneficial to the user as it would provide easy means to share their favourite coffee with friends. It would be beneficial to CoffeeDB in that it would promote word-of-mouth advertising.
  • Within the users profile page it should provide recommendations of coffee based on the what ratings the user has given to other coffee submissions. This suggestion would be based submissions that were also highly rated by users that rated the common submission.
  • A future implementation should include the ability to have different user types, with different permissions. Currently all users are treated the same, there is no admin user other than who has access to the code itself (me). I would like to introduce an admin user so that reviews/submissions can be removed easily if they contain inappropriate material.
  • The current build of CoffeeDB only permits sorting by rating on the 'All reviews' page. All other sort functionality is purely by brand name/coffee name and submission date. This was partly due to the nature of MongoDB not fully supporting relationships. A change to the schema may allow this, by separately ratings and reviews to their own collection, but the benefits need to be weighed when deciding whether to proceed with a document based database such as MongoDB versus a relational database like MySQL.
  • Currently a user is unable to easily see all submissions they have rated or reviewed, it would rely on them remembering the submission and finding it via Browse. I would like this to be introduced within their profile page in a future build.
  • The ability to 'watch' other users so that you are notified if someone you follow has submitted a new coffee, or recently reviewed/rated a coffee. This would introduce a light social network to CoffeeDB which I believe could influence users to return more often.
  • The search container is positioned at the top of the second half of the homepage when viewing on desktop. Due to the grid system in bootstrap it is moved below other containers when viewing on mobile. For consistency I would like to update this to be more consistent.
  • A bug exists with the collapsible filter controls on the Browse page that should be rewritten in a future build. If the user clicks to open the filter controls more than once in quick succession it will confuse the toggle state and be in an open state when the controls are closed, and vice-versa. This is more likely a Bootstrap component issue, but a workaround can be found, or alternatively a different method of writing a collapsible container can be included.

Code Walkthrough & Challenges

Auto-suggest

In order to provide suggestions to the user while they are typing into a text input field, I have used jQueryUI's autocomplete plugin. I used this plugin in my MS2 project which was written entirely in JavaScript.

However, because CoffeeDB is a Flask app, the values don't exist natively in the front end, so this presented a challenge on how I was going to pass this information to the source list for autocomplete. If the information was visible on the page, this would be fairly easy, as jQuery could read from this list in the DOM and use this as the source. However, the information that needed to be passed was a list of brands, coffee names and flavour notes.

My research pointed me to the use of AJAX, which I would like to include in a future build as I believe it would be useful elsewhere on CoffeeDB, such as verifying if a users input already exists in real-time (like signing up and typing in a username). My solution was to create a new view called autocomplete which wouldn't be linked to any on CoffeeDB, so it would only be accessible to those who know the URL. While this isn't secure, I determined that this was an appropriate use-case as the data was not sensitive. I considered using this method to include verification of pre-existing usernames, but deemed a list of all usernames to be too sensitive.

@app.route('/autocomplete', methods=['GET'])
def autocomplete():
    # GATHER DATA TO BE PASSED TO FRONTEND
    values = {
        "brands": list(getCoffeeData()['brand_names']),
        "names": list(mongo.db.beans.distinct('name')),
        "origins": list(getCoffeeData()['origin_types']),
        "notes": list(getCoffeeData()['unique_notes'])
    }
    # CONVERT LIST TO JSON
    json_values = json.dumps(values)
    # SEND TO TEMPLATE
    return jsonify(autocomplete_values=json_values)

I created the view, fed the corresponding data into a new dictionary named 'values' and had this converted to JSON and be returned.

    // AUTOCOMPLETE ASSIGNMENT
    $.getJSON('/autocomplete', function(data) {     
        // PARSES JSON DATA PULLED FROM MONGODB
        var jsonParsed = JSON.parse(data.autocomplete_values);
        // CONCATENATES BRANDS AND NAMES TO SINGLE LIST
        var autocompleteData = jsonParsed.brands.concat(jsonParsed.names)
        // ASSIGNS LIST OF BRANDS, NAMES TO SEARCH INPUT AUTOCOMPLETE
        $( "#searchInput" ).autocomplete({
            source: autocompleteData
        });
    });

In the JavaScript code of any page that needs the auto-suggest feature I had a function communicate with the JSON data and then pass it to jQueryUI's autocomplete source.

Storing Images in MongoDB

This was probably the most time-consuming challenge I faced while working on CoffeeDB.

Version 1 (Image URL)

The original implementation of allowing users to input an image with their coffee submission was simply to allow them to enter a url to the image into a text input field. This would then store the url as the value to a key within the submission document in the 'beans' collection in MongoDB. This worked, but it didn't produce a great experience on mobile, as copying and pasting the url feels long-winded. It also didn't have a way of verifying for sure that url entered resulted in an image. I briefly implemented some validation that checked the url ended with a filetype such as .jpg, .png etc. but increasingly more websites are obscuring their url from including the filetype.

Version 2 (FileStack API)

I had decided that I wanted a file picker. This would solve the issue for mobile users having to copy/paste in order to input an image, as they could directly upload from their photo album (after saving an image from a website) or directly take a photo of the coffee themselves. As MongoDB can't directly store images, I decided to research into implementing a third party API that meant the file picker was actually sending the users image file to an external service which provided the hosting. FileStack worked great, and had the added benefit of presenting the user with a sleek UI that allowed them to crop/resize their image before submitting. My implementation of FileStack's API was done on the frontend, in JavaScript, so this presented a challenge of how to pass the returned image URL to Flask. I decided to allow users the choice between using this file picker or entering an image url. This meant I could use JavaScript to return the uploaded image's URL into the text input field, which would then get submitted along with the rest of the submission data in the POST request. Unfortunately, it became apparent that FileStack's free tier was very limiting as I soon ran out of my free allowance for number of requests to their server and the file picker stopped functioning when the user tried clicking on it.

Version 3 (Base64 in MongoDB)

While working on MS2 I had worked with base64 to encode some CSV data to allow the user to download it to their disk. I decided to encode the users uploaded image in base64 and store this string in MongoDB. This was the solution that I had landed on for the longest time as it seemed to tick all the boxes and work without any issues.

Unfortunately all of a sudden CoffeeDB's homepage, browse page and profile pages all became extremely slow to use. After some troubleshooting I discovered this was because of MongoDB, as when I tried to view the data in the database it was also very slow to use. After contacting MongoDB's developer support they explained that because I was using the free tier they had throttled my account as it had sent 12gb of requests within the period of a week.

I initially didn't understand how this was possible as the combined size of data in MongoDB came to just a few megabytes. I looked through my code and discovered that where I was not able to write a MongoDB find query to get access to sub-arrays (ratings), I had written a loop to extract the information I needed and this meant iterating over the large base64 strings over multiple times. I was able to mitigate this by using 'projection' in my MongoDB query.

    # # COLLECTION CONTAINING ALL DOCUMENTS WITH RATINGS
    ratingsTrue = mongo.db.beans.find({"rating": {"$exists": True}},
                    projection={"rating": 1, "brand": 1, "name": 1})

Using 'projection' meant I could ask it to only return the values I needed. While this would have definitely improved the size of the requests I was making, it was too late and the throttling had been enabled and I was advised it would be a while until it would be disabled. I decided to avoid storing base64 encoded images in MongoDB and explore another API solution.

Version 4 (ImageKit.io API)

The final solution I settled on was similar to version 2, in that I implemented a third party API to store the images externally. Similarly, the direct image url would be stored in MongoDB. I was able to find a service that offered a generous 20gb of storage for free and provided a Python API, which resolved the issue of how to pass the url from the front-end to the back.

def uploadImage(image):
    imagekitUpload = imagekit.upload_file(
        file = image,
        file_name = image.filename,
        options={
            "is_private_file": False,
        },
    )
    return imagekitUpload

I created the above function, guided by the ImageKit documentation, and had it return the response JSON. I have another function named 'gatherInputs' which is used twice on CoffeeDB, on both the Add page and Edit page. This gathers all of the user inputs of a coffee submission, and calls the uploadImage function in order to get the image url to store into MongoDB.

Dynamic Queries

The default view of the Browse page is to show all results, in order of most recently added. The user is able to select from multiple options to form their criteria, this includes roast type, searchable brand name/coffee name, origin, whether submissions should be organic, and tasting notes. In order to process the query, the form sends a GET request which is processed by the browse view in Python.

I created an empty list called 'dynamicQuery' which gets added to depending on the arguments of the GET request. If arguments are detected for each input in the filter controls then it gets appended to 'dynamicQuery'. Once all of the arguments have been appended, a MongoDB find query is made using '$and' so that multiple arguments can be passed.

There are two instances where the query doesn't use '$and'. The search input field where the user can type in the brand name or coffee name uses '$or' so that it can check whether the query text matches either the 'brand' key or 'name' key.

The other instance is when the user can choose between the results displaying all or their selected tasting notes, or any of the selected notes. This uses MongoDB's '$all' and '$in' respectively.

Shortened example of the dynamicQuery appending the arguments for 'roast' and 'notes':

dynamicQuery = []

if request.args.getlist("roast"):
            dynamicQuery["$and"].append({"roast": {"$in":
                                        request.args.getlist("roast")}})
if request.args.getlist('tag'):
            if request.args['conditionType'] == "all":
                dynamicQuery["$and"].append({"notes": {"$all":
                                            request.args.getlist('tag')}})
            if request.args['conditionType'] == "any":
                dynamicQuery["$and"].append({"notes": {"$in":
                                            request.args.getlist('tag')}})
beans = mongo.db.beans.find(dynamicQuery)

Word cloud (tasting notes)

One of the most fun challenges to work on was forming a 'word cloud' to represent which tasting notes were most common in the database. I intentionally didn't research how to go about writing this, as I wanted to challenge myself to see if I could do it independently.

The word cloud function receives 2 arguments, a list and a unique list. For my usage, I passed a list of all tasting notes used in the database, duplicates included, and another list containing only the unique values (using MongoDB's 'distinct' query).

# CREATES TUPLE WITH UNIQUE ITEM AND ITS NUMBER OF OCCURANCES
    itemCount = []
    for x in uniqueList:
        itemCount.append((x, list.count(x)))
    itemCount = sorted(itemCount, key=lambda item: item[1], reverse=True)

I used a for loop to iterate over the unique list and count how many times that particular note occurred in the non-unique list. This list was then sorted in order of the most frequently occurring note in the database.

    itemPercentage = {}
    for item in itemCount:
        # RETURNS NUMBER OF NOTES
        length = len(uniqueList)
        # RETURNS HOW MANY TIMES NOTES OCCUR
        occurance = item[1]
        # DIVIDES TOTAL NUMBER OF NOTES BY OCCURANCE
        percentage = occurance / length * 100
        # ADDS TO DICTIONARY
        itemPercentage[item] = percentage

Rather than having the fixed number of how many times each tasting note occurred, I wanted the value to be a percentage relative to the length of the list of unique values.

Having the percentage allowed me to then organise the font size of the note depending on what bracket it fell into. I decided on 4 font sizes, determined by whether the note fell into <25%, >25%, >50% or >75%. The filter criteria only shows 10 notes by default, the list passed to the view is in order of the highest percentage, so I applied a CSS class called 'extra-note' to any values after 10 which had 'display: none' applied to it. This display property gets reversed by JavaScript if the user clicks the button to see all notes.

Pagination

Along with the word cloud, creating my own pagination was also a very exciting challenge. Similarly, I didn't include flask-paginate, a common pagination plugin for Flask apps, I also avoided researching into the functionality in order to pose a challenge to myself.

As the pagination is used in multiple places throughout CoffeeDB, I wrote the UI code in its own Jinja Macro so it could be easily imported. To write the logic, I wrote a function that received 2 arguments, the number of items to be viewed on any given page and the total amount of items to be displayed.

The default 'page' variable value is 1, offset is set to 0 and the 'pageQuantity' variable, which represents how many pages the data will be split over, is calculated by dividing the total number of items by the number of items per pay (6 in CoffeeDB's usage).

Within the function it checks for GET request arguments, specifically looking for a page number which the UI sends any time the user clicks on the next page or previous page buttons, the value is calculated by taking the current page number and adding or subtracting 1.

Once the function receives the page number via the GET request, it multiplies this number (subtracting 1 to represent the page number the use came from, not landed on) and multiplies it by the argument passed to it representing how many items to be shown on each page (6). This all results in an 'offset' variable which can be passed to the find query given to MongoDB when making the 'dynamicQuery' (see above) request.

# CREATES OFFSET AMOUNT BASED ON SPECIFIED QUANTITY SHOWN PER PAGE
def pagination(perPage, dataCount):
    page = 1
    offset = 0
    pageQuantity = math.ceil(dataCount / perPage)
    if 'page' in request.args:
        page = int(request.args.get("page"))
        currentPage = page - 1
        offset = currentPage * perPage
    return offset, perPage, page, dataCount, pageQuantity

Pagination function, also returning values that are passed to the view that form 'Showing results 1 to 6 (Page 1 of 4)' for example.

# REASSIGNS VALUES TO PAGINATION VARIABLES
offset, perPage, page, beansCount, pageQuantity = pagination(
  					6, mongo.db.beans.count_documents(dynamicQuery))
# REASSIGNS BEANS VARIABLE TO INCLUDE QUERY
# AND PAGINATION OFFSET/LIMIT
beans = mongo.db.beans.find(
  dynamicQuery).sort("_id", -1).skip(offset).limit(perPage)

Demonstrating sending the offset and limit queries to MongoDB.

Checkbox toggles

Throughout CoffeeDB you will see custom toggle buttons, they are used for specifying criteria to the Browse filter, along with adding a submission. These toggles are actually masked checkboxes which have their default checkbox UI hidden using CSS. The button text is still wrapped in a label, making these custom toggles accessibility-friendly.

Filter criteria validation

When a user makes a filter criteria query, they are presented with some UI validation which allows them to remove individual elements of their query, while informing them of the actions they just made.

Filter criteria validation screenshot

This validation is entirely built using Jinja. Jinja requests the current URL arguments, checks for origin, roast, organic status and tasting notes and for each category it forms a new GET request without including the element that was clicked. This results in the page being refreshed without the element the user wanted to remove from the criteria.

In addition to this, if the user opened the filter validation panel again they would find the options they had checked already highlighted, and if an origin or tasting note was ordinarily hidden in the 'show more' section, they would find that it is not grouped in the hidden section and instead if highlighted and visible immediately.

View structure

This diagram describes how the Jinja macros fit into the project and which views are inheriting which base templates.

Diagram of base templates and macros

Database Schema

Screenshot of MongoDB schema

UX

Wireframes

Wireframe

The project started with making a wireframe of the homepage and Browse page. You can see that the UI evolved fairly substantially, but from the beginning the UI decisions led the functional implementations, demonstrated by the early wireframe showing a word cloud as a way to display tasting notes. The dropdown selector within the search input field was eventually dropped in favour of a universal search, so it searches for brands and coffee names without the user having to specifically select which one.

Visual structure/language

The structure throughout CoffeeDB is made up of cards, making use of a Bootstrap UI class. This provided a clean, evenly spaced vertical structure to house the information about the coffee, which could be repeated in a loop easily (eg. Browse), or used individually (eg. homepage).

The homepage presents a hero splash page, with the CoffeeDB logo occupying the majority of the viewport, along with a dedicated navigation bar. This almost mimics the UX of an app launching, rather than being immediately met with your typical website structure you are instead presented with the brand and have easy access to the pages the user is likely wanting to visit first, eg. Browse or Add.

Further down on the homepage are a series of containers that relay the state of the database in that moment the user is visiting. It shows the most recent addition, the top 5 highest average ratings and the most recent 3 reviews. These all link the user to 'view submission' pages, providing an alternative way to discover coffee than just entering filters on the Browse page.

In order to keep the UX consistent, and to make use of repeatable code, both the Add/Edit pages are virtually the same, with small differences such as the title and form parameters, and the Signup/Update Account pages again relying on very similar code to produce a familiar experience.

By default, the Browse page filter controls are hidden, so the user is immediately greeted with the most recent coffee submissions so their discovery can begin. Bootstrap's collapse class was used in order to produce a smooth scrolling animation.

Original tasting notes screenshot

An older version of the checkbox UI before redesigning them to have a hidden checkbox.

Logo iterations

A selection of logos made before deciding on what became the final logo.

Colour

Given that CoffeeDB is largely about coffee, it made sense to have the colours match accordingly. However from the beginning I wanted to avoid using too much brown as I felt it created quite a dark theme and didn't feel fresh or inviting. I settled for a orangey/brown hue, with slight variations of the same colour throughout.

Main theme colour: #F8A825

Secondary theme colour: #643F00

Screenshot showing thick borderScreenshot showing box-shadow

A reoccurring motif throughout CoffeeDB is the use of strong borders and/or box-shadows. I felt as though having strong borders made identifying certain elements of the structure very quick and easy. A users first impression when they arrive at a page and they immediately know where to look as it is boxed in specific areas. Adding the box-shadow to certain elements, but not all, provided 'depth' to the design so it didn't feel overly boxy and square.

Typography/Icons

All non-default fonts were imported via Google Fonts.

Final logo

The CoffeeDB logo, found on the homepage and the navigation bar is made up of two fonts. 'Coffee' is using Pacifico, and 'DB' is using Codystar.

Header font screenshot

Headers, such as the title of a coffee submission, are using a bold Montserrat.

All other fonts are using default Bootstrap sans-serif fonts.


Homepage bean icon

3 bean icons are scattered on the homepage, providing a visual cue that the theme of the website is coffee. Source of icon.

Logo iterations

A variation on the bean icon presented on the 404 error page of an unknown endpoint, edited by me from source above.

404 bean icon

Arrow icons throughout CoffeeDB all originate from Bootstrap's icons collection.

Testing

Accessibility

Whereas an image is displaying, the appropriate alt tags are included. In addition to this, where there is a button element an aria-label has been written to ensure screen readers can identify the nature of the button and what its action is.

Google Lighthouse's accessibility checker provided the following results:

Page Score
Homepage 95
Browse 97
View Submission 96
Add/Edit 97
Signup/Update Account 95
Delete Account 97

Performance, Best Practices, SEO

After adjusting the MongoDB queries to project only what keys they need click here for more detail), performance scores using Google's Lighthouse were greatly improved.

Google Lighthouse's performance checker provided the following results:

Page Performance Best Practices* SEO
Homepage 100 80 90
Browse 99 80 90
View Submission 99 80 90
Add/Edit 98 80 90
Signup/Update Account 100 93 100
Delete Account 100 93 100

The 'Best Practices' score was only hampered by the fact that the website does not use HTTPS, however this is due to it being deployed using Heroku. If CoffeeDB had a public release it would be hosted elsewhere and use HTTPS to conform to modern security standards.

Manual testing

As there are numerous pages, and many features to test, I have written a dedicated markdown document detailing the manual test cases to ensure CoffeeDB is operating as expected.

Click here to view the manual test cases.

Code Validation

I have tested every page of CoffeeDB using W3's HTML and CSS validator, and all have passed with the exception of anything relating to the fact that the HTML code uses Jinja and anything referencing Bootstrap's own CSS files.

I have run the Python code through a pep8 compliancy checker, and it has passed without any issues.

Deployment

CoffeeDB is running off of Heroku, with synchronisation enabled to GitHub. The env.py file, which is not sent externally, contains the necessary MongoDB information (IP, PORT, SECRET_KEY, MONGO_URI, MONGO_DBNAME), as well as the public, private and endpoint for the ImageKit API. Before the final deployment I performed a pip freeze in order to update my requirements.txt file with any modules needed to run the app. Since the beginning of developing CoffeeDB I used Git for version control and worked in a virtual environment in order to keep track of which modules were imported, it also has the added benefit of freezing the version of the modules so that if I ever installed a newer version on my computer in the future it wouldn't risk breaking old code reliant on an older version of the module. The final action before releasing CoffeeDB to the public was to set Debug to False within the Python Flask code.

Credits & Attributes

Real Favicon Generator: for easy image resizing for various favicons.

Animated down arrow on homepage is based on a combination of StackOverflow posts. Link 1 and Link 2.

The homepage features two videos in a loop. Both videos are free for non-commercial use, and request a credit to the author. Link 1 and Link 2.

Credit to this Stackoverflow post for guidance on how to avoid Jinja printing None when appending to an array.

Credit to this Stackoverflow post for guidance on how to offset the position of a hyperlink scrolling to an anchor point to account for the height of a stickied navbar.

ImageKit API Documentation

This article for guidance on how to keep the footer stickied to the bottom of the viewport even if the content fits all on one page.

Credit to this Stackoverflow post for guidance on how to extract the timestamp from the Object ID in MongoDB.

The homepage and 404 page both make use of this icon, which is free for non-commercial use.

Credit to this Stackoverflow post for guidance on how to get the top value from a dictionary.

Credit to this Stackoverflow post which served as inspiration for how I would write a dynamic query for the Browse page.

Credit to this Stackoverflow post which helped when trying to figure out how to get the search input value from the homepage to a GET request landing on the Browse page.

Credit to this Stackoverflow post to help implement a random ordered list.

Credit to this Stackoverflow post for guidance on how to convert UTC to the users local timezone.

This page in Python's documentation was used for guidance on how to sort a dictionary by a sub-level value.

Credit to this Stackoverflow post for answers on how to work around DOM bubbling when targeting a dynamically created element.

Credit to this Stackoverflow post for guidance on how to convert a user uploaded image to base64 for the purposes of previewing it before submitting a form request.

This code from W3 was used to enable smooth scrolling on multiple browsers including Safari.

Credit to this Stackoverflow post on Regex examples for password criteria.

ms3-coffeedb's People

Contributors

rbsam176 avatar

Watchers

 avatar

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.