Spotted is a social media application tailored for local communities. It is an online hub for staying updated on everything happening in your neighbourhood. Share what’s happening in your community, from the latest accidents to exciting developments, traffic updates to lost items.
The following screencast demonstrates the app's functionality.
spotted-app-screencast.mp4
The deployed application is available at https://jbraddockm.github.io/spotted-app/.
The app uses the following frameworks and libraries:
- jQuery - v3.7.1
- Tailwind CSS - v3.4.1
- Flowbite - v2.2.1
- Quill.js - v1.3.6
- Dayjs - v1.11.10
The app uses the following APIs
The application does not require any installation or build system. Please visit https://jbraddockm.github.io/spotted-app/ to test the app. See the Usage section below for further details.
- The app utilises an initialisation process to mimic a single-page application. It retrieves data from corresponding APIs, creates objects, persists them to localStorage, and finally renders the UI.
- Viewing posts is open to the public and does not require authentication. However, to compose a new post, the app requires authentication. Please use the following credentials to log in OR if you want to test with different users, you can get credential details from localStorage.
- Login Details
Email address
: demo@demo.comPassword
: 12345
- Login Details
- The app supports new user registration with each input being checked and validated. After the form is validated and user is saved, the app will authenticate the newly created user automatically.
- Getting current location information requires user permission. Follow the browser prompt to allow location access.
- The app implements user authentication, generates an authentication token, and saves it to sessionStorage
- The app implements user registration and supports form validation
- The app supports reverse GeoCoding with Open Street Map API. It gets the user's current coordinates and resolves it to an address.
- The app supports Quill.js text editor; therefore, it supports rich text format.
- The app uses Modal and Dropdown objects from the Flowbite library to control modals and the user dropdown menu programmatically
- The app has internal APIs/Methods to work with localStorage, implement authentication, and manage global variables. It borrows the naming convention from Spring Framework.
- Persistence Context:
- This API uses Map as the data structure
- This API allows persisting data to and retrieving from localStorage.
- This API generates sequential IDs while creating objects.
- Persistence context is extended by three objects:
- UserRepository: The object provides methods to work with the User object:
saveUser()
: Saves the given user object.Returns
user.createUser()
: Assigns a sequential ID to the user object and saves it to localStorage.Returns
user.findAll()
: Returns an array of all users.findUserById()
: Finds and returns the user with the given id. Returnsfalse
if the user does not exist.findUserByUsername()
: Finds and returns the user with the given username. Returnsfalse
if the user does not exist.findUserByEmail()
: Finds and returns the user with the given email address. Returnsfalse
if the user does not exist.getAllPostsByUserId()
Returns all posts by the user with the given id. Returnfalse
if no post is available.
- PostRepository: The object provides methods to work with the Post object:
savePost()
: Saves the given post object.Returns
post.createPost()
: Assigns a sequential id to the post object and saves it to localStorage.Returns
post.findAll()
: Returns an array of all posts.findPostById()
: Finds and returns the post with the given id. Returnsfalse
if the post does not exist.
- LocationRepository: The object provides methods to work with the Location object:
saveLocation()
: Saves the given location object.Returns
location.findAll()
: Returns an array of all locations.findLocationById()
: Finds and returns the location with the given id. Returnsfalse
if location does not exist.getLocationByPostId
: Finds the location for the given id. Returnsfalse
if location does not exist.
- UserRepository: The object provides methods to work with the User object:
- Security Context
- This API implements user authentication, generates a token, and saves it to sessionStorage.
- The app supports user log-out, which clears the authentication token from sessionStorage.
- If localStorage is cleared while the authentication token remains in sessionStorage, the app clears sessionStorage and requires re-initialisation.
- This API has the following methods:
getAuthenticationToken()
: Retrieves authentication token from sessionStorage. Returnsfalse
ifjwt-token
key does not exist in sessionStorage.isAuthenticated()
: Checks if a given user is authenticated. Returnstrue
orfalse
.isCredentialsCorrect()
: Checks if a given user's credentials are correct. Returnstrue
orfalse
.authenticateUser()
: Authenticates user, generates a token, and saves it to sessionStorage. ReturnsauthToken
.logout()
: Logs out the authenticated user and clears sessionStorage. Returnsfalse
, if logout fails.
- Application Context
- This API contains global variables to keep them organised and manageable.
- This API has the following variables:
resolvedLocation
: Location variable that stores the user's current locationformValidationSuccess
: CSS styling for valid form inputsformValidationError
: CSS styling for invalid form inputsformInputDefaultStyle
: Default CSS styling for form inputs
- Persistence Context:
- The app currently does not support multi-user authentication
- The app currently does not support delete or edit operations
- The app currently does not support searching locations by name, which is also known as forward GeoCoding
- The app currently only searches through available posts in the DOM. It does not search in localStorage
Please refer to the LICENSE in the repo.