The Jotar2D Engine is a simple 2D game development engine in C++ that has been my learning ground for C++ programming and design patterns. The engine includes essential features for building and managing a 2D game:
The game loop runs in the following sequence:
Time Update -> Input -> Fixed Update Collision -> Fixed Update Components -> Update -> Late Update -> Sound Update -> Render -> Destroyed Objects Clean Up -> Input Clear Frame Events
The Scene Manager is a singleton that helps load and manage scenes. Currently, a Start() function must be called manually when loading a new scene, though I may change this in the future for better integration with the engine.
The engine’s component-based architecture, combined with a scene graph, enables easy customization and reuse of game object behaviors. Each object automatically includes a transform component, and additional components, like texture or collider, can be added as needed.
For example, a player object can be created with some components, and to remove them:
auto scene = Jotar::SceneManager::GetInstance().CreateScene("scene1"); // Creating the scene
auto playerObj = scene.CreateGameObject("Bomberman") // Creating the GameObject
playerObj->AddComponent<TextureComponent>( ...);
playerObj->AddComponent<ColliderComponent>(...);
if (playerObj->HasComponent<ColliderComponent>())
{
playerObj->RemoveComponent<ColliderComponent>();
}
The scene graph allows adding child objects through the SetParent(parent) function or by creating child objects directly from a parent object.
auto HUDObj = scene.CreateGameObject("HUD") // Creating the parent GameObject
// Method 1
auto playerTextObj = scene.CreateGameObject("PlayerText") // Creating the child GameObject
playerTextObj->SetParent(HUDObj,..); // Setting the parent
// Method 2
auto fpsCounterObj = HUDObj->CreateChildGameObject("fpsCounter"); // Creating the child GameObject and add it as child
// To remove a child
fpsCounterObj->SetParent(nullptr);
The input system, compatible with both Xinput for controllers and SDL for keyboard input, uses the pImpl pattern to separate the library specifics from the engine. The Input Manager enables objects to respond to controller or keyboard inputs, and also offers methods like IsKeyUp() and IsControllerButtonUp() for use throughout the game.
The Observer pattern is available in the engine, allowing components to respond to events. For example, a health display component can observe and react to damage events:
class HealthDisplayComponent final : public Component, public Observer<Event> { // Add the Observer with the eventType
// ... other functions
void OnNotify(const Event& event) override; // override the function
}
void Jotar::HealthDisplayComponent::OnNotify(const Event& event)
{
if (typeid(event)) == typeid(DamageHealthEvent))
{
// ...
}
}
The Collision Manager is a Singleton that tracks colliders in each scene. The observer pattern notifies the collider components when there is a Collision or trigger. The Collider Component itself also Notifies when the Trigger Begins and Ends to anyone that is listening. The Collision Manager also handles RayCasting with a threading pool
The CollisionManager.cpp FixedUpdate() handles the collisions.
See CollisionManager.cpp CodeThe sound system uses SDL_Mixer, managed through the pImpl pattern, with the Service Locator pattern enabling sounds to be queued and played using threads. Sounds are enqueued with a simple command and played in the Update() method. Here’s how a sound effect might be added and called:
See SoundSystem.cpp Code
SoundServiceLocator::GetSoundSystem().AddSound("Path", "name"); // To Add Sound
SoundServiceLocator::GetSoundSystem().Play("name"); // To play the Sound
The renderer, based on SDL, supports layering textures and using multiple cameras for split-screen. Each camera can be independently configured, although the layout of the camera’s view rectangle must be set up manually. HUD textures are rendered once, regardless of the number of cameras.
This singleton provides access to total time, FPS, and delta time across the game, eliminating the need to pass these values explicitly through update methods.