NasNas
An intuitive and beginner friendly 2D game framework for C++
App.hpp
1 
6 #pragma once
7 
8 
9 #include <array>
10 #include <functional>
11 #include <list>
12 #include <memory>
13 #include <stack>
14 #include <string>
15 #include <type_traits>
16 #include <vector>
17 
18 #include <SFML/Graphics/Color.hpp>
19 #include <SFML/Graphics/RenderTexture.hpp>
20 #include <SFML/System/Clock.hpp>
21 #include <SFML/System/Vector2.hpp>
22 #include <SFML/Window/Event.hpp>
23 
24 #include <NasNas/core/data/Config.hpp>
25 #include <NasNas/core/data/Rect.hpp>
26 #include <NasNas/core/data/ShaderHolder.hpp>
27 
28 #include <NasNas/core/AppState.hpp>
29 #include <NasNas/core/Camera.hpp>
30 #include <NasNas/core/Debug.hpp>
31 #include <NasNas/core/Scene.hpp>
32 #include <NasNas/core/Window.hpp>
33 
34 namespace ns {
35 
36  class StateMachineApp;
37  class StateStackApp;
38 
39  class App : public AppStateInterface, public ShaderHolder {
40  friend StateMachineApp;
41  friend StateStackApp;
42  public:
46  App();
47 
56  App(std::string title, sf::Vector2u resolution,
57  float scale=1.f,
58  int fps=Settings::user_config.frame_rate,
59  int ups=Settings::user_config.update_rate
60  );
61 
65  virtual ~App();
66 
72  auto getTitle() const -> const std::string&;
73 
79  auto getWindow() -> AppWindow&;
80 
81  auto getMousePosition() const -> sf::Vector2f;
82 
83  auto getMousePosition(Camera& cam) const -> sf::Vector2f;
84 
85  auto getTouchPosition(int finger) const -> sf::Vector2f;
86 
87  auto getTouchPosition(int finger, Camera& cam) const -> sf::Vector2f;
88 
89 
95  auto allScenes() -> std::list<Scene>&;
96 
103  auto getScene(const std::string& name) -> Scene&;
104 
110  auto allCameras() -> std::list<Camera>&;
111 
118  auto getCamera(const std::string& name) -> Camera&;
119 
123  void toggleFullscreen();
124 
125  auto getDt() const -> float;
126 
130  void run();
131 
132  protected:
133 
142  auto createScene(const std::string& name) -> Scene&;
143 
154  auto createCamera(
155  const std::string& name, int order,
156  const ns::IntRect& view={{0, 0}, sf::Vector2i(Settings::user_config.resolution)},
157  const ns::FloatRect& viewport={0, 0, 1.f, 1.f}
158  ) -> Camera&;
159 
169  auto createCamera(const std::string& name, int order, const ns::FloatRect& viewport) -> Camera&;
170 
174  void addDebugText(const std::string& label, const sf::Vector2f& position, const sf::Color& color = ns::DebugTextInterface::color);
175 
190  template<typename T>
191  void addDebugText(const std::string& label, T* var_address, const sf::Vector2f& position, const sf::Color& color = ns::DebugTextInterface::color);
192 
203  template<typename T>
204  void addDebugText(const std::string& label, std::function<T()> fn, const sf::Vector2f& position, const sf::Color& color = ns::DebugTextInterface::color);
205 
210  void sleep();
211 
216  void awake();
217 
226  void onEvent(const sf::Event& event) override;
227 
234  void update() override;
235 
236  private:
237  AppWindow m_window;
238  sf::RenderTexture m_renderer;
239  std::string m_title;
240  bool m_fullscreen;
241  int m_ups;
242  int m_fps;
243  sf::Clock m_fps_clock;
244  float m_dt;
245  bool m_sleeping = false;
246 
247  std::list<Camera> m_cameras;
248  std::list<Scene> m_scenes;
249  std::vector<std::unique_ptr<DebugTextInterface>> m_debug_texts;
250  std::vector<sf::Vertex> m_debug_bounds;
251 
252  std::function<void(const sf::Event&)> m_cb_onevent=[](const sf::Event&){};
253  std::function<void()> m_cb_update=[]{};
254  std::function<void()> m_cb_prerender=[]{};
255 
263  static void storeInputs(const sf::Event& event);
264 
268  void render();
269 
270  void renderDebugBounds();
271  void storeDrawableDebugRects(const ns::FloatRect& drawable_bounds, Camera& cam,
272  const sf::FloatRect& render_bounds, sf::Vector2f& offset,
273  const sf::FloatRect& global_vport, const sf::FloatRect& local_vport);
274  };
275 
276  template<typename T>
277  void App::addDebugText(const std::string& label, T* var_address, const sf::Vector2f& position, const sf::Color& color) {
278  auto* dbg_txt = new DebugText<T>(label, var_address, position);
279  dbg_txt->setFillColor(color);
280  m_debug_texts.emplace_back(dbg_txt);
281  }
282 
283  template<typename T>
284  void App::addDebugText(const std::string& label, std::function<T()> fn, const sf::Vector2f& position, const sf::Color& color) {
285  auto* dbg_txt = new DebugText<T>(label, fn, position);
286  dbg_txt->setFillColor(color);
287  m_debug_texts.emplace_back(dbg_txt);
288  }
289 
290  class StateMachineApp : public App {
291  std::unique_ptr<AppState> m_state = nullptr;
292  public:
293  using App::App;
294  virtual void update() {}
295 
296  template <typename T, typename... Targs, typename = std::enable_if<std::is_base_of_v<AppState, T>>>
297  void setState(Targs... args) {
298  m_state.reset();
299  m_state = std::make_unique<T>(std::forward<Targs>(args)...);
300  m_state->setup();
301  m_cb_onevent = [&](const auto& event) { m_state->onEvent(event); };
302  m_cb_update = [&] { m_state->update(); };
303  m_cb_prerender = [&] { m_state->preRender(); };
304  }
305  };
306 
307  class StateStackApp : public App {
308  std::stack<std::unique_ptr<AppState>> m_state_stack;
309  public:
310  using App::App;
311  virtual void update() {}
312 
313  template <typename T, typename... Targs, typename = std::enable_if<std::is_base_of_v<AppState, T>>>
314  void pushState(Targs... args) {
315  m_state_stack.emplace(std::make_unique<T>(std::forward<Targs>(args)...));
316  m_state_stack.top()->setup();
317  m_cb_onevent = [&](const sf::Event& event) { m_state_stack.top()->onEvent(event); };
318  m_cb_update = [&] { m_state_stack.top()->update(); };
319  m_cb_prerender = [&] { m_state_stack.top()->preRender(); };
320  }
321 
322  inline void popState() {
323  if (m_state_stack.empty())
324  return;
325 
326  m_state_stack.pop();
327 
328  if (m_state_stack.empty()) {
329  m_cb_onevent = [&](const sf::Event& event) {};
330  m_cb_update = [&] {};
331  m_cb_prerender = [&] {};
332  } else {
333  m_state_stack.top()->setup();
334  m_cb_onevent = [&](const sf::Event& event) { m_state_stack.top()->onEvent(event); };
335  m_cb_update = [&] { m_state_stack.top()->update(); };
336  m_cb_prerender = [&] { m_state_stack.top()->preRender(); };
337  }
338  }
339  };
340 }
auto getWindow() -> AppWindow &
Get the AppWindow.
void sleep()
The App enters sleep mode, the App will not update. Used for Android when application runs in backgro...
DebugText evaluates a variable or a method and display the value on the AppWindow.
Definition: Debug.hpp:53
auto createCamera(const std::string &name, int order, const ns::IntRect &view={{0, 0}, sf::Vector2i(Settings::user_config.resolution)}, const ns::FloatRect &viewport={0, 0, 1.f, 1.f}) -> Camera &
Creates a Camera object and returns a reference to it.
auto getScene(const std::string &name) -> Scene &
Get a Scene by name.
static sf::Color color
set DebugText color
Definition: Debug.hpp:35
virtual void update()
App update method.
Definition: App.hpp:311
void onEvent(const sf::Event &event) override
Handles SFML events.
auto getTitle() const -> const std::string &
Returns the title of the App.
void toggleFullscreen()
Toggle fullscreen display.
Definition: App.hpp:39
auto createScene(const std::string &name) -> Scene &
Creates a Scene object and returns a reference to it.
App()
Contructs an App from ns::Config configuration.
auto allCameras() -> std::list< Camera > &
Get all the Camera objects created within the App.
void addDebugText(const std::string &label, const sf::Vector2f &position, const sf::Color &color=ns::DebugTextInterface::color)
Creates a label only DebugText object and render it on the AppWindow directly.
auto getCamera(const std::string &name) -> Camera &
Get a Camera by name.
virtual ~App()
Delete all Scene, Camera and DebugText objects and free loaded resources.
void update() override
App update method.
virtual void update()
App update method.
Definition: App.hpp:294
void run()
Starts the game loop.
void awake()
The App awakes from sleep mode. Used for Android when application run as main process.
auto allScenes() -> std::list< Scene > &
Get all the Scene objects created within the App.