diff --git a/assets/maps/testmap.json b/assets/maps/testmap.json new file mode 100644 index 00000000..5e8a2556 --- /dev/null +++ b/assets/maps/testmap.json @@ -0,0 +1,8 @@ +{ + "name": "Test Map", + "layers": [ + { + + } + ] +} \ No newline at end of file diff --git a/assets/prefabs/npc.json b/assets/prefabs/npc.json index 3b9d169a..a4a919a3 100644 --- a/assets/prefabs/npc.json +++ b/assets/prefabs/npc.json @@ -9,6 +9,9 @@ }, "components": { + "mesh": { + "type": "QuadMesh" + }, "material": { "type": "SimpleTexturedMaterial", "texture": "rosa" diff --git a/assets/prefabs/rosa.json b/assets/prefabs/rosa.json index 671178ca..034eee6c 100644 --- a/assets/prefabs/rosa.json +++ b/assets/prefabs/rosa.json @@ -9,6 +9,9 @@ }, "components": { + "mesh": { + "type": "QuadMesh" + }, "material": { "type": "SimpleTexturedMaterial", "texture": "rosa" diff --git a/src/dawn/component/display/mesh/CMakeLists.txt b/src/dawn/component/display/mesh/CMakeLists.txt index 77b7959b..27b06a1c 100644 --- a/src/dawn/component/display/mesh/CMakeLists.txt +++ b/src/dawn/component/display/mesh/CMakeLists.txt @@ -5,5 +5,7 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE + MeshComponent.cpp CubeMeshComponent.cpp + QuadMeshComponent.cpp ) \ No newline at end of file diff --git a/src/dawn/component/display/mesh/CubeMeshComponent.cpp b/src/dawn/component/display/mesh/CubeMeshComponent.cpp index c862bc42..0b019a3f 100644 --- a/src/dawn/component/display/mesh/CubeMeshComponent.cpp +++ b/src/dawn/component/display/mesh/CubeMeshComponent.cpp @@ -5,26 +5,24 @@ #include "CubeMeshComponent.hpp" #include "display/mesh/CubeMesh.hpp" -#include "component/display/MeshRenderer.hpp" +#include "util/JSON.hpp" using namespace Dawn; -void CubeMeshComponent::onInit() { - if(!mesh) mesh = std::make_shared(); +void CubeMeshComponent::meshBuffer(std::shared_ptr mesh) { mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT); CubeMesh::buffer( - mesh, glm::vec3(-0.5f, -0.5f, -0.5f), glm::vec3(1.0f, 1.0f, 1.0f), 0, 0 + mesh, + -(size/2.0f), + size, + 0, 0 ); - - auto renderer = getItem()->getComponent(); - if(renderer) renderer->mesh = mesh; } -void CubeMeshComponent::onDispose() { - mesh = nullptr; -} - -void CubeMeshComponent::load(std::shared_ptr ctx) { - SceneComponent::load(ctx); - if(!mesh) mesh = std::make_shared(); +void CubeMeshComponent::meshLoad(std::shared_ptr ctx) { + if(ctx->data.contains("size")) { + this->size = JSON::vec3(ctx->data["size"]); + } else { + this->size = glm::vec3(1.0f, 1.0f, 1.0f); + } } \ No newline at end of file diff --git a/src/dawn/component/display/mesh/CubeMeshComponent.hpp b/src/dawn/component/display/mesh/CubeMeshComponent.hpp index ac62c2f9..f641ad52 100644 --- a/src/dawn/component/display/mesh/CubeMeshComponent.hpp +++ b/src/dawn/component/display/mesh/CubeMeshComponent.hpp @@ -4,16 +4,14 @@ // https://opensource.org/licenses/MIT #pragma once -#include "scene/SceneItem.hpp" -#include "display/mesh/Mesh.hpp" +#include "MeshComponent.hpp" namespace Dawn { - class CubeMeshComponent final : public SceneComponent { - public: - std::shared_ptr mesh; + class CubeMeshComponent final : public MeshComponent { + protected: + glm::vec3 size; - void onInit() override; - void onDispose() override; - void load(std::shared_ptr ctx) override; + void meshLoad(std::shared_ptr ctx) override; + void meshBuffer(std::shared_ptr mesh) override; }; } \ No newline at end of file diff --git a/src/dawn/component/display/mesh/MeshComponent.cpp b/src/dawn/component/display/mesh/MeshComponent.cpp new file mode 100644 index 00000000..fc2381f5 --- /dev/null +++ b/src/dawn/component/display/mesh/MeshComponent.cpp @@ -0,0 +1,31 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "MeshComponent.hpp" +#include "component/display/MeshRenderer.hpp" + +using namespace Dawn; + +void MeshComponent::buffer() { + if(!mesh) mesh = std::make_shared(); + this->meshBuffer(mesh); +} + +void MeshComponent::onInit() { + this->buffer(); + + auto renderer = getItem()->getComponent(); + if(renderer) renderer->mesh = mesh; +} + +void MeshComponent::onDispose() { + mesh = nullptr; +} + +void MeshComponent::load(std::shared_ptr ctx) { + SceneComponent::load(ctx); + if(this->mesh == nullptr) this->mesh = std::make_shared(); + this->meshLoad(ctx); +} \ No newline at end of file diff --git a/src/dawn/component/display/mesh/MeshComponent.hpp b/src/dawn/component/display/mesh/MeshComponent.hpp new file mode 100644 index 00000000..e34e5c78 --- /dev/null +++ b/src/dawn/component/display/mesh/MeshComponent.hpp @@ -0,0 +1,39 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "scene/SceneItem.hpp" +#include "display/mesh/Mesh.hpp" + +namespace Dawn { + class MeshComponent : public SceneComponent { + protected: + /** + * Called when the mesh should be loaded. + * + * @param ctx Context to load the mesh from. + */ + virtual void meshLoad(std::shared_ptr ctx) = 0; + + /** + * Called when the mesh should be buffered. + * + * @param mesh Mesh to buffer. + */ + virtual void meshBuffer(std::shared_ptr mesh) = 0; + + /** + * Buffers the mesh. + */ + void buffer(); + + public: + std::shared_ptr mesh; + + void onInit() override; + void onDispose() override; + void load(std::shared_ptr ctx) override; + }; +} \ No newline at end of file diff --git a/src/dawn/component/display/mesh/QuadMeshComponent.cpp b/src/dawn/component/display/mesh/QuadMeshComponent.cpp new file mode 100644 index 00000000..89724f1e --- /dev/null +++ b/src/dawn/component/display/mesh/QuadMeshComponent.cpp @@ -0,0 +1,44 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "QuadMeshComponent.hpp" +#include "display/mesh/QuadMesh.hpp" +#include "util/JSON.hpp" + +using namespace Dawn; + +void QuadMeshComponent::meshBuffer(std::shared_ptr mesh) { + mesh->createBuffers(QUAD_VERTICE_COUNT, QUAD_INDICE_COUNT); + QuadMesh::buffer( + mesh, + this->positions, + this->uvs, + 0, 0 + ); +} + +void QuadMeshComponent::meshLoad(std::shared_ptr ctx) { + if(ctx->data.contains("positions")) { + this->positions = JSON::vec4(ctx->data["positions"]); + } else { + this->positions = glm::vec4(0.0f, 0.0f, 1.0f, 1.0f); + } + + if(ctx->data.contains("uvs")) { + this->uvs = JSON::vec4(ctx->data["uvs"]); + } else { + this->uvs = glm::vec4(0.0f, 0.0f, 1.0f, 1.0f); + } +} + +void QuadMeshComponent::setPositions(const glm::vec4 &positions) { + this->positions = positions; + this->buffer(); +} + +void QuadMeshComponent::setUVs(const glm::vec4 &uvs) { + this->uvs = uvs; + this->buffer(); +} \ No newline at end of file diff --git a/src/dawn/component/display/mesh/QuadMeshComponent.hpp b/src/dawn/component/display/mesh/QuadMeshComponent.hpp new file mode 100644 index 00000000..02207c82 --- /dev/null +++ b/src/dawn/component/display/mesh/QuadMeshComponent.hpp @@ -0,0 +1,33 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "MeshComponent.hpp" + +namespace Dawn { + class QuadMeshComponent final : public MeshComponent { + protected: + glm::vec4 positions; + glm::vec4 uvs; + + void meshLoad(std::shared_ptr ctx) override; + void meshBuffer(std::shared_ptr mesh) override; + + public: + /** + * Sets the positions of the quad. + * + * @param positions The positions of the quad. + */ + void setPositions(const glm::vec4 &positions); + + /** + * Sets the UVs of the quad. + * + * @param uvs The UVs of the quad. + */ + void setUVs(const glm::vec4 &uvs); + }; +} \ No newline at end of file diff --git a/src/dawn/display/mesh/PlaneMesh.cpp b/src/dawn/display/mesh/PlaneMesh.cpp new file mode 100644 index 00000000..a967e224 --- /dev/null +++ b/src/dawn/display/mesh/PlaneMesh.cpp @@ -0,0 +1,59 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "PlaneMesh.hpp" + +using namespace Dawn; + +void PlaneMesh::buffer( + const std::shared_ptr mesh, + const glm::vec2 planeSize, + const int32_t columns, + const int32_t rows +) { + const int32_t verticeCount = (columns + 1) * (rows + 1) * PLANE_VERTICE_PER_SEGMENT; + const int32_t indiceCount = columns * rows * PLANE_INDICE_PER_SEGMENT; + mesh->createBuffers(verticeCount, indiceCount); + + const float columnWidth = planeSize.x / columns; + const float rowHeight = planeSize.y / rows; + + glm::vec3 vertices[verticeCount]; + glm::vec2 uvs[verticeCount]; + int32_t indices[indiceCount]; + + int32_t verticeIndex = 0; + int32_t indiceIndex = 0; + for (int32_t row = 0; row <= rows; row++) { + for (int32_t column = 0; column <= columns; column++) { + const float x = column * columnWidth; + const float y = row * rowHeight; + + vertices[verticeIndex] = glm::vec3(x, 0.0f, y); + uvs[verticeIndex] = glm::vec2((float)column / columns, (float)row / rows); + verticeIndex++; + } + } + + for (int32_t row = 0; row < rows; row++) { + for (int32_t column = 0; column < columns; column++) { + const int32_t topLeft = (row * (columns + 1)) + column; + const int32_t topRight = topLeft + 1; + const int32_t bottomLeft = topLeft + (columns + 1); + const int32_t bottomRight = bottomLeft + 1; + + indices[indiceIndex++] = topLeft; + indices[indiceIndex++] = bottomLeft; + indices[indiceIndex++] = topRight; + indices[indiceIndex++] = topRight; + indices[indiceIndex++] = bottomLeft; + indices[indiceIndex++] = bottomRight; + } + } + + mesh->bufferPositions(0, vertices, verticeCount); + mesh->bufferCoordinates(0, uvs, verticeCount); + mesh->bufferIndices(0, indices, indiceCount); +} \ No newline at end of file diff --git a/src/dawn/display/mesh/PlaneMesh.hpp b/src/dawn/display/mesh/PlaneMesh.hpp new file mode 100644 index 00000000..43ead475 --- /dev/null +++ b/src/dawn/display/mesh/PlaneMesh.hpp @@ -0,0 +1,22 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "display/mesh/Mesh.hpp" + +#define PLANE_VERTICE_PER_SEGMENT 4 +#define PLANE_INDICE_PER_SEGMENT 6 + +namespace Dawn { + class PlaneMesh { + public: + static void buffer( + const std::shared_ptr mesh, + const glm::vec2 planeSize, + const int32_t columns, + const int32_t rows + ); + }; +} \ No newline at end of file diff --git a/src/dawn/game/IGame.cpp b/src/dawn/game/IGame.cpp index 868d646f..26b1dd4a 100644 --- a/src/dawn/game/IGame.cpp +++ b/src/dawn/game/IGame.cpp @@ -14,6 +14,7 @@ #include "component/display/material/SimpleTexturedMaterial.hpp" #include "component/display/MeshRenderer.hpp" #include "component/display/mesh/CubeMeshComponent.hpp" +#include "component/display/mesh/QuadMeshComponent.hpp" #ifdef DAWN_ENABLE_PHYSICS #endif @@ -26,6 +27,8 @@ IGame::IGame() { SceneComponentRegistry::reg("MeshRenderer"); SceneComponentRegistry::reg("CubeMeshComponent"); SceneComponentRegistry::reg("CubeMesh"); + SceneComponentRegistry::reg("QuadMeshComponent"); + SceneComponentRegistry::reg("QuadMesh"); } void IGame::init() { diff --git a/src/dawn/game/IGame.hpp b/src/dawn/game/IGame.hpp index e2e047d9..56699605 100644 --- a/src/dawn/game/IGame.hpp +++ b/src/dawn/game/IGame.hpp @@ -23,7 +23,7 @@ namespace Dawn { INITIALIZED = FLAG(1), }; - class IGame : public std::enable_shared_from_this { + class IGame : public std::enable_shared_from_this { private: std::shared_ptr currentScene = nullptr; std::shared_ptr nextFrameScene = nullptr; diff --git a/src/dawn/util/JSON.cpp b/src/dawn/util/JSON.cpp index 38fa12ac..f39425d7 100644 --- a/src/dawn/util/JSON.cpp +++ b/src/dawn/util/JSON.cpp @@ -10,6 +10,7 @@ using namespace Dawn; glm::vec3 JSON::vec3(const json &j) { if(j.type() == json::value_t::array) { + assertTrue(j.size() == 3, "Invalid size for vec3"); return glm::vec3(j[0].get(), j[1].get(), j[2].get()); } else if(j.type() == json::value_t::object) { assertTrue(j.contains("x"), "Missing x in vec3"); @@ -27,6 +28,33 @@ glm::vec3 JSON::vec3(const json &j) { return glm::vec3(0.0f); } +glm::vec4 JSON::vec4(const json &j) { + if(j.type() == json::value_t::array) { + assertTrue(j.size() == 4, "Invalid size for vec4"); + return glm::vec4( + j[0].get(), + j[1].get(), + j[2].get(), + j[3].get() + ); + } else if(j.type() == json::value_t::object) { + assertTrue(j.contains("x"), "Missing x in vec4"); + assertTrue(j.contains("y"), "Missing y in vec4"); + assertTrue(j.contains("z"), "Missing z in vec4"); + assertTrue(j.contains("w"), "Missing w in vec4"); + + return glm::vec4( + j["x"].get(), + j["y"].get(), + j["z"].get(), + j["w"].get() + ); + } + + assertUnreachable("Invalid JSON type for vec4"); + return glm::vec4(0.0f); +} + struct Color JSON::color(const json &j) { if(j.type() == json::value_t::array) { return { diff --git a/src/dawn/util/JSON.hpp b/src/dawn/util/JSON.hpp index 2d055fd4..71f31208 100644 --- a/src/dawn/util/JSON.hpp +++ b/src/dawn/util/JSON.hpp @@ -18,6 +18,14 @@ namespace Dawn { */ static glm::vec3 vec3(const json &j); + /** + * Parses the given JSON string as a vec4. + * + * @param j JSON obj to parse. + * @return Parsed vec4. + */ + static glm::vec4 vec4(const json &j); + /** * Parses the given JSON string as a color. * diff --git a/src/dawnrpg/CMakeLists.txt b/src/dawnrpg/CMakeLists.txt index 3111623d..16dac516 100644 --- a/src/dawnrpg/CMakeLists.txt +++ b/src/dawnrpg/CMakeLists.txt @@ -12,6 +12,7 @@ target_include_directories(${DAWN_TARGET_NAME} # Subdirs add_subdirectory(component) add_subdirectory(game) +add_subdirectory(rpg) # Assets tool_texture(rosa rosa.png) diff --git a/src/dawnrpg/component/RPGEntity.cpp b/src/dawnrpg/component/RPGEntity.cpp index 6c383960..966b5fe2 100644 --- a/src/dawnrpg/component/RPGEntity.cpp +++ b/src/dawnrpg/component/RPGEntity.cpp @@ -5,7 +5,7 @@ #include "RPGEntity.hpp" #include "scene/Scene.hpp" -#include "display/mesh/QuadMesh.hpp" +#include "component/display/mesh/QuadMeshComponent.hpp" #include "asset/loader/TextureLoader.hpp" #include "display/tileset/TilesetGrid.hpp" @@ -15,22 +15,12 @@ void RPGEntity::onInit() { const glm::vec2 size = { RPG_ENTITY_SIZE, RPG_ENTITY_SIZE }; const glm::vec2 position = -size; - mesh = std::make_shared(); - mesh->createBuffers(QUAD_VERTICE_COUNT, QUAD_INDICE_COUNT); - QuadMesh::buffer( - mesh, - glm::vec4(position.x, position.y, size.x, size.y), - glm::vec4(0, 1.0f, 1, 0.75f), - 0, 0, 0 - ); - - getItem()->getComponent()->mesh = mesh; - + auto mesh = getItem()->getComponent(); + mesh->setPositions(glm::vec4(position.x, position.y, size.x, size.y)); updateSprite(); } void RPGEntity::onDispose() { - mesh = nullptr; } enum RPGEntityDirection RPGEntity::getFacingDirection() { @@ -74,6 +64,7 @@ void RPGEntity::updateSprite() { // Convert row/col to UV coordinates. glm::vec4 tile = tileset.getTile(col, row); - tile = glm::vec4(tile.x, tile.w, tile.z, tile.y);// swap Y axis. - QuadMesh::bufferCoordinates(mesh, tile, 0); + tile = glm::vec4(tile.x, tile.w, tile.z, tile.y); + auto mesh = getItem()->getComponent(); + mesh->setUVs(tile); } \ No newline at end of file diff --git a/src/dawnrpg/component/RPGEntity.hpp b/src/dawnrpg/component/RPGEntity.hpp index 10857c3f..5faed759 100644 --- a/src/dawnrpg/component/RPGEntity.hpp +++ b/src/dawnrpg/component/RPGEntity.hpp @@ -20,8 +20,6 @@ namespace Dawn { private: protected: - std::shared_ptr mesh; - enum RPGEntityDirection facingDirection = RPGEntityDirection::SOUTH; void updateSprite(); diff --git a/src/dawnrpg/game/Game.cpp b/src/dawnrpg/game/Game.cpp index 79361138..2350c10a 100644 --- a/src/dawnrpg/game/Game.cpp +++ b/src/dawnrpg/game/Game.cpp @@ -21,7 +21,8 @@ std::string Game::getInitialScene() { } void Game::initManagers() { - + this->rpgManager = std::make_shared(); + this->rpgManager->init(shared_from_this()); } Game::~Game() { diff --git a/src/dawnrpg/game/Game.hpp b/src/dawnrpg/game/Game.hpp index b5d2c2c8..be77f472 100644 --- a/src/dawnrpg/game/Game.hpp +++ b/src/dawnrpg/game/Game.hpp @@ -5,14 +5,19 @@ #pragma once #include "game/IGame.hpp" +#include "rpg/RPGManager.hpp" namespace Dawn { - class Game : public IGame { + class Game : + public IGame + { protected: std::string getInitialScene() override; void initManagers() override; public: + std::shared_ptr rpgManager; + Game(); ~Game(); }; diff --git a/src/dawnrpg/rpg/CMakeLists.txt b/src/dawnrpg/rpg/CMakeLists.txt new file mode 100644 index 00000000..a254eb47 --- /dev/null +++ b/src/dawnrpg/rpg/CMakeLists.txt @@ -0,0 +1,9 @@ +# Copyright (c) 2024 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +target_sources(${DAWN_TARGET_NAME} + PRIVATE + RPGManager.cpp +) \ No newline at end of file diff --git a/src/dawnrpg/rpg/RPGManager.cpp b/src/dawnrpg/rpg/RPGManager.cpp new file mode 100644 index 00000000..136d2582 --- /dev/null +++ b/src/dawnrpg/rpg/RPGManager.cpp @@ -0,0 +1,19 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "RPGManager.hpp" +#include "assert/assert.hpp" + +using namespace Dawn; + +void RPGManager::init(const std::shared_ptr &game) { + this->game = game; +} + +std::shared_ptr RPGManager::getGame() { + auto l = this->game.lock(); + assertNotNull(l, "Game is null?"); + return l; +} \ No newline at end of file diff --git a/src/dawnrpg/rpg/RPGManager.hpp b/src/dawnrpg/rpg/RPGManager.hpp new file mode 100644 index 00000000..51d77379 --- /dev/null +++ b/src/dawnrpg/rpg/RPGManager.hpp @@ -0,0 +1,21 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "dawn.hpp" + +namespace Dawn { + class Game; + + class RPGManager { + private: + std::weak_ptr game; + + public: + void init(const std::shared_ptr &game); + + std::shared_ptr getGame(); + }; +} \ No newline at end of file