diff --git a/src/dawn/assert/assert.hpp b/src/dawn/assert/assert.hpp index a5fe4a42..223b4b06 100644 --- a/src/dawn/assert/assert.hpp +++ b/src/dawn/assert/assert.hpp @@ -84,6 +84,30 @@ void assertTrueImplement( map.find(key) != map.end(), __VA_ARGS__ \ ) +/** + * Asserts that a given value has a specific flag turned off. + * + * @param value Value to check. + * @param flag Flag to check for. + * @param message Message (sprintf format) to send to the logger. + * @param args Optional TParam args for the sprintf message to accept. + */ +#define assertFlagOff(value, flag, ...) assertTrue( \ + (value & flag) == 0, __VA_ARGS__ \ +) + +/** + * Asserts that a given value has a specific flag turned on. + * + * @param value Value to check. + * @param flag Flag to check for. + * @param message Message (sprintf format) to send to the logger. + * @param args Optional TParam args for the sprintf message to accept. + */ +#define assertFlagOn(value, flag, ...) assertTrue( \ + (value & flag) == flag, __VA_ARGS__ \ +) + /** * Asserts that the current code is deprecated and should not be used anymore. * @deprecated diff --git a/src/dawn/component/display/CMakeLists.txt b/src/dawn/component/display/CMakeLists.txt index e45b01b1..19661266 100644 --- a/src/dawn/component/display/CMakeLists.txt +++ b/src/dawn/component/display/CMakeLists.txt @@ -6,4 +6,5 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE Camera.cpp + MeshRenderer.cpp ) \ No newline at end of file diff --git a/src/dawn/component/display/Camera.cpp b/src/dawn/component/display/Camera.cpp index ddd673b5..57099cb7 100644 --- a/src/dawn/component/display/Camera.cpp +++ b/src/dawn/component/display/Camera.cpp @@ -3,6 +3,7 @@ // This software is released under the MIT License. // https://opensource.org/licenses/MIT +#include "assert/assert.hpp" #include "Camera.hpp" using namespace Dawn; @@ -11,6 +12,11 @@ void Camera::onInit() { std::cout << "Camera" << std::endl; } +void Camera::onDispose() { + std::cout << "~Camera" << std::endl; + // renderTarget = nullptr; +} + glm::mat4 Camera::getProjection() { switch(this->type) { case CameraType::ORTHOGONAL: @@ -34,4 +40,8 @@ glm::mat4 Camera::getProjection() { assertUnreachable("Invalid Camera Type!"); return glm::mat4(1.0f); +} + +float_t Camera::getAspect() { + return 1.0f; } \ No newline at end of file diff --git a/src/dawn/component/display/Camera.hpp b/src/dawn/component/display/Camera.hpp index ce5f6dd7..e42cdf9d 100644 --- a/src/dawn/component/display/Camera.hpp +++ b/src/dawn/component/display/Camera.hpp @@ -28,6 +28,7 @@ namespace Dawn { // std::shared_ptr renderTarget; void onInit() override; + void onDispose() override; /** * Returns the aspect ratio that the camera is using. In future I may diff --git a/src/dawn/component/display/MeshRenderer.cpp b/src/dawn/component/display/MeshRenderer.cpp new file mode 100644 index 00000000..a109e806 --- /dev/null +++ b/src/dawn/component/display/MeshRenderer.cpp @@ -0,0 +1,16 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "MeshRenderer.hpp" + +using namespace Dawn; + +void MeshRenderer::onInit() { + +} + +void MeshRenderer::onDispose() { + mesh = nullptr; +} \ No newline at end of file diff --git a/src/dawn/component/display/MeshRenderer.hpp b/src/dawn/component/display/MeshRenderer.hpp index 61a815e6..2778ee97 100644 --- a/src/dawn/component/display/MeshRenderer.hpp +++ b/src/dawn/component/display/MeshRenderer.hpp @@ -4,6 +4,7 @@ // https://opensource.org/licenses/MIT #pragma once +#include "display/mesh/Mesh.hpp" #include "scene/SceneItem.hpp" namespace Dawn { @@ -12,5 +13,6 @@ namespace Dawn { std::shared_ptr mesh; void onInit() override; + void onDispose() override; }; } \ No newline at end of file diff --git a/src/dawn/display/CMakeLists.txt b/src/dawn/display/CMakeLists.txt new file mode 100644 index 00000000..63a05219 --- /dev/null +++ b/src/dawn/display/CMakeLists.txt @@ -0,0 +1,15 @@ +# Copyright (c) 2022 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +# Sources +target_sources(${DAWN_TARGET_NAME} + PRIVATE + Color.cpp +) + +# Subdirs +# add_subdirectory(font) +# add_subdirectory(mesh) +# add_subdirectory(shader) \ No newline at end of file diff --git a/src/dawn/display/Color.cpp b/src/dawn/display/Color.cpp new file mode 100644 index 00000000..faa830b6 --- /dev/null +++ b/src/dawn/display/Color.cpp @@ -0,0 +1,82 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "Color.hpp" + +using namespace Dawn; + +struct Color Color::fromString(const std::string str) { + // Convert to lowercase + auto lower = stringToLowercase(str); + + if(stringIncludes(lower, "cornflower")) { + return COLOR_CORNFLOWER_BLUE; + + } else if(stringIncludes(lower, "magenta")) { + return COLOR_MAGENTA; + + } else if(stringIncludes(lower, "white")) { + return COLOR_WHITE; + + } else if(stringIncludes(lower, "black")) { + return COLOR_BLACK; + + } else if(stringIncludes(lower, "red")) { + return COLOR_RED; + + } else if(stringIncludes(lower, "green")) { + return COLOR_GREEN; + + } else if(stringIncludes(lower, "blue")) { + return COLOR_BLUE; + + } else if(stringIncludes(lower, "transparent")) { + return COLOR_TRANSPARENT; + } + + // Hex code? + if(lower[0] == '#') { + // Remove the hash + lower = lower.substr(1); + + // Convert to RGB + if(lower.length() == 3) { + // Convert to 6 digit hex + lower = lower[0] + lower[0] + lower[1] + lower[1] + lower[2] + lower[2]; + } + + // Convert to RGB + return { + (float_t)std::stoi(lower.substr(0, 2), nullptr, 16) / 255.0f, + (float_t)std::stoi(lower.substr(2, 2), nullptr, 16) / 255.0f, + (float_t)std::stoi(lower.substr(4, 2), nullptr, 16) / 255.0f, + 1.0f + }; + } + + // Split by comma + auto splitByComma = stringSplit(str, ","); + if(splitByComma.size() == 3) { + // RGB + return { + (float_t)std::stof(splitByComma[0]), + (float_t)std::stof(splitByComma[1]), + (float_t)std::stof(splitByComma[2]), + 1.0f + }; + } else if(splitByComma.size() == 4) { + // RGBA + return { + (float_t)std::stof(splitByComma[0]), + (float_t)std::stof(splitByComma[1]), + (float_t)std::stof(splitByComma[2]), + (float_t)std::stof(splitByComma[3]) + }; + } + + // TODO: Parse other kinds of colors + assertUnreachable("Failed to find a color match for %s", str); + return {}; +} \ No newline at end of file diff --git a/src/dawn/display/Color.hpp b/src/dawn/display/Color.hpp new file mode 100644 index 00000000..f00b27f9 --- /dev/null +++ b/src/dawn/display/Color.hpp @@ -0,0 +1,89 @@ +// Copyright (c) 2022 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "dawnlibs.hpp" + +namespace Dawn { + struct ColorU8 { + uint8_t r, g, b, a; + }; + + #pragma pack(push, 4) + struct Color final { + /** + * Returns a color from a string. + * + * @param str String to parse. + * @return Color parsed. + */ + static struct Color fromString(const std::string str); + + float_t r, g, b, a; + + const struct Color& operator = (const struct Color &val) { + this->r = val.r; + this->g = val.g; + this->b = val.b; + this->a = val.a; + return *this; + } + + struct Color operator * (const float_t &x) { + return { + r * x, + g * x, + b * x, + a * x + }; + } + + struct Color operator - (const struct Color &color) { + return { + r - color.r, + g - color.g, + b - color.b, + a - color.a + }; + } + + struct Color operator + (const struct Color &color) { + return { + r + color.r, + g + color.g, + b + color.b, + a + color.a + }; + } + + const bool_t operator == (const struct Color &other) { + return r == other.r && g == other.g && b == other.b && a == other.a; + } + + operator struct ColorU8() const { + return { + (uint8_t)(r * 255), + (uint8_t)(g * 255), + (uint8_t)(b * 255), + (uint8_t)(a * 255) + }; + } + }; + #pragma pack(pop) + + #define COLOR_DEF(r,g,b,a) { r, g, b, a } + #define COLOR_WHITE COLOR_DEF(1.0f, 1.0f, 1.0f, 1.0f) + #define COLOR_RED COLOR_DEF(1.0f, 0, 0, 1.0f) + #define COLOR_GREEN COLOR_DEF(0, 1.0f, 0, 1.0f) + #define COLOR_BLUE COLOR_DEF(0, 0, 1.0f, 1.0f) + #define COLOR_BLACK COLOR_DEF(0, 0, 0, 1.0f) + #define COLOR_MAGENTA COLOR_DEF(1.0f, 0, 1.0f, 1.0f) + #define COLOR_DARK_GREY COLOR_DEF(0.2f, 0.2f, 0.2f, 1.0f) + #define COLOR_LIGHT_GREY COLOR_DEF(0.8f, 0.8f, 0.8f, 1.0f) + #define COLOR_CORNFLOWER_BLUE COLOR_DEF(0.4f, 0.6f, 0.9f, 1.0f) + #define COLOR_WHITE_TRANSPARENT COLOR_DEF(1.0f, 1.0f, 1.0f, 0.0f) + #define COLOR_BLACK_TRANSPARENT COLOR_DEF(0.0f, 0.0f, 0.0f, 0.0f) + #define COLOR_TRANSPARENT COLOR_WHITE_TRANSPARENT +} \ No newline at end of file diff --git a/src/dawn/display/RenderTarget.hpp b/src/dawn/display/RenderTarget.hpp new file mode 100644 index 00000000..ddee46ad --- /dev/null +++ b/src/dawn/display/RenderTarget.hpp @@ -0,0 +1,72 @@ +// Copyright (c) 2022 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "dawnlibs.hpp" + +enum RenderTargetClearFlag { + COLOR = 1 << 0, + DEPTH = 1 << 0 +}; + +namespace Dawn { + class RenderTarget { + public: + + /** + * Return the width of the render target. + * + * @return The width of the render target. + */ + virtual float_t getWidth() = 0; + + /** + * Return the height of the render target. + * + * @return The height of the render target. + */ + virtual float_t getHeight() = 0; + + /** + * Returns the scale (as in pixel density) of the render target. This is + * typically 1.0f, but on high DPI displays this may be 2.0f or higher. + * + * @return The scale of the render target. + */ + virtual float_t getScale() = 0; + + /** + * Sets the clear color of the render target when the clear method for + * the color buffer is requested. + * + * @param color Color to use for the clear operation. + */ + virtual void setClearColor(const struct Color color) = 0; + + /** + * Request the existing data in the render target to be cleared out. We + * typically assume the render target can support multiple buffer types, + * so you can opt to only clear certain buffer types. + * + * @param clearFlags Flags to request what is going to be cleared. + */ + virtual void clear(const flag8_t clearFlags) = 0; + + /** + * Bind the render target for rendering to. The proceeding render requests + * will want to render to this render target directly. In future I may + * see if we can have multiple render targets bound at once to make this + * operation perform faster. + */ + virtual void bind() = 0; + + /** + * Destroys the render target. + */ + virtual RenderTarget() { + + } + }; +} \ No newline at end of file diff --git a/src/dawn/display/mesh/IMesh.hpp b/src/dawn/display/mesh/IMesh.hpp index 0427b99d..32df33a2 100644 --- a/src/dawn/display/mesh/IMesh.hpp +++ b/src/dawn/display/mesh/IMesh.hpp @@ -97,7 +97,6 @@ namespace Dawn { * Cleanup a previously initiated mesh. */ virtual ~IMesh() { - this->disposeBuffers(); } }; } \ No newline at end of file diff --git a/src/dawn/scene/SceneComponent.cpp b/src/dawn/scene/SceneComponent.cpp index b4179afc..e2c95e8d 100644 --- a/src/dawn/scene/SceneComponent.cpp +++ b/src/dawn/scene/SceneComponent.cpp @@ -3,15 +3,58 @@ // This software is released under the MIT License. // https://opensource.org/licenses/MIT +#include "assert/assert.hpp" +#include "util/Flag.hpp" #include "SceneComponent.hpp" using namespace Dawn; void SceneComponent::init(const std::shared_ptr item) { + assertFlagOff( + sceneComponentState, + SCENE_COMPONENT_STATE_INIT, + "SceneComponent is already initialized!" + ); + Flag::turnOn( + sceneComponentState, + SCENE_COMPONENT_STATE_INIT + ); this->item = item; this->onInit(); } +void SceneComponent::dispose() { + assertFlagOn( + sceneComponentState, + SCENE_COMPONENT_STATE_INIT, + "SceneComponent is not initialized!" + ); + assertFlagOff( + sceneComponentState, + SCENE_COMPONENT_STATE_DISPOSED, + "SceneComponent is already disposed!" + ); + Flag::turnOn( + sceneComponentState, + SCENE_COMPONENT_STATE_DISPOSED + ); + this->onDispose(); + this->item.reset(); +} + std::shared_ptr SceneComponent::getItem() { return this->item.lock(); +} + +SceneComponent::~SceneComponent() { + if(Flag::isOn( + sceneComponentState, + SCENE_COMPONENT_STATE_INIT + )) { + assertFlagOn( + sceneComponentState, + SCENE_COMPONENT_STATE_DISPOSED, + "SceneComponent is initialized but was not properly disposed!" + ); + } } \ No newline at end of file diff --git a/src/dawn/scene/SceneComponent.hpp b/src/dawn/scene/SceneComponent.hpp index 1b4d4e1c..7bfe0844 100644 --- a/src/dawn/scene/SceneComponent.hpp +++ b/src/dawn/scene/SceneComponent.hpp @@ -6,12 +6,16 @@ #pragma once #include "dawnlibs.hpp" +#define SCENE_COMPONENT_STATE_INIT 0x01 +#define SCENE_COMPONENT_STATE_DISPOSED 0x02 + namespace Dawn { class SceneItem; class SceneComponent : std::enable_shared_from_this { private: std::weak_ptr item; + uint_fast8_t sceneComponentState = 0; protected: /** @@ -20,6 +24,12 @@ namespace Dawn { */ virtual void onInit() = 0; + /** + * Custom component listener that is invoked when the component is meant + * to dispose. + */ + virtual void onDispose() = 0; + public: /** * Initializes this scene component. @@ -28,11 +38,21 @@ namespace Dawn { */ void init(const std::shared_ptr item); + /** + * Disposes this scene component. + */ + void dispose(); + /** * Returns the scene item that this scene component belongs to. * * @return Reference to the scene item that this component belongs to. */ std::shared_ptr getItem(); + + /** + * Disposes this scene component. + */ + virtual ~SceneComponent(); }; } \ No newline at end of file diff --git a/src/dawn/scene/SceneItem.cpp b/src/dawn/scene/SceneItem.cpp index 84b110cb..98c97e94 100644 --- a/src/dawn/scene/SceneItem.cpp +++ b/src/dawn/scene/SceneItem.cpp @@ -24,5 +24,11 @@ std::shared_ptr SceneItem::getScene() { } SceneItem::~SceneItem() { - + std::for_each( + components.begin(), + components.end(), + [](auto &component) { + component->dispose(); + } + ); } \ No newline at end of file diff --git a/src/dawn/scene/item/SceneItemComponents.cpp b/src/dawn/scene/item/SceneItemComponents.cpp index 9c6f70f3..481b1206 100644 --- a/src/dawn/scene/item/SceneItemComponents.cpp +++ b/src/dawn/scene/item/SceneItemComponents.cpp @@ -16,6 +16,7 @@ void SceneItemComponents::removeComponent( ) { auto it = std::find(components.begin(), components.end(), component); if(it == components.end()) return; //Not found? + it->get()->dispose(); components.erase(it); } diff --git a/src/dawn/util/Flag.hpp b/src/dawn/util/Flag.hpp new file mode 100644 index 00000000..9c5c3eb5 --- /dev/null +++ b/src/dawn/util/Flag.hpp @@ -0,0 +1,32 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "dawnlibs.hpp" + +namespace Dawn { + class Flag final { + public: + template + static void turnOn(T &flag, const T check) { + flag |= check; + } + + template + static void turnOff(T &flag, const T check) { + flag &= ~check; + } + + template + static bool_t isOn(const T flag, const T check) { + return (flag & check) == check; + } + + template + static bool_t isOff(const T flag, const T check) { + return (flag & check) == 0; + } + }; +} \ No newline at end of file diff --git a/src/dawn/util/Math.hpp b/src/dawn/util/Math.hpp index 61a3f7d4..2221d11d 100644 --- a/src/dawn/util/Math.hpp +++ b/src/dawn/util/Math.hpp @@ -11,7 +11,7 @@ #define MATH_PI 3.1415926535897f namespace Dawn { - class Math { + class Math final { /** * Returns the largest of the two provided int32 numbers. * @@ -32,7 +32,7 @@ namespace Dawn { * @return Smaller of the two numbers. */ template - static T mathMin(T left, T right) { + static T min(T left, T right) { return left < right ? left : right; } @@ -46,7 +46,7 @@ namespace Dawn { * @return The value, or the closest clamped value. */ template - static T mathClamp(T val, T min, T max) { + static T clamp(T val, T min, T max) { return mathMin(mathMax(val, min), max); } @@ -58,7 +58,7 @@ namespace Dawn { * @return The absolute value (-value if value < 0) */ template - static T mathAbs(T value) { + static T abs(T value) { return value < 0 ? -value : value; } @@ -70,11 +70,11 @@ namespace Dawn { * @returns The modulo result. */ template - static inline T mathMod(T value, T modulo) { + static inline T mod(T value, T modulo) { return ((value % modulo) + modulo) % modulo; } - static inline float_t mathMod(float_t value, float_t modulo) { + static inline float_t fmod(float_t value, float_t modulo) { float_t n = fmod(value, modulo); return n; } @@ -85,7 +85,7 @@ namespace Dawn { * @param n Degrees to convert. * @returns The number in radians. */ - static float_t mathDeg2Rad(float_t degrees) { + static float_t deg2rad(float_t degrees) { return degrees * (MATH_PI / 180.0f); } @@ -94,7 +94,7 @@ namespace Dawn { * @param n Radians to convert. * @returns The number in degrees. */ - static float_t mathRad2Deg(float_t n) { + static float_t rad2deg(float_t n) { return (n * 180.0f) / MATH_PI; } @@ -104,7 +104,7 @@ namespace Dawn { * @return Rounded number. */ template - static T mathRound(float_t n) { + static T round(float_t n) { return (T)roundf(n); } @@ -114,7 +114,7 @@ namespace Dawn { * @return Rounded number. */ template - static T mathFloor(float_t n) { + static T floor(float_t n) { return (T)floorf(n); } }; diff --git a/src/dawnopengl/display/mesh/Mesh.hpp b/src/dawnopengl/display/mesh/Mesh.hpp index 618e0b01..227a090e 100644 --- a/src/dawnopengl/display/mesh/Mesh.hpp +++ b/src/dawnopengl/display/mesh/Mesh.hpp @@ -4,6 +4,7 @@ // https://opensource.org/licenses/MIT #pragma once +#include "dawnopengl.hpp" #include "display/mesh/IMesh.hpp" namespace Dawn { @@ -44,5 +45,7 @@ namespace Dawn { const int32_t start, const int32_t count ) override; + + ~Mesh(); }; } \ No newline at end of file