- Background & Motivation
- The Task (Frontend)
- The Support (Backend)
- Constraints & Assumptions
- Teamwork
- Marking Criteria
- Originality of Work
- Submission
- Late Submission Policy
31/10/23 - Removed leftovers from previous Assignment (removed "custom.js" and other minor details) 04/11/23 - Clarified what the dashboard is for 2.2.2.
In October 2023, following the immense financial success of your messaging platform Slackr, you were invited to several tech talks and travelled all over the country. In your travels, you've had to endure several uncomfortable hotel stays. During these stays you've been struck with another brilliant startup idea for a person to person property renting service AirBrB.
You've contacted your developer friends and settled on functionality, feature set and a RESTful specification/interface for AirBrB. You've decided to outsource your back-end to another company and took on the task of building the front-end (optionally with another one of your friends). You wrote a list of requirements and functionalities your frontend should adhere to (described in section 2). You also decided to complete this application in ReactJS, a declarative framework for building single page applications. This front-end will interact with a Restful API that the team you've outsourced to are producing, based on the pre-defined interface.
Because your MVP is only going to be demonstrated once, your team considers it imperative that your front-end is thoroughly tested.
To satisfy modern tastes and expectations you have also decided to ensure that the UI, UX and Accessibility standards are very high.
This assignment is the process you building the front-end for that MVP to the standards described. This assignment is closely modelled off the popular property renting platfrom Airbnb. If you're not familiar with the service, we would recommend spending the time to try it out so that you can get a feel for how this application may function.
You will need to watch at least the following lectures before starting (it will help you get started):
- Javascript Ecosystem
- Node Package Manager
- ReactJS Introduction
- ReactJS Global CSS Usage
- ReactJS Lifecycle
- ReactJS useState hook
- ReactJS useEffect hook
- Working with multiple files
- Components & Props
- Linting
You will need to watch at least the following lectures to finish the assessment completely:
You are to build a frontend for a provided backend. This frontend shall be build with ReactJS. It shall be a single page application that does not require a refresh for state updates.
Features need to be implemented (described below) in order for your ReactJS app to meet the requirements of the task, and to operate with the backend described in 3.2.
The requirements describe a series of screens. Screens can be popups/modals, or entire pages. The use of that language is so that you can choose how you want it to be displayed. A screen is essentially a certain state of your web-based application.
Please note: This assignment, unlike assignment 2, has a lot of functionality available whilst not logged in. Logging in just adds extra functionality. If you're unsure what we mean by this behaviour, you can play around with the Airbnb website for comparison.
This focuses on the basic user interface to register and log in to the site. Login and registration are required to gain access to making bookings as a guest, leave reviews and to manage your own listings as a host.
- A unique route must exist for this screen
- User must be able to enter their
email
andpassword
. - If the form submission fails, a reasonable error message is shown
- A button must exist to allow submission of form
- A unique route must exist for this screen
- User must be able to enter their
email
andpassword
andname
- A confirm
password
field should exist where user re-enters their password. - If the two passwords don't match, the user should receive an error popup before submission.
- If the form submission fails, a reasonable error message is shown
- A button must exist to allow submission of form
- A logout button, when clicked, returns you to the landing screen whilst being no longer logged in.
- On all screens, for a user who is logged in / authorised:
- The logout button exists somewhere
- A button exists that will take the user to the screen to view their hosted listings.
- A button exists that will take the user to the screen to view all listings.
For logged in users, they are able to create their own listings (as a host) that will become visible to all other users who have the option of booking it.
- A unique route must exist for this screen
- A screen of all of YOUR listings (that you created) is displayed, where each listing shows the:
- Title
- Property Type
- Number of beds (not bedrooms)
- Number of bathrooms
- Thumbnail of the listing
- SVG rating of the listing (based on user ratings)
- Number of total reviews
- Price (per night)
- Each listing should have a clickable element relating to it that takes you to the screen to edit that particular listing (
2.2.3
). - A button exists on this screen that allows you to delete a particular listing.
- On the hosted listing screen (
2.2.1
) a button should exist that allows you to create a new listing. When you click on it, you are taken to another screen that requires you to provide the following details:- Listing Title
- Listing Address
- Listing Price (per night)
- Listing Thumbnail
- Property Type
- Number of bathrooms on the property
- Property bedrooms (e.g. each bedroom could include number of beds and their type)
- Property amenities
- Using a button, a new listing on the server is created and visibly added to the dashboard (the Hosted Listings Screen) once all of the required fields have been filled out correctly.
- A unique route must exist for this screen that is parameterised on the listing ID.
- The user should be able to edit the following:
- Title
- Address
- Thumbnail
- Price (per night)
- Type
- Number of bathrooms
- Bedrooms (incorporate editing of beds as part of bedrooms)
- Amenities
- List of property images
- Updates can auto-save, or a save button can exist that saves the updates and returns you to the hosted listings screen.
- For a listing to "go live" means that the listing becomes visible to other AirBrB users on the screen described in
2.4
. - On the hosted listings screen described in
2.2.1
, add the ability to make an individual listing "go live".- A listing must have at least one availability date range (e.g. a listing could be available between 1st and 3rd of November and then between the 5th and 6th of November).
- The way you define the availability ranges is entirely up to you. For example, you could use the following schemas:
//Example 1:
availability: [{ start: date1, end: date2 }, { start: date3, end: date4 }, ...];
//Example 2:
availability: [date1, date2, date3, date4, ...];
- (Note: If the listing has more than 1 availability range, aggregate them on the frontend and submit them all to the backend in one go when publishing the listing).
When the app loads, regardless of whether a user is logged in or not, they can access the landing screen. The landing screen displays a number of listings that you as a guest may be able to book (on another screen). We recommend you create some listings (2.2
) with one user account, and then create a second user account to build/test 2.3
so that you can view their listing as a potential booking option.
- A unique route must exist for this screen.
- This is the default screen that is loaded when a user accesses the root URL.
- This screen displays a list of all published listings (rows or thumbnails). The information displayed in each listing is:
- Title
- Thumbnail of property (or video if advanced)
- Number of total reviews
- (any more information you want, though that's optional).
- In terms of ordering of displayed published listings:
- Listings that involve bookings made by the customer with status
accepted
orpending
should appear first in the list (if the user is logged in). - All remaining listings should be displayed in alphabetical order of title.
- Listings that involve bookings made by the customer with status
- On this listings screen, a search section must exist for the user to filter via search parameters. You are only required to be able to search by one of the parameters described below at a time.
- The search section will consists of an input text box:
- The input text box will take in a search string, and will search title and city location properties of listings, and only display those that match.
- You are expected to do this matching on frontend (after loading all the results from the backend).
- You are only required to do case insensitive substring matching (of each word in the search field), nothing more complicated.
- Other form inputs should also exist that allow the user to search by:
- Number of bedrooms (a minimum and maximum number of bedrooms, expressed either via text fields or a slider)
- Date range (two date fields) - only display bookings that are available for the entire date range as inputted by the user.
- Price (a minimum and maximum price, expressed either via text fields or a slider)
- Review ratings:
- Sort results from highest to lowest review rating or from lowest to highest review rating
- If there is more than one listing with the same rating, their order does not matter
- The search section must have an associated search button that will action the search to reload the results given the new filters.
- A unique route must exist for this screen that is parameterised on the Listing ID
- For
2.3
, when a listing is clicked on, this screen should appear and display information about a specific listing. - On this screen the user is given the listing they have decided to view in 2.4.1. This consists of:
- Title
- Address (displayed as a string, e.g. 1/101 Kensington Street, Kensington, NSW)
- Amenities
- Price:
- If the user used a date range for search in
2.3.2
- display price per stay - If the user did not use a date range for search in
2.3.2
- display price per night
- If the user used a date range for search in
- All images of the property including the listing thumbnail (they don't have to be visible all at once)
- Type
- Reviews
- Review rating
- Number of bedrooms
- Number of beds
- Number of bathrooms
- On this screen if the user is logged in and they have made booking for this listing, they should be able to see the status of their booking (see
2.4.2
). - (Note: if the user has made more than 1 booking for a listing, display the status of all the bookings)
- On the screen described in
2.4.1
, a logged in user should be able to make a booking for a given listing they are viewing between the dates they are after. The user enters two dates (this includes day, month and year), and assume the dates describe a valid booking, a button allows for the confirmation of the booking. - A user can make an unlimited number of bookings per listing even on overlapping date ranges and even if other users have already booked the property for those dates. It is up to the host to check if they have double booked their listing and accept/deny the bookings accordingly.
- A booking's length (in days) is defined based on how many nights a user spends at the listed property (this is how bookings are defined on all modern accommodation platforms). For example, a booking from the 15th to the 17th of November consists of 2 days in length - 15th to the 16th and 16th to the 17th. As this is a late addition to the specification, we will not be strictly enforcing how you chose to calculate a booking's length. So for the case described here, it could also be expressed as a 3 day long booking (15th, 16th and 17th as the 3 days).
- Once a booking is made, the user receives some kind of temporary confirmation on screen.
- A logged in user should be able to leave a review for listings they've booked that will immidiately appear on the listing screen after it's been posted by the user. The review will consist of a score (number) and a comment (text). You can leave an unlimited number of reviews per listing.
- Please note: Normally you'd prohibit reviews until after a booking visit is complete, but in this case for simplicity we allow reviews to be left as soon as a booking's status becomes
accepted
. - If the user has made more than 1 booking for a given listing, you can use any of their
bookingid
s for the purpose of leaving a review. Just as long as the booking has statusaccepted
.
- On the hosted listings screen described in
2.2.1
, add the ability to remove a live listing from being visible to other users. - Once un-published, those who had made bookings for a removed listing will no longer be able to view it on their landing screen
- A unique route must exist for this screen that is parameterised on the listing ID
- This screen should be accessed via a button or link on the hosted listings screen
2.2.1
. - On this screen, a list of booking requests are provided for the listing they are viewing. For each booking request, the host is able to accept/deny it.
- The screen should also display the following information about a listing:
- How long has the listing been up online
- The booking request history for this listing consisting of all booking requests for this listing and their status (accepted/denied)
- How many days this year has the listing been booked for
- How much profit has this listing made the owner this year
- (Note: When counting the days and profits, inlcude all the bookings, past or future, that have been accepted for this year)
- On hover of star rating a tool tip appears which displays the break down of how many people rated the booking (both in percentage terms and absolute terms) within each star category. (e.g. see Amazon product rating for reference)
- If you click on a particular star rating, another screen should appear (that can be closed) that shows all of the individual reviews left for that rating.
- On the screen described in 2.2.1, a graph of how much profit the user has made from all their listings for the past month must be displayed. The X axis should be "how many days ago" (0-30), and the Y axis should be the $$ made on that particular day (sum of income from all listings).
- For
2.2.1
, when a new listing is created, the user can optionally upload a .json file containing the full data for a listing. The data structure is validated on the frontend before being passed to the backend normally. - If you implement this feature, you must attach an example .json into your repo in the project folder. This file must have name
2.6.json
. This is so we can actually test that it works while marking.
- For any given listing, making it possible to use a Playable YouTube video as the listing thumbnail. This youtube video URL becomes a field in the create/edit hosted listing screen.
- Linting must be run from inside the
frontend
folder by runningnpm run lint
.
As part of this assignment you are required to write some tests for your components (component testing), and for your application as a whole (ui testing).
For component testing, you must:
- Write tests for different components (3 if solo, 6 if working in a pair)
- For each of the components, they mustn't have more than 50% similarity (e.g. you can't test a "Card" component and a "BigCard" component, that are virtually the same)
- Ensure your tests have excellent coverage (look at all different use cases and edge cases)
- Ensure your tests have excellent clarity (well commented and code isn't overly complex)
- Ensure your tests are designed well (logical ordering of tests, avoid any tests that aren't necessary or don't add any meaningful value)
- (We encourage you to only use shallow component rendering)
You can use methods discussed in lectures for component testing, or you can use cypress
.
For ui testing, you must:
- Write a test for the "happy path" of an admin that is described as:
- Registers successfully 2. Creates a new listing successfully 3. Updates the thumbnail and title of the listing successfully 4. Publish a listing successfully 5. Unpublish a listing successfully 6. Make a booking successfully 7. Logs out of the application successfully 8. Logs back into the application successfully
- (If working in a pair) also required to write a test for another path through the program, describing the steps and the rationale behind this choice in
TESTING.md
- (If working solo) include a short rationale of the testing you have undertaken within
TESTING.md
- Find a simple primitive component you've written, and if you don't have one, write one. This could include a common button you use, or a popup, or a box, or an input. Often examples of these are just MUI or other library components you might have wrapped slightly and includes some props you've passed in
- Simply write some unit tests that check that for a given prop input, the component behaves in a certain way (e.g. action or visual display), etc etc
- E.G. Creating a
MyButton
that wraps a MUIButton
. - E.G. A simple example is the list of bookings. It takes in booking informed we've defined and renders a bunch of MUI ListItems, Checkboxes, TextFields and IconButtons
- Your app is going to be a set of pages, and those pages are made up of primitive components. But if you don't have layers of components between that it means your code is not well modularised. Another example could be if we said to you - no component should be longer than 50 lines of code. You'd probably go refactor to group common sets of primitives together into a new component.
- For cypress, consider adding
cy.wait(1000)
if necessary to add slight pauses in your tests if you find that the page is rendering slower than cypress is trying to test. - If you're having issues using Cypress on WSL2, try following this guide.
- You can welcome to use
enzyme
for testing if you prefer - as long as everything works by runningnpm run test
. - One topic that has helped students is mocking fetch calls with jest.
- The tutor will run an empty (reset) backend when running
npm run test
whilst marking. - Some students will run into
enzyme adapter
compatibility issues. If you run into these issues you can either: - Use this unofficial React 17 adapter: https://www.npmjs.com/package/@wojtekmaj/enzyme-adapter-react-17; or
- Downgrade react and react-dom to 16, though this could break other things depending on what other dependencies you're using.
Tests must be run from inside the frontend
folder by running npm run test
.
You are welcome to modify the npm run test
command by updating the test
script inside frontend/package.json
. For example, if you would like to run standard react testing alongside cypress UI tests you can use react-scripts test —watchAll=false && npm run cypress open
and if you've used cypress for both component and happy path test, then you can replace that line with cypress open
.
- The port you can use to
fetch
data from the backend is defined infrontend/src/config.json
- This article may be useful to some students
- For users of typescript, follow this guide
- For images, you can just pass in base64 encoded images
- For certain requests you may want to "poll" the backend, i.e. have the friend end repeatedly make an API call every 1 second to check for updates.
Navigate to the frontend
folder and run npm install
to install all of the dependencies necessary to run the ReactJS app. Then run npm start
to start the ReactJS app.
You will not need to do any meannigful work in the backend. However, some properties that the backend takes in are defined as blank objects. These are objects that can be defined by you, as the backend will simply store your object on some routes and then return it to you on other routes (i.e. the backend doesn't need to understand the schema of some objects you pass it).
For example, one of the ways you could choose to define the address
object could be using the following schema:
address: {
street: '1 Kensington Street'
city: 'Kensington'
state: 'NSW'
postcode: '2032'
country: 'Australia'
}
There are several objects that will need to be defined by you using the schema you think is best for the optimal solution. Within the listing
object, you will need to define address
and metadata
completely. You will also need to define the schema for a single review and availability range to be added to the reviews
and availability
arrays respectively. For the booking
object, you will need to define the schema of the dataRange
that the booking has been made for.
This approach we've taken is actually designed to make the assignment easier, as it gives you control without having to worry about backend architecture.
You are prohibited from modifying the backend. No work needs to be done on the backend. It's provided to you simply to power your frontend.
The backend server exists in your individual repository. After you clone this repo, you must run npm install
in backend
directory once.
To run the backend server, simply run npm start
in the backend
directory. This will start the backend.
To view the API interface for the backend you can navigate to the base URL of the backend (e.g. http://localhost:5005
). This will list all of the HTTP routes that you can interact with.
Your backend is persistent in terms of data storage. That means the data will remain even after your express server process stops running. If you want to reset the data in the backend to the original starting state, you can run npm run reset
in the backend directory. If you want to make a copy of the backend data (e.g. for a backup) then simply copy database.json
. If you want to start with an empty database, you can run npm run clear
in the backend directory.
Once the backend has started, you can view the API documentation by navigating to http://localhost:[port]
in a web browser.
The port that the backend runs on (and that the frontend can use) is specified in frontend/src/config.js
. You can change the port in this file. This file exists so that your frontend knows what port to use when talking to the backend.
Please note: You CAN modify the backend for bonus marks. But you cannot modify the backend to make the core requirements of the assignment easier (e.g. by adding routes to make the other feature building easier). You are required to seek approval bonus backend changes via the forum. You are also proibhited from writing your own backend outright.
- You must implement this assignment in ReactJS. You cannot use other declarative frameworks, such as AngularJS, or VueJS.
- You must use ReactJS solutions wherever possible, and avoid doing any direct DOM manipulation unless completely unavoidable (check with course staff).
- You can use any CSS libraries that you would like, such as bootstrap or material-ui.
- You are able to use and install any library that is available to install via
npm install
.
- You should ensure that your programs have been tested on one of the following two browsers:
- Locally, Google Chrome (various operating systems) - make sure is latest version.
- On CSE machines.
- You may use small amounts (< 10 lines) of general purpose code (not specific to the assignment) obtained from a site such as Stack Overflow or other publically available resources. You should attribute clearly the source of this code in a comment with it. You can not otherwise use code written by another person.
- The specification is intentionally vague to allow you to build frontend components however you think are visually appropriate. Their size, positioning, colour, layout, is in virtually all cases completely up to you. We require some basic criteria, but it's mainly dictating elements and behaviour.
- Besides those described to avoid, you may use any other packages available on npm.
- The use of universal CSS is banned - you must use either CSS libraries (e.g. material-ui) or styled components.
This assignment may be completed in a team of two (pair). However, you are also welcome to complete it on your own, if you choose. The groups were organised and coordinated by the course coordinator separately.
If you formed a pair, you will be unable to leave your pair unless under extreme circumstances. You will be assessed together for the assignment.
If your contributions to the assignment are not approximately equal, then the teaching staff may make discretionary calls based on your gitlab history to award different marks to each student.
Please note: Your contributions will be measured based on the lines and commits contributed via gitlab. Please commit via your own machine or account. If you're in a pair, your contributions will not be considered yours if it is your partner who pushes the code to gitlab.
Please note: When special consideration is granted for one individual in a pair, it will only either 1) extend the deadline for the person who gets special consideration (it does not extend for the other individual); or 2) Result in a scale of the mark. To determine which outcome is appropriate, the person who receives special consideration is required to email the lecturer to notify them of how the work is split up prior to deadline.
Your assignment will be hand-marked by tutor(s) in the course according to the criteria below.
Criteria | Weighting | Description |
---|---|---|
Functionality of the Feature Set + Mobile Responsiveness | 60% |
|
Linted Code | 5% |
|
Code Style | 10% |
|
Testing | 15% |
|
UI/UX | 5% |
|
Accessibility | 5% |
|
(Bonus Marks) Extra Features | 5% |
|
The work you submit must be your own work. Submission of work partially or completely derived from any other person or jointly written with any other person is not permitted.
The penalties for such an offence may include negative marks, automatic failure of the course and possibly other academic discipline. Assignment submissions will be examined both automatically and manually for such submissions.
Relevant scholarship authorities will be informed if students holding scholarships are involved in an incident of plagiarism or other misconduct.
Do not provide or show your assignment work to any other person — apart from the teaching staff of COMP6080.
If you knowingly provide or show your assignment work to another person for any reason, and work derived from it is submitted, you may be penalized, even if the work was submitted without your knowledge or consent. This may apply even if your work is submitted by a third party unknown to you.
Every time you make commits or pushes on this repository, you are acknowledging that the work you submit is your own work (as described above).
Note you will not be penalized if your work has the potential to be taken without your consent or knowledge.
PLEASE NOTE: To ensure the originality of your work, we are requiring that you regularly commit your work to git throughout the weeks this assignment has been released. Regular and small commits (essentially at least once a day that you work on the assignment) are critical. Failures to commit regularly (or at minimum, failures to commit in small chunks) may results in either penalties of up to 20% of result in allegations of plagiarism.
This assignment is due Friday 17th November, 10pm.
To submit your assignment, you must you've pushed all of your code to your gitlab master branch. You can check if you've done this properly by seeing what code is on the gitlab site on your master branch.
We will collect the latest work on your master branch of gitlab at the time of submission.
It is your responsibiltiy to ensure that your code can run successfully when cloned fresh from Gitlab.
You can run a dryrun to sanity check your code runs basically by:
- Pushing your code to master on gitlab
- On a CSE terminal (vlab or lab machine), run
6080 ass4dryrun GROUP_NAME
where GROUP_NAME is the name of your group
No late submission are accepted.