For the third Milestone Project, I decided to design a website where tattoo artists can register and add their profile to a database. Users (customers) can then search the database without registering and find their ideal tattoo artist. I had this idea since it is not easy to find a tattoo artist that does exactly what we want. This research might take a considerable amount of time, and people nowadays have always less free time. People need to find information easily and fast. The structure of the website is divided into two: one area is visible to all users, and one area is visible only to registered users.
View the live project here. Get My Tattoo
The base page has elements that all the pages have in common, which are a sticky navbar with collapsible toggler and a search box, plus a footer.
The home page welcomes the users and explains briefly what the website is about. Then a section with four cards shows some tattoo styles to inspire users, but also to guide them through the navigation. Many times users don’t have a clear idea in mind and need a guide to help them to start the navigation.
Right after this section, I repeated the search box and an area with the last 5 tattoo artists that added their profile. All users can access this page and search freely the database.
The search page displays the results of the user's research.
For each result, there is a row with a profile picture of the tattoo artist (on the left), an information section (on the centre) displaying the name of the artist and his/her biography, and a carousel (on the right) with three pictures of the artists’ tattoos. In this way, the user can understand at a glance if the tattoo artist might be of interest, or not. In case the user wants to know more about one tattoo artist, there is a link at the bottom of the central information section that leads directly to the artist page. A target attribute has been added to the link that leads to the artist's page: in this way the user doesn't lose the search results if clicks on a link.
This page displays all the artists in the database, in alphabetical order. The results are paginated 5 at a time.
That page displays a list of artists who share the same tattoo style.
The layout is the same as the artists' section on the home page.
This page displays all the information about a single artist. Here users can find the address, the contact area, information about the languages that the artist speaks, the style he/she has, and a gallery with up to six images showing some of the best work of art the artist made. A bottom at the end of the page takes the user back to the home page.
The register page has an image on the background of a bearded tattooed man that looks in the direction of the form. The form has two fields: username (that accept letters from the Latin alphabet and numbers) and password (with a length between 5 and 20 char). A small text under the section has a link to the login page for those users that already have an account.
This page repeats the same structure as the register page. If a user writes a wrong username or a wrong password, a message tells that there is some mistake.
This page is visible only after a user registered his/her profile and is not available from the menu. That happens because a user is allowed to have only one profile.
A form with all the fields needed is shown. Here the tattoo artist can add:
- name
- address
- city
- country
- profile picture (URL)
- phone number
- link to Facebook
- link to Instagram
- languages spoken
- tattoo style
- up to six images (URL)
A button lets the user save the profile.
This page takes the same layout of the “Add profile”, but instead of having empty fields, it shows the values that the user already added before. Clicking on the fields, the user can delete the content and type something new. At the bottom of the page, there are two buttons: one allows the user to save the changes, the other one allows to cancel the changes without saving.
This page allows the tattoo artist to delete his/her profile. However, to prevent unintentional deleting, I added an extra step. The page asks for confirmation: if the user really wants to delete the profile, then a button cancels the profile and takes the user back to the general home page. If the user changes his/her mind and wants to go back, a button guides him/her to the profile page.
This page shows the artist’s page with the information that he/she provided. A button allows the user to edit the profile (linking to the “Edit profile” page, and one button to delete it (linking as I mentioned before to the “Delete” page.
I decided to add also a 404 page which I think is always useful to have. In this way, if a user forces the URL to a page that doesn't exist, can still go on with the navigation without feeling frustrated.
- 📑 Strategy Plane
The website should help tattoo artists getting in touch with new customers. It should be a window for artists to show their abilities, and at the same time allows users to find their ideal tattoo.
When a person is looking for a new tattoo, he/she wants to find a person that is able to transform the idea into a permanent design. Not all tattoo artists do the same things, they are always specialized in some particular styles.
Moreover, it is always better when customers and artists speak the same language. They can communicate properly and share ideas, feelings, etc. We live in a world where immigration is a common thing, we have around people coming from all over the world and language is not always the same (or at the same level). This is something that I experienced directly, as an Italian living in Sweden. Also, we live in a society where it is easy to move from one city to another, but also from one country to another: it happens often that customers are willing to travel just to have the tattoo done by a specific person.
There are two big websites that have a great database for tattoo artists: Tattodoo and Inkstinct. They have very rich search options, but they are missing the language thing and they are too confusing. They don’t go straight to the point as my website wants to do.
According to an article that appeared on Medium, the average person nowadays that has a tattoo is a person with high education, with a good income, that lives in an urban area. The idea of the tattooed person as an “outsider” is no longer actual and I think that the design of the website should reflect this aspect. It should be intuitive, modern and with a strong personality. That also should convey an idea of credibility and reliability. The user (customer) can contact the tattoo artist in many ways: with a phone call, with an email, connecting through Facebook or Instagram.
Opportunity / Problem / Feature | Importance | Viability | |
---|---|---|---|
A | Register Functionality | 5 | 5 |
B | Login Functionality | 5 | 5 |
C | Add Profile Functionality | 5 | 4 |
D | Show Profile Functionality | 5 | 4 |
E | Edit Profile Functionality | 5 | 4 |
F | Search Functionality | 5 | 4 |
G | Contact Artist | 5 | 4 |
H | Contact Webmaster | 5 | 5 |
I | Paginate search results | 3 | 3 |
J | Logout Functionality | 5 | 5 |
K | Admin Functionality | 3 | 3 |
As shown on the table, I will leave at the end the admin functionality and pagination that will come out with a second release.
- 📑 Scope Plane
It is very important that the tattoo artist can have his/her own virtual window as a showcase of his/her job. To have the maximum focus on the artist it is of fundamental importance that every single artist has his/her own page with no distractions around. Links that allow connections between artists and customers are also of fundamental importance. To help the tattoo artist editing and updating the profile anytime, there must be a functionality to do it independently. Also, to keep the information on the website consistent for all the tattoo artists, it should not be allowed for a tattoo artist to add extra fields. In this way, everyone has equal instruments to reach and attract customers.
- 📑 Structure Plane
The sticky navbar allows users to navigate easily on the website without going back and forth wasting time, considering also that the website has many pages and without a sticky navbar the user can easily get lost.
The search bar, a fundamental feature, is placed on the navbar and is also repeated on the home page.
All the buttons have a glowing effect on hover, to add focus and attract the user. At the bottom of almost every page there is a button that allows the user to do something: for example, to edit the profile, delete it, or go back to the home page.
A smooth scrolling effect adds some extra class and modernity, obtained with ScrollReveal.js.
Validation on forms is adapted and customized from Bootstrap JavaScript validation. A feedback tells the user if the fields are correctly filled or not. This validation also prevents the form from being submitted with invalid fields.
The website is divided into two different structures: one is for all the users, one is only for registered users (in this case, only for tattoo artists).
The structure visible to all users includes the home page and the search functionality, which allows anyone to access tattoo artists' profiles.
The other structure level is dedicated only to tattoo artists. They can register with a username and add their profile, which will be visible to anyone. Tattoo artists can also edit their own profile or delete it directly without contacting an admin of the website.
- 📑 Skeleton Plane
The pattern design is consistent throughout all the pages.
You can view my prototypes by clicking on the following links:
Prototyping has been made with Adobe XD.
While coding I changed a little bit the layout because new ideas came to my mind, so the final result is a little different from the prototype.
- 📑 Surface Plane
The absence of colours is dominant on this website.
I decided to make black and grey the main characters of the design because I wanted to give the feeling of ink and tattoos on the screen.
-
As a user interested in tattoos, I want to find easily information about tattoo artists in my area.
-
As a user interested in tattoos, I want to find easily information about tattoo artists specializing in a particular tattoo style.
-
As a user interested in tattoos, I want to find easily information about tattoo artists that speak my language.
-
As a user interested in tattoos, I want to browse the database just to have a look at tattoos pictures.
-
As a tattoo artist, I want to add my profile to get new customers.
-
As a tattoo artist, I want to have a look at my competitors' works.
-
As a tattoo artist, I want to contact the owner of the website.
-
As a user interested in tattoos, I want to contact a specific tattoo artist.
-
As a user interested in tattoos, I want to have a look at the last added tattoo artists.
-
As a tattoo artist, I want to edit my profile.
-
As a tattoo artist, I want to delete my profile.
The website has a simple layout based on Bootstrap cards and Bootstrap forms. I customized the design to make it more tailored to my needs and my vision.
The colour palette is very basic, but the shades have been accurately chosen according to different criteria:
- overall harmony of shades;
- no eyestrain;
- evocation of the tattoo and ink concept.
Colour Palette created with Coolors
To make the design more tridimensional and deeper, I used a shadow box on images.
Buttons on hover have a white glowing effect that looks like they are coming out from the screen.
I changed the shades of green and red that came with Bootstrap JavaScript Validation, because the visual effect on the grey background was disturbing.
I darkened them with:
For this project, I decided to pair Montserrat with Shadows Into Light, two Google Fonts that I think are great for this website.
Montserrat, which is modern and clear, has been used for almost any text on the website.
Shadows Into Light, which is a cursive font, has been used for flashed messages and for the tattoo artists' names. This font is very appropriate with the style of tattoos, and I chose it to remind of a handwritten signature.
Images have been taken mostly from other tattoo websites (especially Tattodoo) or artists' real profiles.
In the home page, the hero image comes from:
The four images used in the "style" cards are taken from:
- https://tattoodo-mobile-app.imgix.net/images/posts/20200115_SQFjyld99NGCSOy.jpg
- https://tattoo-ideas.com/wp-content/uploads/2020/11/samurai-medusa.jpg
- https://www.wykop.pl/cdn/c3201142/comment_1587243214rxE3dEaOjgx1LCG4t1HV4Y.jpg
- https://inkppl.com/assets/php/files/082018/38913976_702864310048924_7109833061583290368_n.jpg
In the register/login template, the background image comes from:
The image used in the page 404 comes from this website:
All the other images sources are listed here: Links
Wireframes have been made with Adobe XD.
You can take a look at them here:
The website is responsive on all screens.
The sticky navigation bar collapses into a toggler on tablets and smaller screens.
Despite having many pages, the content is easily navigable thanks to the links provided in the navigation bar.
The search box also helps the user to find easily a specific content. I decided to add the target attribute to search results. In this way, when a user clicks on a specific result to access more details, a new tab is opened and the user doesn't lose the results page.
Images guide the user through the navigation, and a carousel on artists' previews give the chance to access more content without clicking or moving the mouse.
Globally, the website is easily readable, links are easy to understand.
I would like to add an admin dashboard, where an admin can perform various tasks and keep track of users' activity.
I would like also to connect the website with a payment system, to make the login function accessible only to users that have paid a fee.
Other things to implement in the future, are better accessibility and pagination also on search results.
Another thing will be adding custom meta content for description for each page, using Jinja language.
A limit of the search functionality right now is that users can search for a single word. If a user wants to add extra filters (for instance, search for an artist who has a specific style in a specific city), right now it is not possible. So this is something that needs to be implemented.
- HTML5
- CSS3
- JavaScript
- Python 3.8.6
- Bootstrap 4.5.0.
- Google Fonts
- Font Awesome 4.7.0.
- Git
- GitHub
- GitPod
- Heroku
- Randomkeygen
- Chrome DevTools
- Adobe XD
- Favicon.ico
- Html Validator
- CSS Validator
- JSHint
- Flask 1.1.2
- Flask-PyMongo 2.3.0
- pymongo 3.11.1
- Werkzeug 1.0.1
- flask-paginate 0.7.1
- MongoDB
- Autoprefixer CSS 9.7.6
- Am I responsive?
- ScrollReveal.js
- Techsini mockup generator
All the links are working.
Only an anchor link was not working on Firefox (Check "Bugs" section for details).
The form validation is also working, it prevents the user to submit the form with invalid fields and helps to understand which ones need to be corrected.
I realized that the validation JavaScript code should be reinforced with regEx, since it allows a user to type for instance "123" in the phone number field, which is obviously not correct. It is made by numbers, but there are no phone numbers in the world with only 3 digits. This is something I will implement in the future.
The register functionality works properly, it displays the "app-profile" page which is reachable only after a new user registers.
After adding the pieces of information, I have tried to edit successfully the profile: all the previous information added by the user are visible inside the input areas, and it is possible to edit them.
And finally I tried to delete the profile successfully:
At this point, the user is still logged in because the username hasn't been deleted, only the profile.
It is still possible for the user to show again his/her profile simply clicking on "My Profile":
An empty profile is displayed, and the user needs only to edit it again with new information.
If a user wants to delete also his/her username, he/she should contact the website owner. This is easy to do, since in the footer there is a link to contact the webmaster.
I then tested the "search" functionality by searching the word "manga":
And as expected, I can see the results displayed on the page:
I have tried to search for a word that does not exist in the database, and it correctly returns the right message that informs the user that no results have been found:
I went on testing the search method with many other words that should refer to: "name", "language", "city", "country - or state-", "style". Everything works as expected.
I checked if the navbar highlights the right name of the page, and also this functionality works fine.
Html and CSS have been validated with HTML Validator and CSS Validator.
JavaScript has been checked with JSHint
I checked if Python is Pep8 compliant on Pep8 Online and the result was positive for app.py:
I checked also env.py, and I got this result:
However, this is not a problem since the env.py file is ignored.
- As a user interested in tattoos, I want to find easily information about tattoo artists in my area.
When I enter the website, I scroll down the page and I see the images and the titles of the sections. I see that there is a search box, but I take a look first at the navigation bar in case I find something extra. On the navigation bar, I find another search box, so I try to type the city that I am looking for. If the results are positive, tattoo artists working in my city are displayed on the page. From here, I can click on their profile so I can find out more information and eventually contact them. If there are no results, I get a message on the screen that informs me to try with another word.
- As a user interested in tattoos, I want to find easily information about tattoo artists specializing in a particular tattoo style.
When I enter the website, I scroll down the page and I see the images and the titles of the sections. I find immediately a section displaying four cards with four different tattoo styles. The title tells me that this is just a selection of styles, so I understand that if the style I am looking for is not displayed here, I have to search somewhere else. Just below this section, I find a search box, which I can see also on the navigation bar. I try to type a specific style I am looking for, and if the results are positive, all the artists specialized in my favourite style are displayed on the page. From here, I can click on their profile so I can find out more information and eventually contact them. If there are no results, I get a message on the screen that informs me to try with another word. If instead one of the cards in the home page is exactly what I was looking for, I can just click on the button “See All” and I get the results on the page without typing in the search box. However, I might get results about artists that live too far away from me. I am not interested in them if I am not willing to travel, and I feel I need to waste time to understand if someone of them is close to me.
- As a user interested in tattoos, I want to find easily information about tattoo artists that speak my language.
When I enter the website, I scroll down the page and I see the images and the titles of the sections. I see that there is a search box, but I take a look first at the navigation bar in case I find something extra. On the navigation bar, I find another search box, so I try to type the language that I am looking for. If the results are positive, tattoo artists speaking my language are displayed on the page. From here, I can click on their profile so I can find out more information and eventually contact them. If there are no results, I get a message on the screen that informs me to try with another word.
- As a user interested in tattoos, I want to browse the database just to have a look at tattoos pictures.
I enter the website and I start by scrolling down the page. I might be attracted by one of the images I see here, so I click on its link to see more content related. If nothing impresses me, I take a look at the navigation bar. I find a page called “All artist” and I click on it. Here I have plenty of images.
- As a tattoo artist, I want to add my profile to get new customers.
I enter the website and first of all I have a look at the navigation bar. I see the words “Are you an artist?”, so I understand I have to click on it. Here, I find two pages: “register” and “login”. Since I am new to the website, I click on “register” and I am redirected to the proper registration page. The fields are easy to fill and immediately after saving I am redirected to the page “Add profile”. Here I find a form that asks me to fill out all the information needed, from name to gallery images. In this form, some fields have an asterisk. I guess it means they are mandatory, but to be sure I scroll the page up and down to find more details. At the beginning of the page, just before the form, a message informs me that fields with an asterisk are required. I then proceed to add the details and I save them. If all the information were correctly added, the form is submitted and as confirmation, I am redirected to “My Profile” page. Is something is missing or wrong, I get a feedback under each field telling me if the field is ok or not. The green feedback says “Great!”, so I understand that this information is ok. If not, the red message helps me to understand which might be the problem.
- As a tattoo artist, I want to have a look at my competitors' works.
I open the website and I scroll down the page. I take a look at the last 5 artists added, then I scroll to the top and I click on the cards with different styles. I also use the search box to search for something in particular, or I visit the page “All Artists”.
- As a tattoo artist, I want to contact the owner of the website.
I am on the home page. I can’t find any particular information on the navbar, so I scroll down the page. At the bottom, on the footer, I find a link that allows me to send an email to the website owner.
- As a user interested in tattoos, I want to contact a specific tattoo artist.
I type in the search box the name of the artist that I want to contact. If the artist is in the database, I can see his profile page. From here, I can click on one of the four links that I can use to contact the artist: phone, email, Facebook, Instagram.
- As a user interested in tattoos, I want to have a look at the last added tattoo artists.
I scroll down the home page until I reach the section “last added”.
- As a tattoo artist, I want to edit my profile.
I log in and I am redirected to “My Profile” page. Here, at the bottom, I have a button that takes me to the “Edit” page. I can also access this page from the navigation bar. A form with prefilled fields displays my information previously added. I click on the fields that I want to edit and then I save the page.
- As a tattoo artist, I want to delete my profile.
I log in and I am redirected to “My Profile” page. Here, at the bottom, I have a button that takes me to the “Delete” page. A message asks me if I am really sure that I want to delete my profile. If I changed my mind, I can click on the “take me back” button, otherwise, I go on clicking on the “delete” button.
I tested the website not only on Am I responsive?, but also on different mobile screens that I have at home.
The functionality is working everywhere, but there are a few things that need to be fixed on smaller screens. For instance, on a Galaxy Fold which has a width of 280px, the profile image of the artist is not displayed as it should be:
After fixing the issues with the resolutions on screens smaller than 320px, this is how it looks:
On other mobile devices with bigger width, everything is displayed as expected.
Here an example of an artist page on as it renders on iPhone 6/7/8/Plus:
I tested the website on multiple browsers:
- Chrome
- Edge
- Firefox
- Safari
- Opera
To check performance testing I used Google Test Mobile Friendly, Google Lighthouse and Google PageSpeed Insights for the home page and all-artists page.
The website performs well both on mobile screens and desktop; however, accessibility needs to be improved.
To improve speed I added lazy loading by adding loading="lazy" to all my images.
This is the result of the mobile version with Lighthouse:
And this is the result of the desktop version:
The website is mobile-friendly, as tested with Google Test Mobile Friendly:
Google PageSpeed Insight showed that the website needs to improve the speed on mobile devices. This is due especially to images that are loaded from external servers. This might also be a problem for those users that have to pay for extra data.
I decided to add pagination to the "All artists" page. In this way it is simpler for a user to navigate through the results, without having to keep scrolling down the page. I tested again the page and the results are now slighlty different with Lightroom.
This is the new result for the mobile version:
And this is the new result for the desktop version:
Best practices improved very much, but SEO needs now to be improved. This happened because pagination links right now cannot be crawled.
After testing the website on Firefox, I realized that there was a bug in the anchor link inside the footer.
Basically, the link "to the top" doesn't work (but it works on Chrome, Edge, Opera).
I tried to fix it by switching from:
<p class="card-text"><a href="#navigation" class="footer-text link">To the top <i
class="fas fa-angle-double-up"></i></a></p>
to:
<a href="#navigation" class="footer-text link"><p class="card-text">To the top <i
class="fas fa-angle-double-up"></i></p></a>
but it still doesn't work.
I decided to write a JavaScript code that runs when a user clicks the "To the top" link. First, the code detects if the user is using Firefox; then, it scrolls up to the top of the page.
Then, a better validation needs to be implemented, including a better styling with validation feedback.
For instance, this is what happens:
The database used for this project has two collections: user and artist.
The "user" collection stores the username and the password of each registered user. The password is hashed through Werkzeug.
The "artist" collection stores all the information that the user adds about himself/herself. Those data are used to display artists' profiles, and are connected through the field "created_by" to the user collection. In this way, when a user logs in with his/her credentials, can edit and delete his/her own information.
The website is deployed connecting Heroku with GitHub.
This was possible through those steps:
-
First of all, Heroku needs some files that you have to create in your IDE (I used GitPod). From your command line, type
pip3 freeze --local > requirements.txt
This will create a file with all the environment's current package list.
-
Second, always from your command line, type
echo web: python app.py > Procfile
Make sure to write "Procfile" with a capital letter. This file tells Heroku which commands are run by your application.
-
Now open Heroku and create your account.
-
After you logged in with your account, you need to click on the button "New" and select "create new app":
-
Now, choose a name for your app (use only lowercases and a dash or a minus if you need to separate words), select a region and click on the button "Create app".
-
From your dashboard, now click on the tab "Deploy". Click on the "GitHub" button:
Your GitHub profile will appear with a search box on the right. It is intended of course that you already have a repo on GitHub. Now type the name of your repo in this search box and click on "search".
Now you can connect Heroku with GitHub simply by clicking on the "connect" button.
-
Great!
Now you need to add some extra information.
Click on the "settings" tab and scroll down to the "Config Vars" section. Click on "Reveal Config Vars".
-
Now you need to add some parameters.
From your env.py file, just copy and paste the information.
As default, we set the value for IP to 0.0.0.0 and for PORT to 5000.
-
Next step: we have to push the files to the repo.
To do so, from your command line type:
git add requirements.txt
git commit -m "Add requirements.txt"
git add Procfile
git commit -m "Add Procfile"
git push
-
Now, go back to Heroku and go back to the "Deploy" area.
Click on "Enable Automatic Deploys" and after that "Deploy branch".
-
After a couple of minutes you will get the message that "your app was successfully deployed".
-
Just click on "view" and you will see your website.
If you need to work on this code on your own, follow these steps:
- Log in to GitHub;
- Find the repository you are looking for;
- On the top-right of the page you will find a button with the name "Fork";
- Click on it and it will automatically fork the code to your GitHub.
To make a local clone of the site, just follow these stepsg:
- Log in to your GitHub;
- Under your repository section, select the repo that you need;
- You will find a green button with the name "Code". Click on it;
- On the dropdown selection, you will find a link to clone the code with HHTPS;
- Now open Git bash;
- Open the directory where you want to work on the cloned code;
- Type git clone followed by the link you have previously copied.
- Custom validation JavaScript code from Bootstrap
- Scrolling animation from ScrollReveal.js
- Changed placeholder style adapting the code from W3School
- Pagination code adapted from Stephen Moody's -Code Institute- repo
Check the Imagery section.
Here you can find a detailed list of all the links used to get images and text for each artist profile: Links
I want to thank my beloved husband for giving me the subject idea for this project. He is a tattoo lover and was always looking for an app like this one.
Thanks also to Code Institute, both Tutor and Mentor Support (Spencer Barriball), for supporting and helping through the project. I faced a few issues with GitPod and the support has been amazing.