Our player is trapped in the depths of a dungeon and must solve puzzles and fight monsters to advance through each room. Each room may bring them closer to freedom, but there’s no way to know what dangers might be in store for them. They may encounter mages, gnomes, and knights or a merchant who might or might not help in their journey.
Player movement is controlled using WASD. The melee weapon is controlled by the mouse position, and Left Click to initiate a swing. The player have 4 spells, bounded to Right Click
, Q
, E
, and Space
, that casts a Fireball, Lighting, Water, and Blizzard respectively. Similarly to the melee weapon, the spells are also guided by mouse position, and a button hold is used to represent the cast time of the spell. Release the spell button after 0.5 seconds for Fireball and 1 second for the rest to cast the spell. There are special mechanics to these spells. The Fireball spell is capable of piercing through and burning monsters, and is able to bounce off the wall 3 times. The Lightning spell summons 3 auto-targetting lightning bolts that target to the nearest monster, and damage the monsters. The Water spell summons a AOE damage circle that starts at the player and moves towards the direction of where the mouse was when the spell was casted. The Blizzard spell creates a AOE circle based on the mouse position, like Mage's Blizzard in WoW, and freezes any mob that gets hit.
The player also has an inventory system, bounded to I
, where the player can see the weapons, armor, and potions they have collected. The player may click on any weapon or armor and click Equip to equip the new weapon or armor. The player may also navigate to the Items tab in order to access their potions that they may use during their adventure to heal themselves when at low health.
There are 4 types of rooms: Combat, Puzzle, Shop, and Boss. Within the Combat Rooms, the player must kill the spawned monsters in order to clear the the Combat Room to progress to the next room. Within Puzzle Rooms, there are unlit torches scattered across the walls of the room that must be hit with a Fireball spell within a set amount of time, either through direct fire or through bounces. These torches do extinguish after 10 seconds, so all torches must be lit at the same time in order to progress to the next room. If the player completes the puzzle early, as indicated on the chest timer, the player will be rewarded with random item drops. If the player fails to complete the Puzzle Room within the set amount of time, the player will have to defeat the activated knights before they may progress. Beware, these knights are very difficult. The Shop Room is where the player may spend their gold that they have collected during their journey. The player may buy items such as new weapons, armor, and health potions to strengthen themselves for the remaining exploration. The player may also sell any items they have collected during their adventure. Finally, the player will encounter the red Boss Room, where the player must walk to the center to be moved to start the boss fight.
There are 2 difficulties to the game, normal and hard mode, shown as portals on the top of the Starting Room. The player may interact with either portal with F to select their difficulty. The difference of choosing hard mode compared to normal at the moment is currently only a lighting change, where the game becomes much darker and harder to see. There is also a special mode that must be unlocked through a Konami-Code like system, where you must input a specific button order to unlock the mode. To use the konami-code, you need to move player to the magic-array painting in the center of start room, press "Space" to activate the system, then input/tap "S-W-A-S-W-A" to triger the unlock.
- The player can attack with no weapon equipped.
- The Player can attack while frozen.
- Casting a Fireball while hugging the top wall always sends it off-screen.
Describe your user interface and how it relates to gameplay. This can be done via the template.
The player-controlled movement is done by capturing directional input and manipulating the Player
game object's transform
in FixedUpdate()
to have more control over the movement and stability with physic interactions. The direction of movement is captured using Input.GetAxis() on Horizontal
and Vertical
for x and y values; then normalized to ensure the player moved at the same speed in all directions. The movement itself is performed by calculating the move-offset using speed * speedModifier * Time.fixedDeltaTime
and applying move-offset using transform.position or use transform.Translate(). The speedModifier is used for slow debuffon the player and speed-bust power-ups.
There are also non-player controlled movements when a special event is happening, such as interaction with portals. Multiple boolean values were used to ensure the player's movements corresponded with the free-move / event.
To separate the weapon logic from player control, I designed the weapon to function independently from PlayerController and receive the signal from PlayerController for the attack. The CRWeapon
(Close Range Weapon) prefab is attached under Player
, so there move together. The attack is achieved by activating the weapon hitbox
and sprite renderer
to show the swinging weapon and enable the hitbox detection. The swing of the weapon is controlled by a customized lerp function Attack()
that lerp attackProgress from 0f to 1f and repositions the weapon to corresponding angle. With these controls, the weapon can show up, swing through configured attackAngle centered at the mouse position, and then disappear when finished attack.
To make sure the weapon rotation center is the (0, 0) of the weapon prefab, the counter weight
sub-object was attached to the weapon prefab to be the same shape, not rendered, and opposite position across (0,0) to make sure the weapon prefab's center is in (0, 0) for rotation to function correctly.
When the weapon is idealing (not attacking), the weapon follower
will show up near the player to show the equipped weapon. The logic of weapon follower
is similar to the PositionFolloweCamera
from exercise 2.
- It stays still when it's distance to the player is less than
InnerRadius
- oves at
followSpeed
that's slightly slower than the player when distance with the player in betweenInnerRadius
andOuterRadius
- Maintains the max
OuterRadius
when is reaching max leash distance.
Weapon | ||
---|---|---|
Weapon Attack | Weapon Follow | Counter-weight Idea |
note: the Counter-weight is compeletely transparent and collider diabled in game, here is showing idea
The purpose of spells is to enhance the player's ranged combat ability. In the early stage, spell instantiation was included inside PlayerController, but the script soon became too large and hard to manage. So I followed the factory pattern
from exercise 4 to pack spell instantiation logics into the ISpellMaker
interface that has ShowRange()
and Execute()
for indication of spell position and instantiation. The spell management structure was inspired by the SpellFactory
participation exercise to include all Spell Execution logics in the spell prefabs so all the internal logic is customized and self-contained for easy utilization by spell makers. The utilization of the particle system for spell effect was inspired by the SpellFactory
exercise.
The spell of 4 elements was designed to serve different roles:
Fire-Ball
(bound tofire2
) is the main range dagame spell and serves puzzle rooms. It shoots a fireball toward the mouse position damage and burns the enemy it hits.Water-Surge
(bound toE
) is the AOE damage spell. It creates a circle range that slowly moves towards the mouse position and damage and stuns the enemy periodically. Inspired by theLerp Playground
demo, the curved particle speed of the particle system is used to show thecharging
andburst
state.Blizzard
(bound toSpace
) is the AOE control spell. It randomly generates ice-sharps in selected areas, and each ice-sharp will go through forming, falling, and exploding stages and then do a small AOE damage around the ice-sharp.Thunder
(bound toQ
) is the Monster-Targted spell. It randomly selects a few monsters to chase and then strike. It serves more on information gain in hard/dark mode which will guaranty to reveal some monsters' position in the dark.
Spells | |||
---|---|---|---|
Fire-Ball | Water-Surge | Blizzard | Thunder |
The spell also has different level UPR 2D light effect attached to them for hard mode information gain.
The casting/ready stage of spell casting is shown by particle effect around player and is inspired by the (healing effect from classmate's SpellFactory
exeersice)[].
The initial plan on the spell part was to make 2 spells per element for the player to choose/purchase and implement the interaction between elements to allow players to build their combat style. For example, the original design for the water spell is for a wet
status on monsters that make them easier to be frizzed or induce lightning. We also planned to include a few special purposed spells for consumable scroll items, such as a light scroll
that light up the entire room for hard mode combat assist. But these plans were selectively pushed back due to task transferring and time manner.
My goal was to create a robust inventory backend that could be used in this game and future games to come. This goal however, presented the challenge of designing a generic, reusable system that could be easily adapted to the wide variety of inventories seen in different games. For example, some inventories may have a limited capacity, or limited amount of physical space where items could be arranged in. Items could also be stored individually or in stacks, sometimes with a limit on the number of items in a stack. Since these variations greatly impact the user interface side of inventory systems, I decided to start by designing the backend.
The inventory backend design centers around the principle of reliability. As items are exchanged between and within different systems, it is possible for there to be unintended duplications or deletions of items. I wanted to design the backend in such a way where game breaking behavior could not happen.
To help achieve this, the static ItemManager class keeps track of inventory items by assigning each item a unique ID and storing the item in a dictionary where each item is referenced by its ID. Each inventory item derives from a base GameItem class whose base constructor registers itself with the ItemManager, adding it to the dictionary and attaining an ID. Rather than exchanging a reference to the object itself, these systems passed the item’s unique ID and were required to go through the ItemManager to get the actual item. With centralized item tracking and unique ID assignments, duplicating items is far more difficult.
To prevent items from being assigned to multiple conflicting systems at once, each GameItem stores an ItemStatus enum, whose value represents a specific item state. For example, McDungeon has four ItemStatuses:
- Unowned - item is unowned by the player
- ItemInventory - item is present in the player’s inventory of items
- PlayerInventory - item is present in the player’s inventory of equipment
- Equipped - item is equipped by the player
Note that each status can represent a different number of objects possessing the item, such as unowned with 0 possesers, and equipped with both the active player and the equipment inventory possessing the item. Therefore each ItemStatus has a flexible list of collections that are connected to that status. Specifically, the ItemCollection interface represents something in the game that can possess items, and the RegisterItemCollectionWithStatus() method ensures that when an item has a particular ItemStatus, the registered ItemCollection is guaranteed to contain that item. Furthermore, the transition of an item from one ItemStatus to another is performed through the ChangeItemStatus() function, which makes sure each party is okay with the transition before it is performed. Namely, every item collection registered with the previous ItemStatus has to be okay with the item’s removal, the item itself has to be okay with the transition, and every item collection registered with the future ItemStatus has to be okay with the item’s addition. This design is to further prevent unwanted item duplication or removal as they are passed between systems.
For McDungeon, I designed three ItemCollections. The first is the NonStackableInventoryCollection which represents items that cannot be stacked such as equipment, and the second is the StackableInventoryCollection which represents items that can be stacked such as HP potion items. Each collection was designed to store its items in multiple different orders depending on the inventory sort method and its accompanying comparator, but this was not used in McDungeon. The third is the player inventory, which represents equipped items by the player and is limited to a weapon and armor slot.
Following this, I moved onto designing the frontend. While this implementation would be specific to McDungeon, I was still having trouble designing a reliable system. I learned about frontend design patterns such as the EBC/EBI and MVC and decided on the model view controller design pattern.
Originally I had planned for a 3 panel inventory screen.
- The right panel would display inventory items in a grid that the user could select.
- The center panel would display information about the selected item and buttons to perform actions, such as equipping, or selling the item.
- The left panel would display a player card that shows the currently equipped items and various stats and currencies.
However, after showcasing implementations of the design with the team, we decided that a single panel design would work better.
The single right panel would:
- Display inventory items in a grid that the user could interact with
- When mousing over an item, a hovering panel would display information about the item
- When selecting an item, the hovering panel would fix and have buttons the users could interact with
- Display equipped weapon, armor, and various stats/currencies above the inventory grid
Notably, the InventoryController interacted with the ItemManager backend and handled any actions the user could take to modify the inventories and the InventoryCollections model stored the various ItemCollection classes that data was retrieved from.
There are however, drawbacks and challenges to the inventory design. One challenge I faced was when to refresh the inventory UI. Originally, I had planned that opening the inventory would freeze other aspects of the game. When this design element did not go through, players could collect items or buy items from the merchant while the inventory was open, which forced panel refreshing to be triggered by external sources.
This also caused problems when players were fighting and wanted to use a health potion. They would have to press the I key, navigate to the items page, click the potion, then use it during the fight as opposed to when time was stationary. I also noticed that deselecting items was not intuitive. Players kept trying to click outside of the hover panel UI to deselect an item rather than clicking on the inventory slot again to deselect it. Moving forwards, I would definitely do some user interface research and get feedback from people who were not familiar with the system.
Besides inventory UI, I created other UI elements.
The player health bar displays player health in the upper left corner. The health controller was designed such that the number of health points contained in each heart could be changed, alongside max health and current health. The bar would then display the correct number of full hearts, half hearts, and empty hearts. Because of this flexibility, the number of hearts was rounded up, such that if each heart was worth 3 hit points and the player had 2 hit points, then a full heart would be displayed. Similarly, 1 hit point would display a half heart.
To create the shop room and its corresponding UI I created a central ShopMerchantController class to facilitate the buying and displaying of elements.
The ShopUIManager class manages the UI that displays item stats and allows the player the buy the item.
The ShopCrateController class represents each crate in the shop room. Besides storing the item it is showcasing, it also detects when the player is close enough and informs the ShopMerchantController.
The merchant room was designed such that the number of crates and items it can host can be dynamically increased or decreased. The items sold are initialized inside the ShopMerchantController class.
Towards the end of the project, I created the rudimentary pause menu. By pressing Escape, the user can pause the game with an option to restart.
The game was paused by setting the Time.timeScale to 0. This however, only pauses certain systems in the game. Accompanying the timeScale change, a global state object has its isPaused boolean set to true. This is used to disable certain features that aren’t affected by timeScale, such as Update() functions where key presses are detected to toggle inventory.
There is a bug currently where if the inventory is open, the game can be paused. Pausing the game should instead close the inventory.
The restart button works by using the SceneManager to reload the active scene. Unfortunately this caused several game breaking bugs that were difficult to fix. Some interesting bugs we encountered were that:
- The Start() scripts were called in a different order than the initial ordering - Switched from Start() to Awake()
- Update() function of a script is called before Start() is finished calling - Finding this was difficult, but to fix it, simply add an isStartFinished boolean
- GameObjects that once existed during the Start function are missing after the start function but exist again after a random period of time - Specifically, the Player’s weapon hitbox does not exist for a short period after CRWeaponController’s start() is called. To fix this, add a check to make sure the game object is not null, then call the method multiple times so that there will be one successful call when the game object is no longer null.
Upon player death, the death screen lowers a black curtain and shows a dead player along with a restart button. The restart button functions exactly the same as the one in the pause menu.
The items in the game were divided primarily into two types: Consumable items and Equipment items. All of these items extended the GameItem class used by the Inventory system.
Since none of the above items are a monobehaviour however, item drops themselves had to be a separate object and have its own controller. Specifically, the EquipmentDrop class contains either a Weapon or Armor that it adds to the player inventory when collected, and the HealthPotionDrop class contains a HealthPotion that it adds to the item inventory once collected. Both of these classes derived from the DroppedItemController. Item drop in the puzzle room appear like so:
To aid with the creation of items, the ItemFactory class creates different item prefabs. The ItemFactory was primarily used by the PuzzleController, which indicates whether a big or small puzzle reward should be provided. The ItemFactory then pulls random items from different prize tiers, groupings of items with relatively similar strength, that it then deposits for the player to pickup.
The biggest challenge with the item system however, was integrating the items with other systems.
The Weapon object specifies stats the player’s weapon should adopt. To ensure that the player wields the correct weapon, the SyncWeaponWithInventory() method was added to PlayerController, which passes on the weapon stats to the CRWeaponController class.
In the case where the equipped weapon is null, which indicates that the player does not have a weapon equipped, there is currently no behavior to disable the weapon. Instead, the CRWeaponController will maintain the same stats it had from the previously equipped Weapon.
The HealthPotion integration was simpler, as the RestoreHealth() method can be invoked, and the health potion heal amount is passed in.
The concept of the Mob AI was to track and move toward the player, and attack the player when the player is within the mob's attack range. There were 6 mobs in concept: Slime, Skeleton, Knight, GNome, GNelf, and Mage.
In its earliest phases, the design was very simplistic, the mob had to take damage, and the mob had to keep track, move towards, and then attack the player. There were no Mob assets to begin with so they started out with simple Hexagon prefabs so that the interaction between player and mob could be tested. Each mob was designed very simplistically, with Slime, Skeleton, and Knight all melee attacking. GNome was designed to throw a projectile and spawn GNelfs, and Mage was designed to cast either a Fireball or Frostbolt. These projectiles would have their own controllers similarly to Exercise 4 and my self designed pirate command in Exercise 1 . Each mob had its own controller, derived from the IMobController
Interface, that allowed the player easy access to call the function TakeDamage()
. The MobManager.cs
was designed after to keep track of all the mobs in a list, with functions Subscribe()
, Unsubscribe()
, and Notify()
to keep track and control all mobs that were spawned, just like in [Exercise 3](SOME LINK). There were some simple functions designed, such as SpawnMobs()
andGetMobs()
, so that the room manager was able spawn mobs and keep track of whether the list of mobs was empty so that the player may progress.
We wanted more variety within the mobs, as many of the mobs shared the same attack patterns or traits, so we redesigned Skeleton, and Knight and made a small rework to the Slime. The skeleton was redesigned so that it threw a bone projectile, and it needed to retrieve its bone before it can throw it again. The decision to have the Skeleton pick up its thrown bone was to differentiate from the GNome, who threw knife projectiles at the player. The Knight was redesigned so that it had a respawning shield that takes a free hit. The Slime was minorly changed so that it slowed the player.
A design bug came up while reworking Skeletons, where a different Skeleton can steal another's bone. In such case, the Skeleton that got their bone stolen ended up dancing inplace. I decided to make this bug into a feature, reworking how Skeletons would pick up thier bones, with the possibility that another Skeleton could steal a bone. During this development phase, a race condition was created where a bone could interact with two Skeleton hitboxes simultaneously, and a Skeleton could interact with two bone hitboxes simultaneously. Using knowledge of semaphores, they were implemented to prevent two Skeletons being able to gain a bone from 1 bone, and vice versa.
Each mob had a state machine system. The Slime, GNelf, and Knight have a similar system, move toward the player, if the player was within its attack range, attack the player, and repeat the cycle. Skeleton had a modification to that. Skeletons would move toward the player, and if it was within its attack range, it would throw its bone projectile. It then would retrieve the bone, and repeat the cycle. The Mage was designed to move toward the player just like the rest of the mobs, and if the player was in range of the Mage, it would start casting either a Fireball or Frostbolt and the cycle would repeat. Similarly to the Skeleton and Mage, the GNome combines both systems. It would move toward the player, and if it was within range of the player, throw a knife projectile. And similarly to Mages, if its cast cooldown was up, it would start casting on the spot until it gets inturrupted by a hit. If this cast went off, GNelfs would spawn around the GNome, and the cycle repeats.
As we wanted to implement status effects such as burn, freeze, and slow, I designed a Scriptable Object, StatusEffects.cs
, that could be used globally for statuses. This Scriptable Object would store the status prefabs and status durations for death, stun, burn, and freeze, and would contain functions that would instantiate these prefabs. A design issue came up during this, where each mob had hundreds of lines of code that were identical between all mob controllers. This prompted me to rewrite all mob controllers using the abstract class Mob.cs
that would still inherit the interface IMobController.cs
, and allow generic functions to be written for all mob controllers. Some of these functions were virtual functions, allowing certain mob types to override the existing function to fit into the mob design. Each of these status effects use Coroutines to control its duration, where each time a status effect would occur, the existing Coroutine for that status effect would be stopped and a new one would be started, continuing the status effect.
In the end, I realized that I could have changed this implementation so that instead of a prefab being generated every time a status effect occurred, each mob prefab would contain the status effect sprites instead so that the SetActive()
function could be used to show different statuses on the model.
This Scriptable Object was then extended to the playerController.cs
to allow mobs to apply statuses onto the player, with accompanying functions added for the player.
The concept of the Boss AI was to take the Among Us player and create a boss fight with multiple attack patterns designed from the Among Us game.
Having Mob AI already made, the Boss AI was relatively simple in comparison. Many of the functions that were used in Mob AI were transferrable to Boss AI, however the Boss AI would need different attack patterns. As the theme of the boss was Among Us, we decided that the two attack patterns were the button slam and the laser beam. Having learned about child GameObjects and SetActive()
, two hands representing the left and right hand, a button, and the laser beam was added into the Boss prefab as child objects. These child objects would be activated when the Boss AI is on that attack pattern and deactivated once it has been finished.
As assets were introduced, Animators were added to all mob prefabs, and animations and animator controllers were implemented based on the design of the mob.
Due to the design of each mob, every mob had a different animation state machine. However, all mobs contained booleans for Attack, Freeze, and Stun within its animator controller that led to its attack animation or idle/still animation respectively. Each mob's controller contained a function, spriteControl()
, that changed the direction of the sprite to either up, down, left, or right while the mob was moving. As all mobs had an attack animation, except for GNelf who just charges at you, an attack state machine was needed to be implemented on all mobs. This attack state machine would track the elapsed attack animation time, and trigger the player damage or projectile instantiation at the correct point of time during the animation. This was heavily adjusted during Game Feel testing, which I also conducted.
We wanted to put in an Easter Egg player spell for the game's special mode.
TA AI, or Arunpreet AI, does 1 job, attract mobs and throw attendance codes at them. When Arunpreet is spawned, a black hole effect is triggered. Similarly to my implementation of a black hole in the spell factory extra exercise, it takes the list of mobs and moves them towards Arunpreet's character model. Then he randomly throws attendance code projectiles in all directions and disappears.
The context of the game's environment was imperative to even beginning the process of creating assets. Knowing this, I created a number of concepts for the player, mobs, and the map itself so that my teammates and I would have a clearer vision of what the game would look and feel like. This way we could continue on with a relative consensus of what the game's feel and story was like so that their individual parts would not stray too far from the concepts.
All concept art was made using the app GoodNotes
on the iPad.
Concepts | |
---|---|
Initial Map | |
Final Map | |
Player | |
Monster Ideas | |
Slime | |
Skeleton |
Now that the theme and feel of the game was set, I tried my best to keep the feel the same throughout creation. Since the game did not have a set narrative, I had to make sure that all the assets, at least, had a consistent pixel sizing and palette so that the game had the symblance of being cohesive. We decided on have the animations made in a 64-bit style since it was closest to what I had done before in Exercise 1, but I ended up just making assets that fix what my teammates had coded. With the decided medieval style dungeon, it was important for me to keep the color palette muted and almost muddy to emphasize the implied atmosphere that comes with being in a dungeon. I was mostly inspire by the "dark academia" color palette when choosing colors, using the website coolors to quickly generate colors that might work. In the cases where I needed a tint or shade of a color, I used the site Tint and Shade Generator. All assets were made by me (Krystal) using the free website Piskel unless specified otherwise.
- Dark Gray outline around the entirely of the character and major parts of design
- "Glowing" effects do not need to be outlined
- Colored using the set palette
- Each GIF much run at 12 frames per second
These rules were set after creating the first Player sprite described below.
Our game is a 2D top down dungeon crawler, so the first this we had to figure out was the number of directions each mob and the player could move. Originally, we had decided to do all 8 directions (up, down, left right, and the diagonals), but we later decided against it and chose to do 4 directions due to the shear number of assets I would have to create in the end and the reasurrance that it could work since the game Don't Starve
also does not have 8 directions.
Once the style of animation was chosen, I animated the rest of the directions, not without problems, but with much more ease than the first.
Up | Down | Left/Right | Idle | Dead |
---|---|---|---|---|
The idle animation and death icon where created much later in the process compared to the previous player assests.
From the player onwards there were not many style issues due to the color palette and drawing style being set; the only real challenge was making the movements feasible to the mob itself.
Since we had many typical mob types, it was important for me to make them unique as unique as I could think of, even if their mechanic ends up being similar to another game.
The mobs are listed below in the order they were created.
Up/Down | Left/Right | Attack |
---|---|---|
Type | Up | Down | Left/Right |
---|---|---|---|
Chase | |||
Attack | |||
Bone | The bone is one of the few times I forgot to follow my animation rules (forgetting the gray outline) |
Up | Down | Left/Right |
---|---|---|
Type | Up | Down | Left | Right |
---|---|---|---|---|
Chase | ||||
Attack | ||||
Cast | ||||
Knife |
Type | Up | Down | Left/Right |
---|---|---|---|
Chase | |||
Cast |
Type | Up | Down | Left/Right |
---|---|---|---|
Chase | |||
Attack | |||
Statue | |||
Shield |
Weapons and armor were created as a way for the player to have some kind of customizability to their playthrough. Although exteremly RNG, having one of the more powerful weapons/armor appear in the Merchants room can drastically change the player's experience playing the game with different stats on each item.
All Weapons | All Armor |
---|
Ablaze | Frozen | Stun | Death Poof |
---|---|---|---|
The concept for the Merchant was to set him up to be a potential boss. However due to time constraints, he is not a boss, but his concpet remains. The merchant is a retired Mage that has now devoted his life to selling junk. Although he seems very homeless, he is extremely rich.
His design is almost identical to that of the Mage with a recolor and some cloth patches to emphasis his homelessnes. Since a merchant like him would want to hide his richness, it is protrayed subtly through the gold buttons on the merchant UI panel.
Merchant | Tent | Rug | Together |
---|---|---|---|
You (Professor or Arunpreet) may recall someone asking you what you what you usually where on a typical day. This is the result of that.
The reason for the noticable style change here is that these characters can only be unlocked by completing a special pattern. Since they are hidden under a code, they must also adron a special style specific to them.
I wanted to make cute sprites for once.
Running | Idle |
---|---|
Spin | Attendance Code |
---|---|
Normal | Hard | Special |
---|---|---|
I made sure to make all the UI panels, buttons, and slots as close to the theme as possible, while also being simple enough so that the design itself would not distract from the items in the inventory too much. It is also important to keep the theme consistent in these screens because thery blocks a large portion of the play area that contains the "atmosphere" of the game; keeping the theme as closely as possible helps midigate any damage to game feel.
The puzzle buttons had be extremely obvious in order for Players to get an understanding of the puzzle quickly. I made sure to make walls that correspnded to a specific button very obivous so that there is little trial and error on the player's part regarding puzzle mechanics.
Hearts | Potion | Chest | Gate | Decorations | Coin | Scroll |
---|---|---|---|---|---|---|
As of the time writing this (6/13/2023), this is the complete list of all assets I have created excluding: UI Elements, Boss Health Bar, and a Scrapped Torch.
The main concepts behind most puzzles were made on graph paper, where I could correlate each square on the paper to a tile in game. I chose to desgin on paper because it was the best way to test many different designs quickly and without too much of a turnaround time so that the implementor(Jason) could work on the room.
We chose to do a "bouncing angles" type of puzzle where the goal is to light all the torches in the room by bouncing fireballs of the walls and obsticals of the room to hit them under a certain about of time because our spell implementor(Honghui) ended up creating the fireball spell quite early on in the game creating process. This way the spell could be useful to not only attack, but also to solve puzzles which helped further integrate the mechanic with the game as a whole.
Originally, the puzzles were supposed to reward the player with an upgrade to their fireball spell and to punish failure with the special knight mob that would only activate upon puzzle failure, but the upgrade was scrapped as we felt it was no longer necessary. Despite the scrapping of the previous reward, we felt that it was still important to provide an incentive for players to complete puzzles so that they would be less tempted to wait out the puzzle timer, so we provided random drops that might or might not be better than their current gear and strengthened the knights.
We ended up creating a total of 5 puzzle rooms, including the tutorial, 4 of them designed by Krystal and 1 by Jason.
Most of my puzzles went through multiple design changes as our game was developed due to the screen's restriction of the player's sight. The first itteration of puzzles too large for the player to be able to see all torches on the map, so it forced the player to guess angles to bounce fireballs, rather then making more educated decisions.
Once these problems were brought to my attention, I opted to redesign each puzzle so that they would fit in the screen's 9 by 5 block limit, while trying to keep the rooms as close to the original designs.
Room 1 Version 2 | Room 2 Version 2 |
---|---|
Both of the new designs were inside the constraints of the player's field of vision, coupled with a new mechanic to lock the camera on the center of the room so that the player would always be able to see the entire map. This change effectively made each puzzle much less of blind guessing, but more of the intended educated shooting.
The 3rd room I created was made after the above observations, so it was created with only one drafting. The major difference between this puzzle and the others I created was the new mechanic of moving walls, made by Jason as he was creating his own.
Room 3 |
---|
The last room to mention is the Tutorial puzzle that is supposed to teach the player the mechanics of puzzles, was actually created sometime between the first puzzle version 1 and the second puzzle version 1.
I used a teleport based system in order to link several rooms together to form a cohesive map. This was done to allow for most of the map related gameobjects to be already loaded upon entering the scene. In order for the game to feel unique every run, I made it so that how the rooms are linked are based on the map configuration matrix which is procedurally generated.
In addition to this majority of the map-related assets were created by me (Paolo) using the free website Piskel.
I created the initial design for a single room grid in our game environment. I needed this to be done in order to create the rest of the map.
I decided to use tile maps for room creation as this gave flexibility and ease of use when creating different layouts and design
I had different iterations of the room tile map as we weren't sure yet on how are room should look like. Either having it look isometric at a 45 degree angle or to have it completly top-down similar to The Legend of Zelda (1987) or the Binding of Isaac: Rebirth.
Room Tile Maps | ||
---|---|---|
Initial Design for Wall | ||
Finialized Design for Wall | ||
Initial Design for Floor | ||
Finalized Design for Floor |
We decided on having a total of 6 room types:
Start Room, Tutorial Room, Puzzle Room, Combat Room, and the End Room.
I decided to have a total of 16 rooms in order to prolong the game, this consisted of:
- 1x Start Room
- 1x Tutorial Room
- 8x Combat Rooms
- 2x Shop Rooms
- 3x Puzzle Rooms (3 are chosen out of 4 existing)
- 1x End Room
These rooms had their own respective colors and layout. Starting Room was set to the default room and color.
Boss Room Entrance Found inside the End Room, and teleports the player to the boss room when colliding with the trigger |
Animation and trigger is handled by the VentDetector.cs
, and BossEntrance.cs
script.
Teleportation to boss room is handled by the BossTeleport.cs
script via on trigger.
The map was procedurally generated using a heavily modified drunkard walk algortithm in DrunkardWalk.cs
tailored to fit our layout. This algorithm could potentially make any n x m matrix map by modifying some values, but I kept these values set to follow our map constraints (specific number of total rooms, specific number of combat, puzzle, shop rooms).
I used an enum RoomTypes
that represented different room types:
- 0 being No Room
- 1 being Starting Room
- 2 being Tutorial Room
- 3 being Shop Rooms
- 4 being Combat Rooms
- 5 being Puzzle Rooms
- 6 being End Rooms
- 7 being Temporary Rooms
The public function GenerateMatrix
returns a n by n generated matrix that could be fed later to our MapGenerator.cs
script.
GenerateMatrix
first chooses a starting room randomly in the matrix by setting the matrix [n][m] value to 1
. It then calls the GenerateTrainingRoom
which generates a training room by assiging the matrix value to 2
always adjacent to the starting room.
After it picks both the starting room and the training room, the drunkard starts walking randomly in different directions which is stored in the nextRoom
value for 100 steps trying to create a total of 15 rooms (end room is added at the very end). If the drunkard hasn't finished creating all 15 rooms within the amount of steps, it just regenerates the matrix again. This drunkard will assign the value of a temporary room, which is 7
, only on empty spots in the matrix. I also made a constraint that it cannot place a room adjacent to the starting room, as the tutorial room has to be the only room next to it.
After it generates all the temporary rooms, the drunkard stops walking. Multiple functions are then called to assing the different room types that are remaining. GenerateShopLayer()
generates the shops replacing temporary rooms with its value, 3
, this makes sure that no shop rooms are placed adjacent to a shop room. GeneratePuzzleLayer()
functions simialrly to GenerateShopLayer()
placing its value 5
on top of temporary rooms and ensuring that they're not placed adjacent to another puzzle room. GenerateCombatLayer()
just fills the remaining temporary rooms with the combat room value 4
. Finally, GenerateEndRoom()
looks for a room that exist, is not completly surrounded by rooms, and is as far away (by distance not by rooms) from the start room and places the end room 6
on a random empty spot adjacent to it.
I initially had planned to generated the end room to be the furthest amount of rooms away from the starting room but I found it too complicated to do within the remaining time. I also felt like I could've used a better map generation method but decided to go for the more simplistic route.
I used the MapGenerator.cs
script to link all the rooms together via teleporters, esentially creating the pathway between rooms.
MapGenerator.cs
on Start()
, looks for all the type of room gameobjects that exist within the scene and adds them to their respective lists via the AssignList()
method.
Then from RNG, the map is selected, there is a 25% chance of a premade crewmate
shaped map being picked, and a 75% chance of being a randomly generated map from DrunkardWalk.cs
.
Then according to their respective location in the matrix, AssignRoom()
is called to iterate throught the fed n by m matrix and uses its Vector2Int
position in the matrix as the roomDictionary
key and assigns the respective room gameobject as the roomDictionary
value based on the room value in the matrix.
AssignPortal()
is then called which iterates through the map matrix and checks the current room's right and bottom if there exists a room. If there exists a room on the right, the current room's left teleporter PortalA1
will link to the right adjacent room's left teleporter PortalA2
. Similarly, if there exists a room on the bottom, the current room's bottom teleporter PortalB1
will link to the bottom adjacent room's top teleporter PortalB2
.
This linkage is done via the LinkTeleporter.cs
script attached to all teleporters (via prefab) by assigning the TargetRoom
gameobject to each others teleporter. If a player walks to the teleporter via an OnTriggerEnter2D
it teleports the player to the targer room.
While linking these rooms, I also made sure that the teleporter sprites were not rendered via the LinkTeleporter.cs
script by checking if TargetRoom
was null. The wall tiles were also updated properly to reflect the map layout. This meant that if there was no target room the walls will visually get updated, this is all handled by the TileUpdate.cs
script attached to the wall gameobject of each room.
I felt like this could've been done more efficently if I used a central teleporter controller for each room.
The minimap is generated via the MapGenerator.cs
script. It creates a bunch of miniRoom
gameobjects arranged in a matrix and is shown via the canvas ui. The minimap has a transparent background for better visibility. Visited rooms are colored with light grey, current room is colored with white, and unvisited adjacent rooms are colored with dark grey. The boss room has a unique icon. This minimap design was heavily inspired by the Binding of Isaac: Rebirth.
The icon sprites were drawn by me:
Icon | |
---|---|
Room | |
Boss Room |
Each room had their own preset conditions to complete. If the room is not completed when entered, then the teleporters (represented by the gate sprites) will be closed and kept shut. This is mostly handled by the LinkTeleporter.cs
script found in each teleporter. For puzzle rooms/tutorial room, upon entering, the camera shifts to center on the room, and the minimap disables for better visibility. When the puzzle starts, the tiles updates via TileUpdate.cs
to create an enclosed room with no escape until the player completes or defeats the knights that will spawn when failing (by checking the mob count). For combat rooms, the room is shut if there are mobs and completes once they all die.
Sprites | |
---|---|
Candles | |
Chain Ball (unused) |
|
Item Pedastel (unused) |
From my understanding of game design and game development cycles, I conducted regular weekly meetings to check on each teams progress and to make sure that the game development was on track. I scheduled our Initial Game Plan Gantt Chart, with an emphasis on focusing on creating quick and simple systems early in the development cycle, so that certain dependencies could be tested early on to ensure cohesion throughout our game. An Excel Sheet was used to keep track of each person's tasks with expected deadlines to ensure that certain features could be tested at specific times.
Using information from personal playtesting and from the Project Game Showcase, many factors were changed or tweaked to enhance game feel.
Puzzle Rooms - Puzzle rooms initially felt very clunky with the existing spell designs, with high cast time and long cooldowns. Although those cast times and cooldowns were meant for combat, it massively slowed down the gameplay during puzzle rooms so much so that the tutorial room was very easily failable. Some suggested changes were lowering both the cast time and cooldown of the Fireball spell specifically for the puzzle room, so that the player can test multiple angles much faster, and to increase the torch duration once it had been hit by a Fireball. Another issue that came up during the Project Game Showcase was that certain UI elements were blocking torch visibility, so moving the timer to back to the center of the screen and disabling the minimap and zooming out more for puzzle rooms were suggested as solutions.
Combat Rooms - Initially we had long cast times, and not being able to cast while moving, however it was concluded that the casting felt very clunky, and too inhibiting. This was then changed to a lower cast time, and being able to cast while moving, making it less punishing on the player if they wanted to use spells. I had also initially known that mobs were very overpowered or strong, where some projectiles were not dodgeable or cast times and cooldowns were way too low, so many of the mob numbers were tweaked so that there is a slight challenge to fighting mobs.
Tutorial Puzzle Room - The tutorial room currently gives weapons and potions as rewards for completing it fast, however it didn't make sense for the tutorial room to give a weapon that would replace the starting weapon that had not been used at all. Thus, a suggestion for removing weapon drops for the Tutorial Puzzle Room was made.
Player - During personal playtests and duing the Project Game Showcase, the general consensus was that the player moved too slow. I had suggested to increase the player move speed to help with the flow of the game, however due to time constraints and required testing with new player speed, we were not able to implement the change.
My goal for puzzle implementation was to design a flexible, reusable puzzle engine that could handle a wide variety of puzzle elements. Since the specific puzzle elements and designs were undecided at first, I designed the puzzle engine similar to how I designed the inventory system, using a flexible MVC design pattern.
However, as I was planning out my object hierarchy I suspected that it would make more sense for the View and Controller for each puzzle element to be tied together since both would have to be MonoBehaviours and were closely related. After talking with the professor, I changed my design so that View and Controller were tied together, labeled as Controller. My initial design was fairly simple:
Since puzzle elements were a nebulous concept, I created a generic design centered around the PuzzleController.
The PuzzleController class stores the state of the puzzle and the state of its elements in PuzzleStateModel, and manages the various puzzle element controllers. When a puzzle element is created, it is assigned a unique ID and its controller is stored in a dictionary by the unique ID inside PuzzleController. Similarly, the associated PuzzleElementStateModel is stored by the same unique ID in a dictionary inside PuzzleStateModel.
For puzzle elements to interact with each other, a RespondTo() method was specified in the PuzzleElementController. This method would be invoked by a trigger, namely a change in another puzzle element’s state. The PuzzleController stores a list of elements whose RespondTo() function would be called if a state change of an element with a specific ID occurred. In order to register an element was a trigger to another element’s state change, the AddItemTrigger() method was used. Whenever a PuzzleElementModel had its puzzle state changed, it would call TriggerResponders() to call the various RespondTo() functions of registered respondents. With these core features of puzzle elements implemented, I began to design specific elements based on initial designs.
Based on the initial designs, there were three primary types of elements for the puzzle rooms.
- Buttons - Interactive buttons the player could step on to change other elements in the puzzle.
- Walls - Walls that served as additional obstacles to the room, some static and others changeable.
- Torches - Lightable torches that served as the objective of the puzzle.
While I left my win condition for puzzles flexible at first, the objective of every puzzle ended up being lighting up all the torches to complete the puzzle room.
Each button had two possible states, as can be seen in ButtonStateModel: Pressed and Unpressed. Two types of buttons were created:
- Push button - Pressed when the player stepped on it, and Unpressed when the player was not actively stepping on it.
- Switch button - toggle between Pressed and Unpressed each time the player began to newly step on the button, which meant that the player would have to step off the button then on again in order to trigger a state change.
Each wall had four possible states, as can be seen in WallStateModel: Open, Closed, Opening, Closing. The reason for the additional transition states(opening and closing) was so that wall transitions did not appear jarringly instantaneous. Three types of walls were created:
- Static wall - stationary wall
- Disappearing wall - wall that vanished when Open and materialized when Closed
- Sliding wall - wall that moved between two points, depending on if open or closed
Variations of the sliding wall and disappearing wall that responded to a specific button state ID were created. Parameters of each individual wall were left flexible and could be tuned, including:
- Wall dimensions
- Transition time between open and closed
- Delay time when reacting to a button state change
- The block sprite overlaying the wall
Walls were particularly tricky because depending on the size of the wall, block sprites needed to be overlayed at the right positions, and other items resized.
One such item was the bouncy wall prefab. For the fireballs to bounce, a bouncy wall prefab object needed to be used, which was a resizable rectangular sprite that the fireball would calculate bounce angles from.
Four bouncy wall prefabs were placed on the top, bottom, left, and right of each wall object.
In the first iteration, the bouncy wall colliders were too thick, such that when the fireball hit anywhere near a corner, it would detect two simultaneous bouncy walls and bounce at an awkward angle. Therefore the walls were adjusted to as thin as possible. Even now that the bouncy walls are paper thin, there are times when the fireball will bounce off near corners at awkward angles.
Another issue was the reappearance of walls when a fireball was inside the wall, such as when a disappearing wall begins to reappear with a fireball in the center. Therefore an internal wall collider was added that would destroy the fireball when such incidents occurred. Tuning the size of the interior wall collider was tricky since it couldn’t be too large or else the fireballs would be destroyed on bounce from the exterior, or too small since the fireball would appear traveling inside the wall for a period.
Each torch had two possible states, as can be seen in TorchStateModel: Lit and Unlit.
Each torch would start off Unlit but would transition to Lit every time a fireball touched it.
Torches were designed with a possibility of expiring after a certain duration of time, transitioning from Lit to Unlit, which would add additional time pressure to the player. Note that each time a fireball collided with the torch it would reset the expiration time.
The puzzle timer was created to add a time constraint to the player as well as to reward the player for completing puzzles efficiently. If a player completes before the first time listed, they get a large reward. If they complete before the timer reaches zero, they get a smaller reward. If they fail to complete the puzzle, the knights activate.
The rewards for the tutorial room specifically are different. Completing the tutorial room by the efficient time grants a smaller reward, and completing it before zero grants a few health potion drops.
Rather than placing the objects on the scene directly, I wanted to create the puzzles at runtime. To achieve this, I originally had a method that initialized a series of puzzle elements but this took up too much space. Instead, I stored puzzle room data inside of a CSV file that was loaded while the game was in progress. The puzzle room data had to follow a specific format
Interestingly enough, the Unity Build could not use System.IO to load files so instead Resouces.LoadText was used to replace it.
After discussing with the TA, I learned that perhaps an XML or JSON file would make more intuitive sense than a CSV, but unfortunately I did not get around to making the switch.
Developing the puzzle rooms involved coordinating with a lot of other systems in the game, including the map, players, and enemies.
At one point during the puzzle implementation, an issue arose between the puzzle designs and map mechanics. Since players would be entering the puzzle room through doors, they could end up at an unanticipated starting position for the puzzle. Furthermore, doors were not intended to exist while playing through the puzzle. To resolve this problem, I proposed a puzzle start button. To start the puzzle, the player would have to step on a button in a specific part of the room, which would close the doors and spawn puzzle elements.
Since each puzzle room had a different center position and the elements were loaded at runtime based on xy coordinates, a PuzzleHome object was created at the bottom left of each room to make placing elements consistent.
When developing and testing the puzzles, there was originally no fireball cooldown or cast time. Once this was added, the puzzles became more difficult and so design adjustments were made.
The addition of knights went smoothly since they had methods that were designed to be called at different stages of the puzzle.
When testing the puzzles, we found that it was difficult to solve puzzles without being able to see the entirety of the puzzle room. Therefore each puzzle was resized to 9x5 and the camera was fixed when entering the puzzle room.
I make multiple attempts to create my own soundtracks using a free to use website called BeepBox, but was unsuccessfull as what every I created was alway either too happy for a dunegon atmosphere or just bad in general. After my failures, my goal was to find a soundtrack that not only fit the medieval fantasy dungeon vibe of the game, but was also 8-bit. My reasoning for wanting an 8-bit soundtrack is quite simply inspired by Undertale
and it's outstanding soundtrack. Therefore, I found 8-Bit Fantasy & Adventure Music
, royalty free music by xDeviruchi, using Track 10 for combat/puzzle rooms and Persona 5 Among Us Remix
for the boss fight.
8-Bit Fantasy & Adventure Music
Track 10 is used for the combat/puzzle rooms because I found the track mysterious which fit with the exploring nature of our game. It was also calm/quiet enough to hear all the sound effects at the same time. It is also a sound that is easy to listen to, so it does not matter how many times it loops and will not get immediately annoying.
Persona 5 Among Us Remix
is used for the boss fight
because of the boss itself. There is a huge style change, but the style change also happens in game so the change is not completly out of no where. According to, the creators of Among Us
, on Inner Sloth's website:
- Educational Use/Teaching
Teachers may integrate the Among Us IP in educational materials for small group classes and team-building activities, but we ask that you refrain from widespread distribution or sale of these materials.
NOTE: Innersloth will not issue licenses for educational use of the Among Us IP.
Examples – Permitted Educational/Teaching Use ● Creating and playing an interactive game with your class based on Among Us IP
● Using Among Us IP in an interactive > slideshow or school-wide program
● Sharing the materials provided above with fellow teachers at your school (at no charge)
Therefore, we should be allowed to use this in our game.
There is/will be special voice lines from Arunpreet that fire in the special mode when using one of the spells. The reasoning for this was that, we thought it would be funny and would add to the "special-ness" of special mode.
Whoosh Sounds Effects HD (No Copyright)
by YouTube Sound Effects - Sword whoosh for both the Player and Knight.
Flash Fire Ignite-Sound Effect (HD)
by Fame FX - Torch Igniting.
All other sound effects are created by me using BeepBox to emulate each sound as best as I could in the 8-bit style.
I made sure to create sound for most attacks to that the Player could distingush the sound of being hit or the sound of an attack from an enemy. I also made sounds to indicate actions like spells or opening the inventory, so that actions would not just result in visual queues and so that these actions are more satisfying to the player.
The sound implementation mostly consisted of sorting sounds into 4 categories: background music, room sounds, player/mob sounds, and special sounds. Then I would create a dedicated SoundManager
prefab for the category, create a custom tag for it, and assign all sounds as a sound component. Depending on what sound it was adding, I would locate the tag and proceed to call the sound through playing the corresponding sound's array index in it's sound manager. This gave the me an easier way to find and add in audio, since I would not have to spend time counting Audio Source
components to find the indexing of each sound. The old exceptions were sounds that were always played upon use, where I opted to directly add the Audio Source
compontent to the game object itself and set it to "Play on Awake."
We used the URP (Universal Render Pipeline) package from unity to enable the 2D lightings. Most of the lighting is pre-defined in the prefabs when we make them, we also make some objects such as UI have un-lit
(not affect by light, always shows up) material to emphasize them. Some of the lights were controled by script for cut-scene effects. Intentionally keep combat rooms dim and a torch light around player to create a isolated and "surrounded by danger" game feel. The global light is disabled in hard mode to further enhance this feeling.
The camera mode is tight to players' action.
- LockOnPlayer: Normal mode in which the camera follows the player, we decided to have this type of camera to enhance the first-person feeling when the player explores the map.
- LockOnRoom: Puzzle mode in which the camera Locks on the center of Room for puzzle solving.
- MoveToTarget: Used for cut-scene when moving camera to a object.
- ReturnToPlayer: Transaction mode when switching back to Lock Player so the camera doesn't instantly teleport to the player.
Camera & Lighting Demo | ||
---|---|---|
Use Normal Mirror | Use Hard Mirror | Unlock Special Mirror |
note: the implmentation of cut-scene effects logic is relying on PlayerController to manage and call APIs of StartRoomLight and Camera for execution.
We porposed to have a start-room intead of a start meau UI for the immersed game experience. The design of start room is that a player been summoned from a initial portal to this world and the initial portal closed after summon so he/she need to find a way out from cross the Normal/Hard world from the mirror portals. The third portal was designed as Easter egg that allow current player to escape immediately but summon Dr. McCoy as "sacrifice"; however, is protected by a magical curtain from use.
The make the special mode unloaking have some magical background, I integrate Dylan's idea of have Konami code with the 4 element we choosed to make a Magic Array
in the center of start room for the ritual (Konami Code Input). I also make the code tights to the element base on the position they shown in the Magical Array. The sequence of code is collect from each member excpet myself by order of their reply and last code was from Dr. McCoy (last lightning - 'A'). The intention of these designs is to fake the special mode as a summon spell in the Magical Array consist [Space-S-W-A-S-W-A] as step to cast the summon.
The implementation of Konami Code compelete animation was breaking by step and controlled by a timmer.
Magic Array Design | |
---|---|
First version design (by me) |
Revised to integrate better to floor (by Krystal) |
Portal Animation | ||
---|---|---|
NormalPortal | HardPortal | McPortal |
Portal Face | ||
---|---|---|
Normal (didn't used) | Hard | Dr. McCoy's github picture |
note: the Konami Code demo shown in Camera demo. And the rest parts of Mirrors was created by Krystal.
Some pushed back designs of the Start Room include cues to the Konami Code by putting some examable items related to each member + Dr. McCoy to cue their credit to the game, and the dialog will include the code choosed by them (excpet me) or how to enter the code ("need some 'space' to cast speical spell") . This was to make the room less empty and also gives credit to ourselves.
I thought of the boss concept, and boss stage design with the clear reference to Among Us. All of the create sprites and tiles, shown under here are all drawn by me using the free website Piskel. The Boss AI and scripting was handled by Orien Cheng.
Upon entering the vent in the final room, the player enters the Boss Room and will be guided upwards. At a certain point when going up, there is a trigger via BossTrigger.cs
that starts the boss, prevents the player from backtracking, and changes the music.
Paint.exe Concept Art | |
---|---|
Boss Entrance & Design (decided on the right one) |
|
Boss 1st Phase (only one we decided using) |
|
Boss 2nd Phase (unused) |
Left Hand | Head | Right Hand |
---|---|---|
Laser Beam | ||
Button |
Tiles for TileMap |
---|