From 7662364f725c41ed1e5a1a59f84de81ffce0eb73 Mon Sep 17 00:00:00 2001 From: lauchmelder Date: Fri, 3 Jan 2025 16:13:34 +0100 Subject: [PATCH] add events --- CMakeLists.txt | 2 + quark/CMakeLists.txt | 5 + quark/src/quark/Application.cpp | 5 + quark/src/quark/Logger.hpp | 4 +- quark/src/quark/events/ApplicationEvent.hpp | 39 ++++++++ quark/src/quark/events/Event.hpp | 100 +++++++++++++++++++ quark/src/quark/events/KeyEvent.hpp | 49 ++++++++++ quark/src/quark/events/MouseEvent.hpp | 102 ++++++++++++++++++++ 8 files changed, 305 insertions(+), 1 deletion(-) create mode 100644 quark/src/quark/events/ApplicationEvent.hpp create mode 100644 quark/src/quark/events/Event.hpp create mode 100644 quark/src/quark/events/KeyEvent.hpp create mode 100644 quark/src/quark/events/MouseEvent.hpp diff --git a/CMakeLists.txt b/CMakeLists.txt index b8752dc..c6e94f5 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -2,6 +2,8 @@ cmake_minimum_required(VERSION 3.24) project(quark) +set(CMAKE_CXX_STANDARD 20) + include(FetchContent) set(SPDLOG_BUILD_SHARED ON CACHE INTERNAL "") diff --git a/quark/CMakeLists.txt b/quark/CMakeLists.txt index 60831e4..2c2dc22 100644 --- a/quark/CMakeLists.txt +++ b/quark/CMakeLists.txt @@ -15,6 +15,11 @@ target_sources(quark src/quark.hpp src/quark/Application.hpp src/quark/Logger.hpp + + src/quark/events/Event.hpp + src/quark/events/ApplicationEvent.hpp + src/quark/events/KeyEvent.hpp + src/quark/events/MouseEvent.hpp ) target_include_directories(quark INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/src) diff --git a/quark/src/quark/Application.cpp b/quark/src/quark/Application.cpp index d9dc027..ebc0db3 100644 --- a/quark/src/quark/Application.cpp +++ b/quark/src/quark/Application.cpp @@ -1,9 +1,14 @@ #include "Application.hpp" +#include "quark/Logger.hpp" +#include "quark/events/ApplicationEvent.hpp" #include namespace qk { void Application::Run() const { + WindowResizeEvent e(1200, 720); + QK_TRACE(e); + for(;;) { } diff --git a/quark/src/quark/Logger.hpp b/quark/src/quark/Logger.hpp index ac4f37b..bc18060 100644 --- a/quark/src/quark/Logger.hpp +++ b/quark/src/quark/Logger.hpp @@ -3,6 +3,7 @@ #include #include +#include namespace qk { class Logger { @@ -36,8 +37,9 @@ namespace qk { #ifndef NDEBUG # define QK_TRACE(...) ::qk::Logger::GetClientLogger()->trace(__VA_ARGS__) #else -# define QK_INFO(...) ::qk::Logger::GetClientLogger()->info(__VA_ARGS__) +# define QK_TRACE #endif +#define QK_INFO(...) ::qk::Logger::GetClientLogger()->info(__VA_ARGS__) #define QK_WARN(...) ::qk::Logger::GetClientLogger()->warn(__VA_ARGS__) #define QK_ERROR(...) ::qk::Logger::GetClientLogger()->error(__VA_ARGS__) #define QK_CRITICAL(...) ::qk::Logger::GetClientLogger()->critical(__VA_ARGS__) diff --git a/quark/src/quark/events/ApplicationEvent.hpp b/quark/src/quark/events/ApplicationEvent.hpp new file mode 100644 index 0000000..d57bded --- /dev/null +++ b/quark/src/quark/events/ApplicationEvent.hpp @@ -0,0 +1,39 @@ +#pragma once +// IWYU pragma: private + +#include "Event.hpp" +#include + +namespace qk { + class WindowResizeEvent : + public Event + { + public: + WindowResizeEvent(unsigned int width, unsigned int height) : + width(width), height(height) + { } + + inline unsigned int GetWidth() const { return width; } + inline unsigned int GetHeight() const { return height; } + + virtual operator std::string() const override { + std::stringstream ss; + ss << "WindowResizEvent: " << width << ", " << height; + return ss.str(); + } + + EVENT_CLASS_TYPE(WindowResize) + EVENT_CLASS_CATEGORY(EventCategoryApplication) + + private: + unsigned int width, height; + }; + + class WindowCloseEvent : + public Event + { + public: + EVENT_CLASS_TYPE(WindowClose) + EVENT_CLASS_CATEGORY(EventCategoryApplication) + }; +} diff --git a/quark/src/quark/events/Event.hpp b/quark/src/quark/events/Event.hpp new file mode 100644 index 0000000..87a40ff --- /dev/null +++ b/quark/src/quark/events/Event.hpp @@ -0,0 +1,100 @@ +#pragma once +// IWYU pragma: private + +#include +#include +#include +#include +#include + +namespace qk { + enum class EventType { + None = 0, + + WindowClose, + WindowResize, + WindowFocus, + WindowLostFocus, + WindowMoved, + + KeyPressed, + KeyReleased, + + MouseButtonPressed, + MouseButtonReleased, + MouseMoved, + MouseScrolled + }; + + enum EventCategory { + None = 0, + + EventCategoryApplication = 0x1, + EventCategoryInput = 0x2, + EventCategoryKeyboard = 0x4, + EventCategoryMouse = 0x8, + EventCategoryMouseButton = 0x10, + }; + +#define EVENT_CLASS_TYPE(type) \ + static EventType GetStaticType() { return (EventType::type); } \ + virtual EventType GetEventType() const override { return GetStaticType(); } \ + virtual const char* GetName() const override { return #type; } + +#define EVENT_CLASS_CATEGORY(category) \ + virtual int GetCategoryFlags() const override { return category; } + + class Event { + friend class EventDispatcher; + + public: + virtual EventType GetEventType() const = 0; + virtual const char* GetName() const = 0; + virtual int GetCategoryFlags() const = 0; + + virtual operator std::string() const { + return GetName(); + } + + inline bool IsInCagegory(EventCategory category) const { + return (GetCategoryFlags() & category) != 0; + } + + protected: + bool handled = false; + }; + + class EventDispatcher { + template + using EventFn = std::function; + + public: + EventDispatcher(Event& event) : + event(event) + { } + + template + requires std::is_base_of_v + bool Dispatch(EventFn func) { + if (event.GetEventType() == T::GetStaticType()) { + event.handled = func(*(T*)&event); + return true; + } + + return false; + } + + private: + Event& event; + }; + + inline std::ostream& operator<<(std::ostream& os, const Event& event) { + os << std::string(event); + return os; + } + +} + +template +struct fmt::formatter, char>> : +fmt::ostream_formatter {}; diff --git a/quark/src/quark/events/KeyEvent.hpp b/quark/src/quark/events/KeyEvent.hpp new file mode 100644 index 0000000..4d9d0a2 --- /dev/null +++ b/quark/src/quark/events/KeyEvent.hpp @@ -0,0 +1,49 @@ +#pragma once +// IWYU pragma: private + +#include "Event.hpp" + +namespace qk { + + class KeyEvent : + public Event + { + public: + inline int GetKeyCode() const { return keyCode; } + + EVENT_CLASS_CATEGORY(EventCategoryKeyboard | EventCategoryInput) + + protected: + KeyEvent(int keyCode) : + keyCode(keyCode) + { } + + int keyCode; + }; + + class KeyPressedEvent + : public KeyEvent + { + public: + KeyPressedEvent(int keyCode, int repeatCount) : + KeyEvent(keyCode), repeatCount(repeatCount) + { } + + inline int GetRepeatCount() const { return repeatCount; } + + EVENT_CLASS_TYPE(KeyPressed) + private: + int repeatCount; + }; + + class KeyReleasedEvent + : public KeyEvent + { + public: + KeyReleasedEvent(int keyCode) : + KeyEvent(keyCode) + { } + + EVENT_CLASS_TYPE(KeyReleased) + }; +} diff --git a/quark/src/quark/events/MouseEvent.hpp b/quark/src/quark/events/MouseEvent.hpp new file mode 100644 index 0000000..c3ba8dd --- /dev/null +++ b/quark/src/quark/events/MouseEvent.hpp @@ -0,0 +1,102 @@ +#pragma once +// IWYU pragma: private + +#include "Event.hpp" +#include + +namespace qk { + class MouseMovedEvent : + public Event + { + public: + MouseMovedEvent(float x, float y) : + mouseX(x), mouseY(y) + { } + + inline float GetX() const { return mouseX; } + inline float GetY() const { return mouseY; } + + virtual operator std::string() const override { + std::stringstream ss; + ss << "MouseMovedEvent: " << mouseX << ", " << mouseY; + return ss.str(); + } + + EVENT_CLASS_TYPE(MouseMoved) + EVENT_CLASS_CATEGORY(EventCategoryMouse | EventCategoryInput) + + private: + float mouseX, mouseY; + }; + + class MouseScrolledEvent : + public Event + { + public: + MouseScrolledEvent(float offsetX, float offsetY) : + offsetX(offsetX), offsetY(offsetY) + { } + + virtual operator std::string() const override { + std::stringstream ss; + ss << "MouseScrolledEvent: " << offsetX << ", " << offsetY; + return ss.str(); + } + + EVENT_CLASS_TYPE(MouseScrolled) + EVENT_CLASS_CATEGORY(EventCategoryMouse | EventCategoryInput) + + private: + float offsetX, offsetY; + }; + + class MouseButtonEvent : + public Event + { + public: + inline int GetMouseButton() const { return button; } + + EVENT_CLASS_CATEGORY(EventCategoryMouseButton | EventCategoryInput) + + protected: + MouseButtonEvent(int button) : + button(button) + { } + + int button; + }; + + class MouseButtonPressedEvent : + public MouseButtonEvent + { + public: + MouseButtonPressedEvent(int button) : + MouseButtonEvent(button) + { } + + virtual operator std::string() const override { + std::stringstream ss; + ss << "MouseButtonPressedEvent: " << button; + return ss.str(); + } + + EVENT_CLASS_TYPE(MouseButtonPressed) + }; + + class MouseButtonReleasedEvent : + public MouseButtonEvent + { + public: + MouseButtonReleasedEvent(int button) : + MouseButtonEvent(button) + { } + + virtual operator std::string() const override { + std::stringstream ss; + ss << "MouseButtonReleasedEvent: " << button; + return ss.str(); + } + + EVENT_CLASS_TYPE(MouseButtonReleased) + }; +}