aboutsummaryrefslogtreecommitdiff
path: root/src/engine
diff options
context:
space:
mode:
Diffstat (limited to 'src/engine')
-rw-r--r--src/engine/components/statusline.cpp99
-rw-r--r--src/engine/components/statusline.hpp45
-rw-r--r--src/engine/engine.cpp15
-rw-r--r--src/engine/engine.hpp4
-rw-r--r--src/engine/graphics/component_renderer.cpp55
-rw-r--r--src/engine/graphics/component_renderer.hpp23
-rw-r--r--src/engine/graphics/scene.cpp15
-rw-r--r--src/engine/graphics/scene.hpp14
-rw-r--r--src/engine/graphics/statusline.cpp112
-rw-r--r--src/engine/graphics/statusline.hpp48
10 files changed, 268 insertions, 162 deletions
diff --git a/src/engine/components/statusline.cpp b/src/engine/components/statusline.cpp
new file mode 100644
index 0000000..ae0f3fa
--- /dev/null
+++ b/src/engine/components/statusline.cpp
@@ -0,0 +1,99 @@
+#include "statusline.hpp"
+
+#include "engine/data/vector2.hpp"
+#include "util/color.hpp"
+
+#include <fmt/color.h>
+
+#include <utility>
+
+StatusLine::StatusLine(std::shared_ptr<ComponentMatrix> component_matrix) noexcept
+ : _component_matrix(std::move(component_matrix)), _need_render(false)
+{
+ _component_matrix->fill(' ');
+}
+
+auto StatusLine::get() const noexcept -> const std::shared_ptr<ComponentMatrix> &
+{
+ return _component_matrix;
+}
+
+auto StatusLine::get_need_render() const noexcept -> bool
+{
+ return _need_render;
+}
+
+void StatusLine::set_need_render(bool need_render) noexcept
+{
+ _need_render = need_render;
+}
+
+auto StatusLine::get_foreground_color() const noexcept -> uint32_t
+{
+ return static_cast<uint32_t>(fmt::color::white);
+}
+
+auto StatusLine::get_background_color() const noexcept -> uint32_t
+{
+ return STATUSLINE_COLOR;
+}
+
+void StatusLine::set_status(
+ StatusLineSection section,
+ const std::string_view &status,
+ int32_t start) noexcept
+{
+ _clear_section(section, start);
+
+ auto section_start = _get_section_start_x(section);
+
+ auto pos = Vector2({.x = section_start + start, .y = 0});
+
+ for (const auto &character : status)
+ {
+ _component_matrix->set(pos, character);
+
+ pos += Vector2::right();
+ }
+
+ set_need_render(true);
+}
+
+void StatusLine::set_section_length(StatusLineSection section, int32_t length) noexcept
+{
+ _section_lengths[section] = length;
+}
+
+auto StatusLine::_get_section_start_x(StatusLineSection section) const noexcept -> int32_t
+{
+ int32_t section_start = 0;
+
+ auto section_index = static_cast<int32_t>(section);
+
+ while (section_index > 0)
+ {
+ const auto prev_section = static_cast<StatusLineSection>(section_index - 1);
+
+ section_start += static_cast<int32_t>(_section_lengths.at(prev_section));
+
+ section_index--;
+ }
+
+ return section_start;
+}
+
+void StatusLine::_clear_section(StatusLineSection section, int32_t start) noexcept
+{
+ auto section_start = _get_section_start_x(section);
+
+ auto pos = Vector2({.x = section_start + start, .y = 0});
+
+ auto section_length = _section_lengths[section];
+
+ for (auto index = 0; index < section_length - start; index++)
+ {
+ _component_matrix->set(pos, ' ');
+
+ pos += Vector2::right();
+ }
+}
diff --git a/src/engine/components/statusline.hpp b/src/engine/components/statusline.hpp
new file mode 100644
index 0000000..3fdb8d3
--- /dev/null
+++ b/src/engine/components/statusline.hpp
@@ -0,0 +1,45 @@
+#pragma once
+
+#include "interfaces/matrix.hpp"
+#include "interfaces/statusline.hpp"
+
+#include "engine/data/vector2.hpp"
+
+#include <memory>
+#include <string_view>
+#include <unordered_map>
+
+constexpr uint32_t STATUSLINE_COLOR = 0x1A1A1AU;
+
+class StatusLine : public IStatusLine
+{
+public:
+ explicit StatusLine(std::shared_ptr<ComponentMatrix> component_matrix) noexcept;
+
+ auto get() const noexcept -> const std::shared_ptr<ComponentMatrix> & override;
+
+ [[nodiscard]] auto get_need_render() const noexcept -> bool override;
+
+ void set_need_render(bool need_render) noexcept override;
+
+ auto get_foreground_color() const noexcept -> uint32_t override;
+
+ auto get_background_color() const noexcept -> uint32_t override;
+
+ void set_status(
+ StatusLineSection section,
+ const std::string_view &status,
+ int32_t start) noexcept override;
+
+ void set_section_length(StatusLineSection section, int32_t length) noexcept override;
+
+private:
+ std::shared_ptr<ComponentMatrix> _component_matrix;
+ std::unordered_map<StatusLineSection, int32_t> _section_lengths;
+ bool _need_render;
+
+ [[nodiscard]] auto _get_section_start_x(StatusLineSection section) const noexcept
+ -> int32_t;
+
+ void _clear_section(StatusLineSection section, int32_t start) noexcept;
+};
diff --git a/src/engine/engine.cpp b/src/engine/engine.cpp
index 4775fe4..f947a45 100644
--- a/src/engine/engine.cpp
+++ b/src/engine/engine.cpp
@@ -9,10 +9,12 @@
CLIGameEngine::CLIGameEngine(
IGameFactory game_factory,
ISceneFactory scene_factory,
+ IComponentRendererFactory component_renderer_factory,
std::shared_ptr<IUserInputObserver> user_input_observer,
std::shared_ptr<ICursorController> cursor_controller) noexcept
: _game_factory(std::move(game_factory)),
_scene_factory(std::move(scene_factory)),
+ _component_renderer_factory(std::move(component_renderer_factory)),
_user_input_observer(std::move(user_input_observer)),
_cursor_controller(std::move(cursor_controller))
{
@@ -22,6 +24,8 @@ void CLIGameEngine::start() noexcept
{
std::shared_ptr<IScene> scene = _scene_factory(_cursor_controller);
+ auto component_renderer = _component_renderer_factory(_cursor_controller);
+
scene->enter();
_cursor_controller->set_bounds(scene->size());
@@ -65,8 +69,17 @@ void CLIGameEngine::start() noexcept
game->on_update();
- last_update_time = std::chrono::system_clock::now();
+ for (auto [component, position] : scene->get_components())
+ {
+ if (component->get_need_render())
+ {
+ component_renderer->render(component, position);
+ component->set_need_render(false);
+ }
+ }
_user_input_observer->clear_currently_pressed();
+
+ last_update_time = std::chrono::system_clock::now();
}
}
diff --git a/src/engine/engine.hpp b/src/engine/engine.hpp
index 1562bbb..a274ed4 100644
--- a/src/engine/engine.hpp
+++ b/src/engine/engine.hpp
@@ -1,5 +1,6 @@
#pragma once
+#include "interfaces/component_renderer.hpp"
#include "interfaces/cursor.hpp"
#include "interfaces/engine.hpp"
#include "interfaces/game.hpp"
@@ -19,6 +20,7 @@ class CLIGameEngine : public ICLIGameEngine,
CLIGameEngine,
IGameFactory,
ISceneFactory,
+ IComponentRendererFactory,
IUserInputObserver,
ICursorController>
{
@@ -26,6 +28,7 @@ public:
CLIGameEngine(
IGameFactory game_factory,
ISceneFactory scene_factory,
+ IComponentRendererFactory component_renderer_factory,
std::shared_ptr<IUserInputObserver> user_input_observer,
std::shared_ptr<ICursorController> cursor_controller) noexcept;
@@ -34,6 +37,7 @@ public:
private:
IGameFactory _game_factory;
ISceneFactory _scene_factory;
+ IComponentRendererFactory _component_renderer_factory;
std::shared_ptr<IUserInputObserver> _user_input_observer;
std::shared_ptr<ICursorController> _cursor_controller;
diff --git a/src/engine/graphics/component_renderer.cpp b/src/engine/graphics/component_renderer.cpp
new file mode 100644
index 0000000..1ab4c08
--- /dev/null
+++ b/src/engine/graphics/component_renderer.cpp
@@ -0,0 +1,55 @@
+#include "component_renderer.hpp"
+
+#include "engine/escape.hpp"
+#include "util/color.hpp"
+
+#include <fmt/core.h>
+
+#include <iostream>
+#include <utility>
+
+ComponentRenderer::ComponentRenderer(
+ std::shared_ptr<ICursorController> cursor_controller) noexcept
+ : _cursor_controller(std::move(cursor_controller))
+{
+}
+
+void ComponentRenderer::render(
+ const std::shared_ptr<IComponent> &component,
+ const Vector2 &position) noexcept
+{
+ const auto previous_pos = _cursor_controller->where();
+
+ _cursor_controller->hide();
+ _cursor_controller->move_to(position);
+
+ const auto component_matrix = component->get();
+
+ const auto foreground_color = component->get_foreground_color();
+ const auto background_color = component->get_background_color();
+
+ fmt::print(
+ "{}{}",
+ get_background_esc_seq(background_color),
+ get_foreground_esc_seq(foreground_color));
+
+ for (const auto &row : *component_matrix)
+ {
+ for (const auto &col : row)
+ {
+ std::cout.put(col);
+ }
+
+ const auto current_pos = _cursor_controller->where();
+
+ _cursor_controller->move_to(
+ Vector2({.x = previous_pos.get_x(), .y = current_pos.get_y() - 1}));
+ }
+
+ fmt::print(RESET_ALL_MODES, fmt::arg("esc", ESC));
+
+ std::cout.flush();
+
+ _cursor_controller->move_to(previous_pos);
+ _cursor_controller->show();
+}
diff --git a/src/engine/graphics/component_renderer.hpp b/src/engine/graphics/component_renderer.hpp
new file mode 100644
index 0000000..4f53e07
--- /dev/null
+++ b/src/engine/graphics/component_renderer.hpp
@@ -0,0 +1,23 @@
+#pragma once
+
+#include "interfaces/component.hpp"
+#include "interfaces/component_renderer.hpp"
+#include "interfaces/cursor.hpp"
+
+#include "engine/data/vector2.hpp"
+
+#include <memory>
+
+class ComponentRenderer : public IComponentRenderer
+{
+public:
+ explicit ComponentRenderer(
+ std::shared_ptr<ICursorController> cursor_controller) noexcept;
+
+ void render(
+ const std::shared_ptr<IComponent> &component,
+ const Vector2 &position) noexcept override;
+
+private:
+ std::shared_ptr<ICursorController> _cursor_controller;
+};
diff --git a/src/engine/graphics/scene.cpp b/src/engine/graphics/scene.cpp
index 9f561f0..e0f4da6 100644
--- a/src/engine/graphics/scene.cpp
+++ b/src/engine/graphics/scene.cpp
@@ -8,7 +8,6 @@
#include <iostream>
#include <sys/ioctl.h>
-#include <utility>
Scene::Scene(
IMatrixFactory<MatrixElement> matrix_factory,
@@ -84,3 +83,17 @@ auto Scene::get_matrix() const noexcept -> const std::shared_ptr<IMatrix<MatrixE
{
return _matrix;
}
+
+void Scene::register_component(
+ const std::shared_ptr<IComponent> &component,
+ const Vector2 &position) noexcept
+{
+ _components.emplace_back(std::make_pair(component, position));
+}
+
+auto Scene::get_components() const noexcept
+ -> std::vector<std::pair<std::shared_ptr<IComponent>, Vector2>>
+{
+ return _components;
+}
+
diff --git a/src/engine/graphics/scene.hpp b/src/engine/graphics/scene.hpp
index 60b541a..c2b11e8 100644
--- a/src/engine/graphics/scene.hpp
+++ b/src/engine/graphics/scene.hpp
@@ -1,14 +1,19 @@
#pragma once
+#include "interfaces/component.hpp"
#include "interfaces/cursor.hpp"
#include "interfaces/matrix.hpp"
#include "interfaces/scene.hpp"
+#include "engine/data/vector2.hpp"
+
#include <fmt/core.h>
#include <memory>
#include <string_view>
#include <termios.h>
+#include <utility>
+#include <vector>
constexpr fmt::string_view ENABLE_ALT_BUFFER = "{esc}[?1049h";
constexpr fmt::string_view DISABLE_ALT_BUFFER = "{esc}[?1049l";
@@ -29,10 +34,19 @@ public:
[[nodiscard]] auto get_matrix() const noexcept
-> const std::shared_ptr<IMatrix<MatrixElement>> & override;
+ void register_component(
+ const std::shared_ptr<IComponent> &component,
+ const Vector2 &position) noexcept override;
+
+ [[nodiscard]] auto get_components() const noexcept
+ -> std::vector<std::pair<std::shared_ptr<IComponent>, Vector2>> override;
+
private:
std::shared_ptr<IMatrix<MatrixElement>> _matrix;
std::shared_ptr<ICursorController> _cursor_controller;
bool _is_shown;
std::shared_ptr<termios> _original_termios = nullptr;
+
+ std::vector<std::pair<std::shared_ptr<IComponent>, Vector2>> _components;
};
diff --git a/src/engine/graphics/statusline.cpp b/src/engine/graphics/statusline.cpp
deleted file mode 100644
index bc83993..0000000
--- a/src/engine/graphics/statusline.cpp
+++ /dev/null
@@ -1,112 +0,0 @@
-#include "statusline.hpp"
-
-#include "engine/escape.hpp"
-#include "util/color.hpp"
-
-#include <string>
-#include <utility>
-
-StatusLine::StatusLine(
- std::shared_ptr<ICursorController> cursor_controller,
- std::shared_ptr<IScene> scene) noexcept
- : _cursor_controller(std::move(cursor_controller)), _scene(std::move(scene))
-{
-}
-
-void StatusLine::initialize_background() noexcept
-{
- const auto previous_position = _move_to_statusline(0);
-
- auto background_color = get_background_esc_seq(STATUSBAR_COLOR);
-
- fmt::print("{}{}", background_color, std::string(_scene->size().get_width(), ' '));
- fmt::print(RESET_ALL_MODES, fmt::arg("esc", ESC));
-
- _move_back(previous_position);
-}
-
-void StatusLine::set_status(
- const StatusLineSection &section,
- const std::string_view &status,
- std::size_t start) noexcept
-{
- _clear_section(section, start);
-
- auto section_start = _get_section_start_x(section);
-
- const auto previous_position =
- _move_to_statusline(section_start + static_cast<int32_t>(start));
-
- auto background_color = get_background_esc_seq(STATUSBAR_COLOR);
-
- auto section_length = _sections_lengths[section];
-
- fmt::print("{}{}", background_color, status);
- fmt::print(RESET_ALL_MODES, fmt::arg("esc", ESC));
-
- _move_back(previous_position);
-}
-
-void StatusLine::set_section_length(
- const StatusLineSection &section,
- uint32_t length) noexcept
-{
- _sections_lengths[section] = length;
-}
-
-auto StatusLine::_move_to_statusline(int32_t x) noexcept -> Vector2
-{
- const auto previous_position = _cursor_controller->where();
-
- _cursor_controller->hide();
-
- _cursor_controller->move_to(Vector2({.x = x, .y = 0}));
-
- return previous_position;
-}
-
-void StatusLine::_move_back(Vector2 previous_position) noexcept
-{
- _cursor_controller->move_to(previous_position);
- _cursor_controller->show();
-}
-
-auto StatusLine::_get_section_start_x(const StatusLineSection &section) const noexcept
- -> int32_t
-{
- int32_t section_start = 0;
-
- auto section_index = static_cast<int32_t>(section);
-
- while (section_index > 0)
- {
- section_start += static_cast<int32_t>(
- _sections_lengths.at(StatusLineSection(section_index - 1)));
-
- section_index--;
- }
-
- return section_start;
-}
-
-void StatusLine::_clear_section(
- const StatusLineSection &section,
- std::size_t start) noexcept
-{
- auto section_start = _get_section_start_x(section);
-
- auto start_int32 = static_cast<int32_t>(start);
-
- const auto previous_position = _move_to_statusline(section_start + start_int32);
-
- auto background_color = get_background_esc_seq(STATUSBAR_COLOR);
-
- auto section_length = _sections_lengths[section];
-
- auto start_uint32 = static_cast<uint32_t>(start);
-
- fmt::print("{}{}", background_color, std::string(section_length - start_uint32, ' '));
- fmt::print(RESET_ALL_MODES, fmt::arg("esc", ESC));
-
- _move_back(previous_position);
-}
diff --git a/src/engine/graphics/statusline.hpp b/src/engine/graphics/statusline.hpp
deleted file mode 100644
index 6f0a4c2..0000000
--- a/src/engine/graphics/statusline.hpp
+++ /dev/null
@@ -1,48 +0,0 @@
-#pragma once
-
-#include "interfaces/cursor.hpp"
-#include "interfaces/scene.hpp"
-#include "interfaces/statusline.hpp"
-
-#include "engine/data/vector2.hpp"
-
-#include <fmt/core.h>
-
-#include <memory>
-#include <string_view>
-#include <unordered_map>
-
-constexpr uint32_t STATUSBAR_COLOR = 0x1A1A1AU;
-
-class StatusLine : public IStatusLine
-{
-public:
- StatusLine(
- std::shared_ptr<ICursorController> cursor_controller,
- std::shared_ptr<IScene> scene) noexcept;
-
- void initialize_background() noexcept override;
-
- void set_status(
- const StatusLineSection &section,
- const std::string_view &status,
- std::size_t start) noexcept override;
-
- void set_section_length(const StatusLineSection &section, uint32_t length) noexcept
- override;
-
-private:
- std::unordered_map<StatusLineSection, uint32_t> _sections_lengths;
-
- std::shared_ptr<ICursorController> _cursor_controller;
- std::shared_ptr<IScene> _scene;
-
- auto _move_to_statusline(int32_t x) noexcept -> Vector2;
-
- void _move_back(Vector2 previous_position) noexcept;
-
- [[nodiscard]] auto
- _get_section_start_x(const StatusLineSection &section) const noexcept -> int32_t;
-
- void _clear_section(const StatusLineSection &section, std::size_t start) noexcept;
-};