Gamified fitness app that transforms users' fitness progress into engaging RPG-style adventures!
Download and install the app HERE
Features | Setup | Technologies | Roadmap | Images | Contributors
-
Quest-Motivated Workouts
Users complete workouts by logging physical activities to gain progression on quests they embarked on.
-
Character Progression System
Users earn experience points to level up their character which gives them attribute points to spend on customizing how their character grows.
-
Dynamic Turn-Based Combat
As users progress through quests, they will have the opportunity to fight against monsters and bosses in a dynamic turn-based combat encounter.
-
Virtual Rewards
As users defeat monsters, they earn gold which can be used to purchase cooler and stronger items.
-
Social Connection
Connect with friends, see when they last did a workout and what quest they have embarked on this week.
Before you start, you will need to have the following tools installed on your PC/Laptop: Git, Node.js (20.x LTS). In addition, it is good to have an editor to work with the code such as VSCode. Moreover, ensure you have an Expo user account. You can make one at this page. You will need an account to run the app and will be prompted after trying to connect the app to the emulator or device. This will ease development as running the app in online mode (default) lets us automatically install the development environment (Expo Go) automatically in our android emulator (in the following steps).
-
Clone the repository:
git clone https://github.com/cse403-fitquest/fitquest cd fitquest
-
Install dependencies
npm install
-
Install eas-cli
npm install -g eas-cli
-
Run the app
npm start
This should display the following:
a. If the QR is not displayed, it is likely that you did not do the previous steps correctly. Ensure your node version is 20.x LTS, delete the
package-lock.json
file and/node_modules
folder generated in step 2, then do anothernpm install
andnpm start
. -
Setup device for development. Run on physical Android device using expo QR code or run on Android Emulator:
With an android device, the fastest way to use the app is to use your physical device through expo. Download and install Expo Go SDK 51 (https://expo.dev/go?sdkVersion=51&platform=android&device=true). When the app is running after completing step 3, scan the QR code given in the terminal in the Expo Go application. If it takes too long to load (over several minutes), you may need to reload the app by pressing "r" in the terminal as shown in the screenshot above. For more troubleshooting guide, refer to the Troubleshoot section below.
Our app currently does not support iPhones, so using Expo Go with your iPhone will likely result in a crash or unintended bug(s).
With other devices such as a laptop, you will need to work with an android emulator to run the app. Following the instructions from the link will take around 15 minutes to setup depending on whether you have installed android studio: https://docs.expo.dev/get-started/set-up-your-environment/?platform=android&device=simulated&mode=expo-go. Be sure to select "Android Emulator" for the first option and "Expo Go" for the second option.
-
Login with your Expo account. After connecting with either the emulator or your android device, you will be prompted to login with your Expo user account. Simply follow the steps given and you will be succesfully logged in and the app should be initialized. If it is taking too long, try closing the server and start again from step 3.
Skip this setup if you want to use data provided by us. You only need to do this setup if you wish to use your own set of data that the app can interface with.
-
Access firebase console (https://console.firebase.google.com/)
-
Create a new project
-
Within the project, register a new web app
-
Copy the configuration parameters provided by firebase and set it in firebaseConfig.ts. Example below
const firebaseConfig = { apiKey: "yourAPIKey", authDomain: "fitquest-xxxxx.firebaseapp.com", projectId: "fitquest-xxxxx", storageBucket: "fitquest-xxxxx.appspot.com", messagingSenderId: "XXXXXXXXXXXX", appId: "X:XXXXXXXXXXXX:web:XXXXXXXXXXXXXXXXXXXXXX", measurementId: "X-XXXXXXXXXX" };
-
You will need to add two new products in your firebase project, Authentication and Firestore. Let's configure both of them:
a. For the Authentication product, simply choose Authentication from the list of products (should also be the first option in recommended products).
b. Click Get Started
c. When prompted for a sign in method, simply click "Email/Password".
d. Enable Email/Password, and ensure that Email Link (passwordless sign-in) is disabled. You should be set for authentication now.
e. For Firestore, simply browse the list of products and click on "Cloud Firestore".
f. Once initialized, click on the "Create database" button.
g. You will now be asked a series of questions to setup the database.
h. For setting location, pick the location closest to you.
i. For rules, pick test mode.
j. Wait a couple seconds for the database to initialize
k. After the database has been created, head over to the "Rules" tab and past the following ruleset:
rules_version = '2'; service cloud.firestore { match /databases/{database}/documents { // Shop Item collection where each document corresponds to an item match /items/{itemId} { // Only allow read for authenticated users allow read: if request.auth != null; allow create: if request.auth != null; allow update: if request.auth != null; allow delete: if request.auth != null; } // Friends collection where each document corresponds to a user match /friends/{userId} { // Only authenticated users can read their own friends document allow read: if request.auth != null; // Only authenticated users can create their own initial friends document allow create: if request.auth != null && request.auth.uid == userId; // Allow updates only if the user is authenticated and is modifying their own document allow update: if request.auth != null && ( // Only allow updates for specified fields: 'pendingFriendRequests', 'sentFriendRequests', 'friends' request.resource.data.keys().hasAny(['pendingRequests', 'sentRequests', 'friends']) && // Ensure the update does not remove any other fields request.resource.data.size() == resource.data.size() || // Or, the update must be limited to the allowed fields only request.resource.data.keys().hasOnly(['pendingRequests', 'sentRequests', 'friends']) ); } // Users collection where each document corresponds to a user's profile match /users/{userId} { // Allow read access to user data for authenticated users allow read: if request.auth != null; // Allow create access to user data for authenticated users for initial user data allow create: if request.auth != null && request.resource.data.id == request.auth.uid; // Allow the user to update their consumables or equipment allow update: if request.auth != null // && request.auth.uid == userId && ( // request.resource.data.keys().hasAny(['equippedItems', 'items']) // ); // Prevent overwriting the entire user document // allow update: if request.resource.data.size() == resource.data.size(); } match /quests/{quest} { allow read, write: if request.auth != null; } } }
l. Your firebase project should now be set up properly. Move on to creating collections and documents inside the database.
-
You will need to create a "quests", "items", "users", and "friends" collection in firestore.
a. For the quests collection, you will need to fill in dummy data in the form:
// An map with key-value pairs like below. '1' is the id of the quest. '1': { name:'Hunt Red Minotaur', questDescription: '', milestones: [ 50, 100, 150, 200, 250, 300, 350, 400, 450, 500, 550, 600, 650, 700, 750, 800, 850, 900, 950, 1000 ], bossThreshold: 500, duration: 7 * 24 * 60 * 60 * 1000, normalMonsters: [ { name: 'Green Slime', maxHealth: 80, health: 80, power: 8, spriteId: 'slime_green', }, { name: 'Blue Slime', maxHealth: 85, health: 85, power: 9, spriteId: 'slime_blue', }, { name: 'Red Slime', maxHealth: 75, health: 75, power: 7, spriteId: 'slime_red', }, ], boss: { name: 'Red Minotaur', maxHealth: 200, health: 200, power: 15, speed: 15, spriteId: AnimatedSpriteID.MINOTAUR_RED, }, createdAt: '', expiredAt: '', },
b. For items collection, you will need to fill in data for items for users to purchase in the form:
{ id: "t1_dagger", name: "T1 Dagger", type: "WEAPON" description: "Small but mighty! Perfect for poking holes in your enemies' plans." spriteID: "t1_dagger" power: 5, speed: 10, health: 0, cost: 50, createdAt: November 9, 2024 at 6:37:29 PM UTC-8 }
c. Users collection can be left empty, as new documents will be created as new users registers an account.
d. Friends collection can also be left empty, as new documents and relations will be made as users add friends.
-
Start (or restart if already running) the server
npm start
-
Install the ESLint extention on VSCode.
-
Reload VSCode to see changes.
-
For ease of use, make sure to turn enable format-on-save on VSCode settings.
-
Tests for the app can be run automatically with the following command:
npm run test
or you can run it in watch mode which reruns the tests automatically when saving changes with the following command:
npm run test:dev
-
Locate app tests under the
tests
folder in the root directory. There are unit tests for three parts of the app: reusable frontend components, services that interface with the firebase backend, and utility functions used throughout the different modules of the app. -
To start creating a new test, find an existing file module (auth, user, etc) or create a new file for a non-existant module to test for.
-
Write tests in accordance to the existing testing convention. Look into sibling files or tests within the same file module you are testing to see how you should be writing your tests.
If you encounter any issues while using the app, follow these step to troubleshoot:
-
Hot Reload:
- While the server is running after running
npm start
, click on the terminal and click "R" to do a hot realod. This usually solves issues related to rendering and is a good first step to try to fix any issue related to the UI.
- While the server is running after running
-
Check the Console for Errors:
- Open the developer console in your browser or the terminal where you ran the app.
- Look for any error messages and try to understand what might be causing the issue.
-
Verify Dependencies:
- Ensure all dependencies are installed correctly by running:
npm install
- Check for any peer dependency warnings and resolve them.
- Ensure all dependencies are installed correctly by running:
-
Clear Cache:
- Sometimes, cached data can cause issues. Run the app after clearing the cache by running:
npx expo start -c
- Sometimes, cached data can cause issues. Run the app after clearing the cache by running:
-
Run Tests:
- Run the test suite to ensure all tests pass:
npm run test
- Run the test suite to ensure all tests pass:
-
Create a New Issue:
- If you cannot resolve the issue, create a new issue on the project's GitHub Issues Page with detailed information about the problem.
This app was built with the following technologies:
- Typescript
- React Native
- Tailwind
- Zustand
- Firebase
- Setup Firebase SDK
- Implement Sign Up
- Implement Sign In
- Implement Onboarding Wizard
- Survey
- Fitness level calculation
- Allocate attribute points
- Implement Profile
- Avatar
- Attributes & inventory
- Workouts per week graph
- Implement Workout
- Active workout session
- Workout templates
- Calculate exp gain from workout
- Workout history
- Implement Quest
- Active quest
- Available quests this week
- Implement Fight
- Speed turn system
- Player and enemy sprite animations
- Player turn actions
- Implement Shop
- Consumables and inventory items
- Confirmation on item purchase
- Implement Social
- Display friends and requests
- Add friend
- Afuza Afuzarahman - UI/UX/Frontend/Backend/QA
- Steven Tang - UI/UX/Frontend/QA
- Dominic Roser - Frontend/Backend/QA
- Prayug Sigdel - Frontend/Backend
- Adam Benazouz - Frontend/Backend
- Colin Dao - Backend