Project info

1

8-9 weeks

Unreal Engine 5

Introduction

Zenithar is a first-person sci-fi game. For this project, I wanted to focus primarily on UI design. I built the entire game in Unreal Engine 5, relying solely on C++, no Blueprint scripting at all. Throughout development, I learned a lot and sharpened my existing skills, making this a valuable experience.


I used 3D assets from the UE5 Marketplace, while all 2D assets were created from scratch in the editor. The helmet in the HUD was a collaboration between me and a family member.

The core gameplay revolves around completing objectives by following the compass and carrying out the tasks given. As of now, the only challenge is figuring out how to unlock a sealed door.


In the future, I plan to expand the game by adding more objectives, providing players with additional challenges and tasks to complete.


WIP

For my graduation project at Futuregames I am creating an inventory system similar to games like Resident Evil and Diablo.

Game mode, Player character & Player Controller

The first thing I did in this project was set up the game mode, player character and player controller. I designed the game mode to use different game states, such as the title screen, main menu, and in-game state. 

- Enum class with all game states

These states also handle the UI, when I for example press the "press to start" button on the title screen, it switches the game state to the main menu, hides the title screen UI, and add the menu UI to the viewport. The same logic applies while in the in-game state which is applied after the player presses "Play" in the main menu. If the player presses the ESC key while in the in-game state, the state changes to the pause menu, which pauses the game and brings up the pause menu UI.

For the player character, I implemented basic movement mechanics, including walking, running, jumping, and looking around. Later, I added inputs for viewing the current objective, interacting and for canceling interactions, such as closing a log or backing out of a keypad interface. The player character also has two components for health and stamina, which I created as separate scripts and communicate both with the player chracter and the player hud. 

In the player controller script, I set up the ESC key to pause the game by changing the game state, but only if the player is currently in the in-game state. I also added public functions for enabling and disabling the mouse pointer when needed.

Player HUD

After completing the menus, I moved on to designing the Player HUD, which includes several key features:

  • Health Bar – Displays the player's current health.
  • Stamina Bar – Represents the player's stamina levels.
  • Compass – Guides the player by showing the direction of the objective point actor.
  • Objective Box – Displays the current objective when a new one is assigned or when the player manually checks it. (Although not technically part of the Player HUD widget, it is added to the viewport alongside it.)

The Health Bar and Stamina Bar interact with the components attached to the player character using delegates, ensuring they update in real-time based on player status. The Compass provides directional guidance by tracking the location of the Objective Point Actor, while the Objective Box appears dynamically whenever a new objective is assigned or when the player presses the designated input to view their current objective.

Health & stamina bars

Both the Health Bar and Stamina Bar receive broadcasts from the respective Health and Stamina components attached to the player. These broadcasts provide the current health and stamina values, which are then converted into percentage values. The progress bars are then updated accordingly to reflect the player's current status in real-time.

- Player HUD

- Health & Stamina bars

Compass

The Compass is a single wide image with its center set as the zero point, representing the direction the player is facing. It rotates based on the player's camera rotation, ensuring it always aligns with the player's view.

To track objectives, I created a separate Objective Point Actor that the compass follows. The tracking works by calculating the objective’s position relative to the player using:


UKismetMathLibrary::FindLookAtRotation(PlayerCameraLocation, ObjectiveLocation);


This function determines the rotation needed to face the objective. The next step is to calculate the objective’s position on the compass by normalizing this rotation relative to the player’s current rotation.

The direction is then determined using the formula:


 (CameraRightVector.DOT(RotationVector) / CameraForwardVector.DOT(RotationVector) 


This value determines the Canvas Slot Position of the objective marker on the compass. To apply it, I cast the marker’s slot to a UCanvasPanelSlot and update its position dynamically.

This approach ensures that the objective marker smoothly moves along the compass, accurately reflecting the direction of the objective relative to the player's viewpoint.

- Compass with objective marker

To ensure the objective marker only appears when the objective is in front of the player, I calculated the angle between the objective and the player's forward direction using the arc cosine (acos) of the dot product between:

  • The normalized location of the objective actor
  • The normalized forward vector of the camera

If this angle indicates that the objective is behind the player, the marker is hidden from the compass.

This approach ensures that the objective marker smoothly moves along the compass while also preventing it from appearing when the objective is behind the player.

Objective

The objective box is a seperate UI widget that gets added by the game mode when called for. When the player overlaps a objective point the player gets a new objective by setting a game instance sub system that i called objective manager and then setting the current objective string to the new objective. Then after that it gets set in the objective box add the widget gets added to the viewport, which after a view seconds get set to hidden to hide the box. It can then be shown again if the player presses the input for showing the current objective. The objective can get changed everytime the player overlaps a objective point depening on if it has been set up like that. The objective point have been set up in a way so that it can give a new objective and what the new objective is supposed to be.

- Objective box

Log & Keypad

For the first objective in my game, I created a log and a keypad, both utilizing diegetic UI. Unlike traditional UI elements like menus or the first-person HUD, diegetic UI exists within the 3D world space, making it feel more immersive and integrated into the game environment.

The log serves as a clue for the player—it must be found and read to discover the code needed to unlock a door controlled by the keypad. The keypad allows the player to input a code, and if the correct sequence is entered, it triggers the door to open.

This approach not only keeps the objective engaging but also reinforces environmental storytelling by embedding the UI elements directly into the game world.

For the keypad system, I created a code generator script that generates a random 4-digit code each time the game is played. Both the keypad and the log reference this code:

  • The keypad uses it to verify the player's input, unlocking the door if the correct code is entered.
  • The log displays the code as a clue for the player to find and use.

This dynamic system ensures that the code changes with each playthrough, adding an extra layer of challenge and replayability.

Log

The log is a custom actor created in code, consisting of several key components:

  • StaticMeshComponent – Represents the physical log object.
  • PointLightComponent – Emits a green light to make the log easier to spot.
  • WidgetComponent (Text Box) – Displays the log's text when interacted with.
  • WidgetComponent (Input Prompt) – Shows a prompt when the log is interactable.
  • BoxComponent – Detects when the player enters or leaves its interaction range.

Interaction Behavior

  • When the player overlaps with the BoxComponent, the log becomes interactable, and the input prompt appears.
  • Pressing the interact key displays the text box, while the input prompt is hidden.
  • If the player moves away while the log is still open, it automatically closes.
  • The cancel key can also be used to close the log manually.
  • Once closed, the input prompt reappears, allowing the player to interact again or leave the area.

This system ensures a smooth and intuitive interaction flow while maintaining immersion in the game world.

- Log actor

Keypad

The keypad is a custom actor created in code, consisting of the following components:

  • StaticMeshComponent – Represents the physical keypad object.
  • BoxComponent – Detects when the player enters or leaves its interaction range.
  • WidgetComponent (Keypad UI) – Displays the keypad interface, including buttons and the screen.
  • WidgetComponent (Input Prompt) – Shows a prompt when the keypad is interactable.
  • CameraComponent – Used to provide a focused view of the keypad during interaction.

Interaction Flow

  • When the player enters the interaction range (overlapping the BoxComponent), the keypad becomes interactable, and the input prompt appears.

  • Pressing the interact key triggers a camera switch using:


PlayerController->SetViewTargetWithBlend(Target, BlendTime);

This smoothly transitions the player's view to the keypad.


  • The keypad UI then appears, and the mouse cursor is enabled, allowing the player to interact with the buttons.

  • If the wrong code is entered, an error message is displayed, prompting the player to try again or search for the correct code.

  • If the correct code is entered, the controlled door unlocks, allowing the player to pass through.

This system ensures an immersive, interactive experience while maintaining smooth gameplay flow.

- Keypad actor