Beat, Heart, Beat is a psychedelic, indie rock rhythm-platformer about slashing demons in pursuit of your heart. Slash enemies to extend your airtime and unleash massive air combos, then use that momentum to traverse the psyche!

Gameplay Footage

Contributions

Programming - Key Contributions

Beat Management System


As a rhythm-platformer hybrid, Beat, Heart, Beat presents the unique challenge of being both synced to the music and requiring physics, collision, and spatial movement. Early concepts of the game saw one of these core features "faked" in service of the other - we tried collision-only enemy detection without timing windows, and we tried basing the jump length on the song bpm. 


Solutions like these didn't feel nearly as responsive or satisfying as I'd liked, so I explored a way to create a system that could:

Initialization with Wwise



When an audio track is prepared, the Beat Manager sets the MusicSyncBeat callback flag, which I used to invoke a global UnityEvent called OnBeat(). This allowed me to create highly reusable listener functions among any features that needed to be consistently synced to the beat.

Initialize the Wwise Event with a MusicSyncBeat callback, which invokes the OnBeat UnityEvent

The OnBeat event ensures all listeners are ALWAYS synced to the beat and can perform custom functionality like pulsing or animating sprites:

Beat Manager helper functions that convert worldspace to song playback

Playback


Since the player moves through physical space during play, every level needs a method to be converted from a song's playback position to world space. 


The Beat Manager contains helper functions that convert these values, which is the basis for functionality such as scrubbing through the Level Editor or respawning the player at checkpoints.


The custom level editor slider scrubs through the playback position of the audio event, which is then converted to the world space position that the player should move to.

Timing Logic


While directional slashes do use hitboxes to collide with enemies, they act as a "trigger" to check the accuracy of the hit compared to the enemy's actual playback position.



This is most notable for the Cue Enemies, which are a unique enemy type that must be hit multiple times to the beat. Every time a Cue Enemy is encountered, the Beat Manager calculates each of the timing windows, ensuring responsive, music-synced feedback.

2.5D Camera System

Beat, Heart, Beat contains both 2D and 3D elements, separated between the play space and backgrounds respectively. The primary benefits of this system include:

Camera Stacking


The Camera System utilizes Unity Camera Stacking to overlay the orthographic 2D camera over the perspective 3D camera.


The background camera can pan, pulse, and zoom independently of the gameplay camera

Perspective-Based Canvases


One of the most satisfying uses of the overlay system is the 3D, perspective-based UI used in the level select. 


The biggest challenge here was getting the world space canvas to render on top of certain UI elements. Since UI elements are always rendered over other elements on the same camera, this was another great place to implement camera stacking:

The level select carousel exists in physical space, but is rendered nicely into the UI using camera stacking

Workflow Improvements

Persistent Song Data


Throughout early development, designers had to manually input each field for a level file's data, even if it was a difficulty variation of an existing song. This meant there was a large risk for inconsistent data, as well as the time cost of manual entry.


Solution:


I created a persistent ScriptableObject list that stores all information related to each song in a serializable list. Whenever a new difficulty file is created, it syncs the shared information if a level file with the same song name exists.


Additional benefits:

Managing Large File Sizes


While GitHub works great for most project files, large files such as .wavs, .mp4s, and generated audio banks are unfeasible for version control. This led to the additional workflow issue of requiring every contributor to generate their own banks through Wwise.


Solution:


Using the open source software Syncthing, I created a system to share large file sizes between contributors without risking the bandwidth limit of Git LFS or a manual download from a cloud-based service.


Once contributors have synced their directories to Syncthing, it will automatically add new files, including the generated audio banks.

Syncthing setup for Wwise project files

Misc. Front-End + Polish Contributions

Results Screen animation using LeanTween routines


Gameplay showcasing a few things I've worked on, including: