From 2af55041c88cea2b7ac21016269823a4b6cfdf8f Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Mon, 2 Dec 2024 14:53:41 -0600 Subject: [PATCH] Example scene loading --- CMakeLists.txt | 3 + assets/test_scene.json | 34 ++++++ src/CMakeLists.txt | 4 + src/dawn/CMakeLists.txt | 4 - src/dawn/assert/assert.hpp | 11 ++ src/dawn/asset/AssetManager.cpp | 28 ++++- src/dawn/asset/AssetManager.hpp | 23 +++- src/dawn/asset/loader/JSONLoader.hpp | 2 - src/dawn/asset/loader/SceneLoader.cpp | 113 +++++++++++++++++- src/dawn/asset/loader/SceneLoader.hpp | 24 +++- src/dawn/component/CMakeLists.txt | 4 +- src/dawn/component/SceneComponentRegistry.cpp | 39 ++++++ src/dawn/component/SceneComponentRegistry.hpp | 25 ++++ src/dawn/component/display/CMakeLists.txt | 3 +- src/dawn/component/display/Camera.cpp | 62 ++++++++++ src/dawn/component/display/Camera.hpp | 1 + .../material/SimpleTexturedMaterial.cpp | 15 +++ .../material/SimpleTexturedMaterial.hpp | 2 + .../component/display/mesh/CMakeLists.txt | 9 ++ .../display/mesh/CubeMeshComponent.cpp | 33 +++++ .../display/mesh/CubeMeshComponent.hpp | 19 +++ src/dawn/dawn.hpp | 5 +- src/dawn/game/IGame.cpp | 30 +++-- src/dawn/game/IGame.hpp | 6 +- src/dawn/scene/Scene.cpp | 3 + src/dawn/scene/Scene.hpp | 2 +- src/dawn/scene/SceneComponent.cpp | 4 + src/dawn/scene/SceneComponent.hpp | 28 +++++ src/dawn/scene/SceneItem.cpp | 24 ++++ src/dawn/scene/SceneItem.hpp | 9 ++ src/dawn/util/CMakeLists.txt | 1 + src/dawn/util/JSON.cpp | 77 ++++++++++++ src/dawn/util/JSON.hpp | 29 +++++ src/dawnphysics/CMakeLists.txt | 26 ++++ src/dawnphysics/component/CMakeLists.txt | 7 ++ .../component/physics/CMakeLists.txt | 0 .../component/physics/Collider.cpp | 0 .../component/physics/Collider.hpp | 0 .../component/physics/CubeCollider.cpp | 0 .../component/physics/CubeCollider.hpp | 0 .../component/physics/SphereCollider.cpp | 0 .../component/physics/SphereCollider.hpp | 0 .../physics/CMakeLists.txt | 0 .../physics/PhysicsManager.cpp | 0 .../physics/PhysicsManager.hpp | 0 src/dawnrpg/CMakeLists.txt | 2 +- tools/assetstool/assetstool.py | 18 ++- tools/texturetool/CMakeLists.txt | 7 +- 48 files changed, 698 insertions(+), 38 deletions(-) create mode 100644 src/dawn/component/SceneComponentRegistry.cpp create mode 100644 src/dawn/component/SceneComponentRegistry.hpp create mode 100644 src/dawn/component/display/mesh/CMakeLists.txt create mode 100644 src/dawn/component/display/mesh/CubeMeshComponent.cpp create mode 100644 src/dawn/component/display/mesh/CubeMeshComponent.hpp create mode 100644 src/dawn/util/JSON.cpp create mode 100644 src/dawn/util/JSON.hpp create mode 100644 src/dawnphysics/CMakeLists.txt create mode 100644 src/dawnphysics/component/CMakeLists.txt rename src/{dawn => dawnphysics}/component/physics/CMakeLists.txt (100%) rename src/{dawn => dawnphysics}/component/physics/Collider.cpp (100%) rename src/{dawn => dawnphysics}/component/physics/Collider.hpp (100%) rename src/{dawn => dawnphysics}/component/physics/CubeCollider.cpp (100%) rename src/{dawn => dawnphysics}/component/physics/CubeCollider.hpp (100%) rename src/{dawn => dawnphysics}/component/physics/SphereCollider.cpp (100%) rename src/{dawn => dawnphysics}/component/physics/SphereCollider.hpp (100%) rename src/{dawn => dawnphysics}/physics/CMakeLists.txt (100%) rename src/{dawn => dawnphysics}/physics/PhysicsManager.cpp (100%) rename src/{dawn => dawnphysics}/physics/PhysicsManager.hpp (100%) diff --git a/CMakeLists.txt b/CMakeLists.txt index 06bf2900..939b0e44 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -23,6 +23,9 @@ set(DAWN_ASSETS_BUILD_DIR "${DAWN_BUILD_DIR}/assets") set(DAWN_GENERATED_DIR "${DAWN_BUILD_DIR}/generated") set(DAWN_TEMP_DIR "${DAWN_BUILD_DIR}/temp") +# Options +option(DAWN_ENABLE_PHYSICS "Enable Physics" OFF) + # Initialize Project First. project(Dawn VERSION 1.0.0 diff --git a/assets/test_scene.json b/assets/test_scene.json index e69de29b..08b7c5a4 100644 --- a/assets/test_scene.json +++ b/assets/test_scene.json @@ -0,0 +1,34 @@ +{ + "name": "Test Scene", + "assets": { + "rosa": { + "type": "texture", + "path": "rosa.texture" + } + }, + "items": { + "camera": { + "lookAt": { + "position": [ 5, 5, 5 ], + "look": [ 0, 0, 0 ], + "view": [ 0, 1, 0 ] + }, + "components": { + "camera": { + "type": "Camera", + "fov": 90 + } + } + }, + + "cube": { + "position": [ 0, 0, 0 ], + "scale": [ 3, 3, 3 ], + "components": { + "mat": { "type": "SimpleTexturedMaterial", "color": "blue", "texture": "rosa" }, + "renderer": { "type": "MeshRenderer" }, + "mesh": { "type": "CubeMesh" } + } + } + } +} \ No newline at end of file diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index d4c38e94..8abf5e53 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -15,6 +15,10 @@ endif() add_subdirectory(dawn) add_subdirectory(dawnrpg) +if(DAWN_ENABLE_PHYSICS) + add_subdirectory(dawnphysics) +endif() + # Host Libraries target_link_libraries(${DAWN_TARGET_NAME} PUBLIC diff --git a/src/dawn/CMakeLists.txt b/src/dawn/CMakeLists.txt index 33b4d199..6ef267b5 100644 --- a/src/dawn/CMakeLists.txt +++ b/src/dawn/CMakeLists.txt @@ -27,14 +27,10 @@ add_subdirectory(component) add_subdirectory(display) add_subdirectory(environment) add_subdirectory(game) -# add_subdirectory(games) -# add_subdirectory(input) add_subdirectory(locale) add_subdirectory(prefab) -add_subdirectory(physics) add_subdirectory(save) add_subdirectory(scene) -# add_subdirectory(state) add_subdirectory(time) add_subdirectory(util) add_subdirectory(ui) diff --git a/src/dawn/assert/assert.hpp b/src/dawn/assert/assert.hpp index 561b11ed..89a1191a 100644 --- a/src/dawn/assert/assert.hpp +++ b/src/dawn/assert/assert.hpp @@ -84,6 +84,17 @@ void assertTrueImplement( map.find(key) != map.end(), __VA_ARGS__ \ ) +/** + * Asserts that a given map does not have a specific key. + * @param map Map to check. + * @param key Key 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 assertMapNotHasKey(map, key, ...) assertTrue( \ + map.find(key) == map.end(), __VA_ARGS__ \ +) + /** * Asserts that a given value has a specific flag turned off. * diff --git a/src/dawn/asset/AssetManager.cpp b/src/dawn/asset/AssetManager.cpp index e23c7cf4..fcf03403 100644 --- a/src/dawn/asset/AssetManager.cpp +++ b/src/dawn/asset/AssetManager.cpp @@ -4,11 +4,13 @@ // https://opensource.org/licenses/MIT #include "AssetManager.hpp" +#include "assert/assert.hpp" using namespace Dawn; -void AssetManager::init() { - +void AssetManager::init(const std::shared_ptr &game) { + assertNotNull(game, "Game is NULL?"); + this->game = game; } void AssetManager::update() { @@ -27,6 +29,22 @@ void AssetManager::update() { } } +void AssetManager::remove(const std::shared_ptr loader) { + auto existing = std::find( + pendingAssetLoaders.begin(), pendingAssetLoaders.end(), loader + ); + if(existing != pendingAssetLoaders.end()) { + pendingAssetLoaders.erase(existing); + } + + existing = std::find( + finishedAssetLoaders.begin(), finishedAssetLoaders.end(), loader + ); + if(existing != finishedAssetLoaders.end()) { + finishedAssetLoaders.erase(existing); + } +} + // Disabled because it does not respect scopes // void AssetManager::removeExisting(const std::string &filename) { // auto existing = std::find_if( @@ -54,6 +72,12 @@ bool_t AssetManager::isEverythingLoaded() { return pendingAssetLoaders.size() == 0; } +std::shared_ptr AssetManager::getGame() { + auto g = game.lock(); + assertNotNull(g, "Game is NULL?"); + return g; +} + AssetManager::~AssetManager() { } \ No newline at end of file diff --git a/src/dawn/asset/AssetManager.hpp b/src/dawn/asset/AssetManager.hpp index c80ef839..c0543ac3 100644 --- a/src/dawn/asset/AssetManager.hpp +++ b/src/dawn/asset/AssetManager.hpp @@ -8,8 +8,11 @@ #include "asset/AssetLoader.hpp" namespace Dawn { + class Game; + class AssetManager final : public std::enable_shared_from_this { private: + std::weak_ptr game; std::vector> pendingAssetLoaders; std::vector> finishedAssetLoaders; @@ -59,8 +62,10 @@ namespace Dawn { public: /** * Initializes this asset manager so it can begin accepting assets. + * + * @param game Game context that this asset manager is attached to. */ - void init(); + void init(const std::shared_ptr &game); /** * Updates the asset manager. @@ -96,6 +101,22 @@ namespace Dawn { return loader; } + /** + * Returns the game context that this asset manager is attached to. + * + * @return The game context. + */ + std::shared_ptr getGame(); + + /** + * Removes the given asset loader from the asset manager, assumes that + * nothing else needs to access it and any dangling shared_ptrs will have + * to remain in memory. + * + * @param loader The asset loader to remove. + */ + void remove(const std::shared_ptr loader); + /** * Dispose the asset manager, and all attached assets. */ diff --git a/src/dawn/asset/loader/JSONLoader.hpp b/src/dawn/asset/loader/JSONLoader.hpp index 43259a21..3f7f4fcf 100644 --- a/src/dawn/asset/loader/JSONLoader.hpp +++ b/src/dawn/asset/loader/JSONLoader.hpp @@ -6,9 +6,7 @@ #pragma once #include "asset/AssetLoader.hpp" #include "asset/AssetDataLoader.hpp" -#include -using json = nlohmann::json; namespace Dawn { enum class JSONLoaderState { diff --git a/src/dawn/asset/loader/SceneLoader.cpp b/src/dawn/asset/loader/SceneLoader.cpp index f8c4ce33..68be477b 100644 --- a/src/dawn/asset/loader/SceneLoader.cpp +++ b/src/dawn/asset/loader/SceneLoader.cpp @@ -4,6 +4,11 @@ // https://opensource.org/licenses/MIT #include "SceneLoader.hpp" +#include "game/Game.hpp" +#include "assert/assert.hpp" +#include "asset/loader/TextureLoader.hpp" +#include "scene/Scene.hpp" +#include "component/SceneComponentRegistry.hpp" using namespace Dawn; @@ -18,6 +23,83 @@ SceneLoader::SceneLoader( { } +void SceneLoader::setupDependencies() { + std::cout << "Setting up dependencies" << std::endl; + // Begin loading dependencies. + auto &data = this->jsonLoader->data; + if(data.contains("assets")) { + for(auto &asset : data["assets"].items()) { + auto &assetName = asset.key(); + auto &assetData = asset.value(); + assertTrue(assetData.contains("type"), "Asset missing type"); + assertTrue(assetData.contains("path"), "Asset missing path"); + auto type = assetData["type"].get(); + auto path = assetData["path"].get(); + + std::shared_ptr loader; + if(type == "texture") { + loader = getAssetManager()->get(path); + } else if(type == "json") { + loader = getAssetManager()->get(path); + } else { + assertUnreachable("Unknown asset type: %s", type.c_str()); + } + + assertMapNotHasKey( + ctx.assets, + assetName, + "Asset already exists: %s", assetName.c_str() + ); + + ctx.assets[assetName] = loader; + } + } + this->state = SceneLoaderState::LOADING_DEPENDENCIES; +} + +void SceneLoader::sceneInitializer(Scene &scene) { + std::cout << "Initializing scene" << std::endl; + + auto &data = this->jsonLoader->data; + if(data.contains("items")) { + // Create the scene items + for(auto &item : data["items"].items()) { + auto &itemName = item.key(); + auto &itemData = item.value(); + auto sceneItem = scene.createSceneItem(); + ctx.items[itemName] = sceneItem; + } + + // Add components to each scene item + for(auto &item : data["items"].items()) { + auto &itemName = item.key(); + auto &itemData = item.value(); + auto sceneItem = ctx.items[itemName]; + + ctx.data = itemData; + sceneItem->load(ctx); + + if(itemData.contains("components")) { + for(auto &cmpItem : itemData["components"].items()) { + auto &cmpName = cmpItem.key(); + auto &cmpData = cmpItem.value(); + assertTrue(cmpData.contains("type"), "Component missing type in %s", itemName.c_str()); + + auto cmpType = cmpData["type"].get(); + auto cmp = SceneComponentRegistry::createComponent(cmpType, sceneItem); + ctx.data = cmpData; + ctx.components[cmpName] = cmp; + cmp->load(ctx); + } + } + } + } + + this->jsonLoader = nullptr; + this->state = SceneLoaderState::DONE; + getAssetManager()->remove(shared_from_this()); +} + void SceneLoader::updateAsync() { switch(this->state) { case SceneLoaderState::INITIAL: @@ -26,6 +108,31 @@ void SceneLoader::updateAsync() { break; case SceneLoaderState::LOADING_JSON: + assertNotNull(this->jsonLoader, "JSON Loader is NULL?"); + if(!this->jsonLoader->loaded) return; + this->setupDependencies(); + break; + + case SceneLoaderState::LOADING_DEPENDENCIES: + // Check if all dependencies are loaded. + for(auto &asset : ctx.assets) { + if(!asset.second->loaded) return; + } + + this->state = SceneLoaderState::DEPENDENCIES_LOADED; + break; + + case SceneLoaderState::DEPENDENCIES_LOADED: + std::cout << "Deps Loaded" << std::endl; + this->loaded = true; + ctx.scene = std::make_shared( + this->getAssetManager()->getGame(), + [this](Scene &scene) -> void { + this->sceneInitializer(scene); + } + ); + this->state = SceneLoaderState::PENDING_STAGE; + this->loaded = true; break; default: @@ -42,9 +149,11 @@ std::string SceneLoader::getAssetType() const { } std::shared_ptr SceneLoader::getScene() { - return scene; + return ctx.scene; } SceneLoader::~SceneLoader() { - + ctx = {}; + jsonLoader = nullptr; + std::cout << "Scene Loader removed" << std::endl; } \ No newline at end of file diff --git a/src/dawn/asset/loader/SceneLoader.hpp b/src/dawn/asset/loader/SceneLoader.hpp index 4f1ae682..79adfd83 100644 --- a/src/dawn/asset/loader/SceneLoader.hpp +++ b/src/dawn/asset/loader/SceneLoader.hpp @@ -11,14 +11,32 @@ namespace Dawn { enum class SceneLoaderState { INITIAL, LOADING_JSON, + LOADING_DEPENDENCIES, + DEPENDENCIES_LOADED, + PENDING_STAGE, + DONE }; - class SceneLoader : public AssetLoader { + class SceneLoader : + public AssetLoader, + public std::enable_shared_from_this + { protected: SceneLoaderState state; - std::shared_ptr jsonLoader; - std::shared_ptr scene; + struct SceneComponentLoadContext ctx; + + /** + * Loads the dependencies of the scene. + */ + void setupDependencies(); + + /** + * Scene intializer function to stage the loaded scene. + * + * @param scene Scene that is being staged. + */ + void sceneInitializer(Scene &scene); public: const static std::string ASSET_TYPE; diff --git a/src/dawn/component/CMakeLists.txt b/src/dawn/component/CMakeLists.txt index e691a29c..4333a63a 100644 --- a/src/dawn/component/CMakeLists.txt +++ b/src/dawn/component/CMakeLists.txt @@ -6,9 +6,9 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE SimpleComponent.cpp + SceneComponentRegistry.cpp ) # Subdirs add_subdirectory(display) -add_subdirectory(ui) -add_subdirectory(physics) \ No newline at end of file +add_subdirectory(ui) \ No newline at end of file diff --git a/src/dawn/component/SceneComponentRegistry.cpp b/src/dawn/component/SceneComponentRegistry.cpp new file mode 100644 index 00000000..8e142446 --- /dev/null +++ b/src/dawn/component/SceneComponentRegistry.cpp @@ -0,0 +1,39 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "SceneComponentRegistry.hpp" + +#include "component/display/Camera.hpp" +#include "component/display/material/SimpleTexturedMaterial.hpp" +#include "component/display/MeshRenderer.hpp" +#include "component/display/mesh/CubeMeshComponent.hpp" + +using namespace Dawn; + +std::shared_ptr SceneComponentRegistry::createComponent( + const std::string &type, + const std::shared_ptr &item +) { + if(type.length() == 0) { + assertUnreachable("Component type is empty!"); + + } else if(type == "Camera") { + return item->addComponent(); + + } else if(type == "SimpleTexturedMaterial") { + return item->addComponent(); + + } else if(type == "MeshRenderer") { + return item->addComponent(); + + } else if(type == "CubeMesh" || type == "CubeMeshComponent") { + return item->addComponent(); + + } else { + assertUnreachable("Unknown component type: %s", type.c_str()); + } + + return nullptr; +} \ No newline at end of file diff --git a/src/dawn/component/SceneComponentRegistry.hpp b/src/dawn/component/SceneComponentRegistry.hpp new file mode 100644 index 00000000..070d8514 --- /dev/null +++ b/src/dawn/component/SceneComponentRegistry.hpp @@ -0,0 +1,25 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "scene/Scene.hpp" + +namespace Dawn { + class SceneComponentRegistry final { + public: + /** + * Creates a scene component by its type name. Type names are unique and + * must be registered in order to be constructed by their string name. + * + * @param type Type name of the scene component to create. + * @param item Scene item that the component belongs to. + * @return The created scene component. + */ + static std::shared_ptr createComponent( + const std::string &type, + const std::shared_ptr &item + ); + }; +} \ No newline at end of file diff --git a/src/dawn/component/display/CMakeLists.txt b/src/dawn/component/display/CMakeLists.txt index 3afde340..c2a3c749 100644 --- a/src/dawn/component/display/CMakeLists.txt +++ b/src/dawn/component/display/CMakeLists.txt @@ -10,4 +10,5 @@ target_sources(${DAWN_TARGET_NAME} ) # Subdirs -add_subdirectory(material) \ No newline at end of file +add_subdirectory(material) +add_subdirectory(mesh) \ No newline at end of file diff --git a/src/dawn/component/display/Camera.cpp b/src/dawn/component/display/Camera.cpp index 2d19232e..788e6203 100644 --- a/src/dawn/component/display/Camera.cpp +++ b/src/dawn/component/display/Camera.cpp @@ -21,6 +21,68 @@ void Camera::onDispose() { renderTarget = nullptr; } +void Camera::load(const SceneComponentLoadContext &ctx) { + SceneComponent::load(ctx); + + if(ctx.data.contains("fov")) { + this->fov = Math::deg2rad(ctx.data["fov"].get()); + } + + if(ctx.data.contains("cameraType")) { + if( + ctx.data["cameraType"] == "orthogonal" || + ctx.data["cameraType"] == "orthographic" || + ctx.data["cameraType"] == "ortho" + ) { + this->type = CameraType::ORTHOGONAL; + } else if(ctx.data["cameraType"] == "perspective") { + this->type = CameraType::PERSPECTIVE; + } else { + assertUnreachable("Invalid Camera Type!"); + } + } + + if(ctx.data.contains("orthoLeft")) { + this->orthoLeft = ctx.data["orthoLeft"].get(); + } else if(ctx.data.contains("left")) { + this->orthoLeft = ctx.data["left"].get(); + } + + if(ctx.data.contains("orthoRight")) { + this->orthoRight = ctx.data["orthoRight"].get(); + } else if(ctx.data.contains("right")) { + this->orthoRight = ctx.data["right"].get(); + } + + if(ctx.data.contains("orthoBottom")) { + this->orthoBottom = ctx.data["orthoBottom"].get(); + } else if(ctx.data.contains("bottom")) { + this->orthoBottom = ctx.data["bottom"].get(); + } + + if(ctx.data.contains("orthoTop")) { + this->orthoTop = ctx.data["orthoTop"].get(); + } else if(ctx.data.contains("top")) { + this->orthoTop = ctx.data["top"].get(); + } + + if(ctx.data.contains("clipNear")) { + this->clipNear = ctx.data["clipNear"].get(); + } else if(ctx.data.contains("near")) { + this->clipNear = ctx.data["near"].get(); + } else if(ctx.data.contains("zNear")) { + this->clipNear = ctx.data["zNear"].get(); + } + + if(ctx.data.contains("clipFar")) { + this->clipFar = ctx.data["clipFar"].get(); + } else if(ctx.data.contains("far")) { + this->clipFar = ctx.data["far"].get(); + } else if(ctx.data.contains("zFar")) { + this->clipFar = ctx.data["zFar"].get(); + } +} + std::shared_ptr Camera::getRenderTarget() { if(this->renderTarget) return this->renderTarget; return getGame()->renderHost->getBackBufferRenderTarget(); diff --git a/src/dawn/component/display/Camera.hpp b/src/dawn/component/display/Camera.hpp index 7e9af0dc..a8ece0b1 100644 --- a/src/dawn/component/display/Camera.hpp +++ b/src/dawn/component/display/Camera.hpp @@ -33,6 +33,7 @@ namespace Dawn { void onInit() override; void onDispose() override; + void load(const SceneComponentLoadContext &ctx) override; /** * Returns the aspect ratio that the camera is using. In future I may diff --git a/src/dawn/component/display/material/SimpleTexturedMaterial.cpp b/src/dawn/component/display/material/SimpleTexturedMaterial.cpp index 00fd64af..75f84c48 100644 --- a/src/dawn/component/display/material/SimpleTexturedMaterial.cpp +++ b/src/dawn/component/display/material/SimpleTexturedMaterial.cpp @@ -4,9 +4,24 @@ // https://opensource.org/licenses/MIT #include "SimpleTexturedMaterial.hpp" +#include "util/JSON.hpp" +#include "asset/loader/TextureLoader.hpp" using namespace Dawn; +void SimpleTexturedMaterial::load(const SceneComponentLoadContext &ctx) { + if(ctx.data.contains("color")) { + this->setColor(JSON::color(ctx.data["color"])); + } + + if(ctx.data.contains("texture")) { + auto asset = ctx.getAsset( + ctx.data["texture"].get() + ); + this->setTexture(asset->getTexture()); + } +} + struct Color SimpleTexturedMaterial::getColor() { return this->data.color; } diff --git a/src/dawn/component/display/material/SimpleTexturedMaterial.hpp b/src/dawn/component/display/material/SimpleTexturedMaterial.hpp index dc91d179..9869caa0 100644 --- a/src/dawn/component/display/material/SimpleTexturedMaterial.hpp +++ b/src/dawn/component/display/material/SimpleTexturedMaterial.hpp @@ -15,6 +15,8 @@ namespace Dawn { std::shared_ptr texture; public: + void load(const SceneComponentLoadContext &ctx) override; + /** * Returns the color of this material. */ diff --git a/src/dawn/component/display/mesh/CMakeLists.txt b/src/dawn/component/display/mesh/CMakeLists.txt new file mode 100644 index 00000000..77b7959b --- /dev/null +++ b/src/dawn/component/display/mesh/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 + CubeMeshComponent.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 new file mode 100644 index 00000000..3e159431 --- /dev/null +++ b/src/dawn/component/display/mesh/CubeMeshComponent.cpp @@ -0,0 +1,33 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "CubeMeshComponent.hpp" +#include "display/mesh/CubeMesh.hpp" +#include "component/display/MeshRenderer.hpp" + +using namespace Dawn; + +void CubeMeshComponent::onInit() { + if(!mesh) { + mesh = std::make_shared(); + } + + 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 + ); + + auto renderer = getItem()->getComponent(); + if(renderer) renderer->mesh = mesh; +} + +void CubeMeshComponent::onDispose() { + mesh = nullptr; +} + +void CubeMeshComponent::load(const SceneComponentLoadContext &ctx) { + if(!mesh) mesh = std::make_shared(); + +} \ No newline at end of file diff --git a/src/dawn/component/display/mesh/CubeMeshComponent.hpp b/src/dawn/component/display/mesh/CubeMeshComponent.hpp new file mode 100644 index 00000000..df692721 --- /dev/null +++ b/src/dawn/component/display/mesh/CubeMeshComponent.hpp @@ -0,0 +1,19 @@ +// 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 CubeMeshComponent final : public SceneComponent { + public: + std::shared_ptr mesh; + + void onInit() override; + void onDispose() override; + void load(const SceneComponentLoadContext &ctx) override; + }; +} \ No newline at end of file diff --git a/src/dawn/dawn.hpp b/src/dawn/dawn.hpp index fd3703db..e972067a 100644 --- a/src/dawn/dawn.hpp +++ b/src/dawn/dawn.hpp @@ -60,4 +60,7 @@ extern "C" { #include #include #include -#include \ No newline at end of file +#include + +#include +using json = nlohmann::json; \ No newline at end of file diff --git a/src/dawn/game/IGame.cpp b/src/dawn/game/IGame.cpp index 4e4e2b6d..c047b856 100644 --- a/src/dawn/game/IGame.cpp +++ b/src/dawn/game/IGame.cpp @@ -25,21 +25,30 @@ void IGame::init() { renderHost->init(selfAsGame); assetManager = std::make_shared(); - assetManager->init(); + assetManager->init(selfAsGame); localeManager = std::make_shared(); localeManager->init(selfAsGame); - physicsManager = std::make_shared(); - physicsManager->init(selfAsGame); + #ifdef DAWN_ENABLE_PHYSICS + physicsManager = std::make_shared(); + physicsManager->init(selfAsGame); + #endif inputManager.init(selfAsGame); saveManager.init(selfAsGame); this->initManagers(); - auto initialScene = this->getInitialScene(); - nextFrameScene = std::make_shared(selfAsGame, initialScene); + // TEST + auto scene = this->assetManager->get("test_scene.json"); + while(!this->assetManager->isEverythingLoaded()) { + this->assetManager->update(); + } + + // auto initialScene = this->getInitialScene(); + // nextFrameScene = std::make_shared(selfAsGame, initialScene); + nextFrameScene = scene->getScene(); } void IGame::deinit() { @@ -51,7 +60,10 @@ void IGame::deinit() { if(nextFrameScene) nextFrameScene->deinit(); nextFrameScene = nullptr; - physicsManager = nullptr; + #ifdef DAWN_ENABLE_PHYSICS + physicsManager = nullptr; + #endif + assetManager = nullptr; renderHost = nullptr; @@ -71,7 +83,11 @@ void IGame::update() { timeManager.update(); if(currentScene) currentScene->update(); - physicsManager->update(); + + #ifdef DAWN_ENABLE_PHYSICS + physicsManager->update(); + #endif + renderHost->update(this->getSelfAsGame()); } diff --git a/src/dawn/game/IGame.hpp b/src/dawn/game/IGame.hpp index 0f87ce54..a35f0721 100644 --- a/src/dawn/game/IGame.hpp +++ b/src/dawn/game/IGame.hpp @@ -50,7 +50,11 @@ namespace Dawn { std::shared_ptr renderHost; std::shared_ptr assetManager; std::shared_ptr localeManager; - std::shared_ptr physicsManager; + + #ifdef DAWN_ENABLE_PHYSICS + std::shared_ptr physicsManager; + #endif + InputManager inputManager; TimeManager timeManager; SaveManager saveManager; diff --git a/src/dawn/scene/Scene.cpp b/src/dawn/scene/Scene.cpp index b4fc5d58..3f3cdd8c 100644 --- a/src/dawn/scene/Scene.cpp +++ b/src/dawn/scene/Scene.cpp @@ -19,6 +19,9 @@ Scene::Scene( void Scene::init() { Scene &selfReference = *this; sceneInitializer(selfReference); + sceneInitializer = [](Scene &scene) -> void { + + }; } void Scene::deinit() { diff --git a/src/dawn/scene/Scene.hpp b/src/dawn/scene/Scene.hpp index 13035067..1a8df233 100644 --- a/src/dawn/scene/Scene.hpp +++ b/src/dawn/scene/Scene.hpp @@ -19,7 +19,7 @@ namespace Dawn { class Scene final : public std::enable_shared_from_this { private: std::weak_ptr game; - const std::function sceneInitializer; + std::function sceneInitializer; std::vector> sceneItems; std::vector> sceneItemsToRemove; bool_t paused = false; diff --git a/src/dawn/scene/SceneComponent.cpp b/src/dawn/scene/SceneComponent.cpp index 041fa0dd..8083b4ed 100644 --- a/src/dawn/scene/SceneComponent.cpp +++ b/src/dawn/scene/SceneComponent.cpp @@ -70,6 +70,10 @@ std::shared_ptr SceneComponent::getGame() { return this->getScene()->getGame(); } +void SceneComponent::load(const SceneComponentLoadContext &context) { + // Override this method to load data from a JSON object. +} + SceneComponent::~SceneComponent() { if(Flag::isOn( sceneComponentState, diff --git a/src/dawn/scene/SceneComponent.hpp b/src/dawn/scene/SceneComponent.hpp index c1842ad5..a2617a6c 100644 --- a/src/dawn/scene/SceneComponent.hpp +++ b/src/dawn/scene/SceneComponent.hpp @@ -5,6 +5,7 @@ #pragma once #include "dawn.hpp" +#include "assert/assert.hpp" #define SCENE_COMPONENT_STATE_INIT 0x01 #define SCENE_COMPONENT_STATE_DISPOSED 0x02 @@ -13,6 +14,25 @@ namespace Dawn { class Game; class Scene; class SceneItem; + class SceneComponent; + class AssetLoader; + + struct SceneComponentLoadContext { + json data; + std::shared_ptr scene; + std::unordered_map> items; + std::unordered_map> components; + std::unordered_map> assets; + + template + std::shared_ptr getAsset(const std::string &j) const { + auto it = assets.find(j); + assertTrue(it != assets.end(), "Asset not found."); + auto asset = std::dynamic_pointer_cast(it->second); + assertNotNull(asset, "Asset is not of the correct type."); + return asset; + } + }; class SceneComponent : std::enable_shared_from_this { private: @@ -75,6 +95,14 @@ namespace Dawn { */ std::shared_ptr getGame(); + /** + * Load data from a JSON object. This is typically done during a scene + * load. + * + * @param json JSON Data that this object needs to load. + */ + virtual void load(const SceneComponentLoadContext &context); + /** * Disposes this scene component. */ diff --git a/src/dawn/scene/SceneItem.cpp b/src/dawn/scene/SceneItem.cpp index 61a44952..5c864dcb 100644 --- a/src/dawn/scene/SceneItem.cpp +++ b/src/dawn/scene/SceneItem.cpp @@ -5,6 +5,7 @@ #include "scene/SceneItem.hpp" #include "scene/Scene.hpp" +#include "util/JSON.hpp" using namespace Dawn; @@ -65,6 +66,29 @@ void SceneItem::deinit() { this->components.clear(); } +void SceneItem::load(const SceneComponentLoadContext &ctx) { + // Transforms + if(ctx.data.contains("position")) { + this->setLocalPosition(JSON::vec3(ctx.data["position"])); + } + + if(ctx.data.contains("lookAt")) { + auto &lookAt = ctx.data["lookAt"]; + glm::vec3 pos = glm::vec3(3, 3, 3); + glm::vec3 look = glm::vec3(0, 0, 0); + glm::vec3 up = glm::vec3(0, 1, 0); + + if(lookAt.contains("position")) pos = JSON::vec3(lookAt["position"]); + if(lookAt.contains("look")) look = JSON::vec3(lookAt["look"]); + if(lookAt.contains("up")) up = JSON::vec3(lookAt["up"]); + this->lookAt(pos, look, up); + } + + if(ctx.data.contains("scale")) { + this->setLocalScale(JSON::vec3(ctx.data["scale"])); + } +} + void SceneItem::remove() { auto scene = getScene(); if(!scene) return; diff --git a/src/dawn/scene/SceneItem.hpp b/src/dawn/scene/SceneItem.hpp index 1ca13e09..1376a262 100644 --- a/src/dawn/scene/SceneItem.hpp +++ b/src/dawn/scene/SceneItem.hpp @@ -41,6 +41,15 @@ namespace Dawn { */ void deinit(); + /** + * Loads this scene item from the given context. Scene items are not + * responsible for loading their components, this is handled by the scene + * loader. + * + * @param ctx Context to load this scene item from. + */ + void load(const SceneComponentLoadContext &ctx); + /** * Returns the scene that this scene item belongs to. * diff --git a/src/dawn/util/CMakeLists.txt b/src/dawn/util/CMakeLists.txt index 2e4f3a5a..8aa00387 100644 --- a/src/dawn/util/CMakeLists.txt +++ b/src/dawn/util/CMakeLists.txt @@ -6,4 +6,5 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE String.cpp + JSON.cpp ) \ No newline at end of file diff --git a/src/dawn/util/JSON.cpp b/src/dawn/util/JSON.cpp new file mode 100644 index 00000000..fb77b16b --- /dev/null +++ b/src/dawn/util/JSON.cpp @@ -0,0 +1,77 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "JSON.hpp" +#include "assert/assert.hpp" + +using namespace Dawn; + +glm::vec3 JSON::vec3(const json &j) { + if(j.type() == json::value_t::array) { + 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"); + assertTrue(j.contains("y"), "Missing y in vec3"); + assertTrue(j.contains("z"), "Missing z in vec3"); + + return glm::vec3( + j["x"].get(), + j["y"].get(), + j["z"].get() + ); + } else { + assertUnreachable("Invalid JSON type for vec3"); + } +} + +struct Color JSON::color(const json &j) { + if(j.type() == json::value_t::array) { + return { + j[0].get(), + j[1].get(), + j[2].get(), + j[3].get() + }; + + } else if(j.type() == json::value_t::object) { + float_t r, g, b, a = 1.0f; + + if(j.contains("r")) { + r = j["r"].get(); + } else if(j.contains("red")) { + r = j["red"].get(); + } else { + assertTrue(j.contains("red"), "Missing red in color"); + } + + if(j.contains("g")) { + g = j["g"].get(); + } else if(j.contains("green")) { + g = j["green"].get(); + } else { + assertTrue(j.contains("green"), "Missing green in color"); + } + + if(j.contains("b")) { + b = j["b"].get(); + } else if(j.contains("blue")) { + b = j["blue"].get(); + } else { + assertTrue(j.contains("blue"), "Missing blue in color"); + } + + if(j.contains("a")) { + a = j["a"].get(); + } else if(j.contains("alpha")) { + a = j["alpha"].get(); + } + + return { r, g, b, a }; + } else if(j.type() == json::value_t::string) { + return Color::fromString(j.get()); + } else { + assertUnreachable("Invalid JSON type for color"); + } +} \ No newline at end of file diff --git a/src/dawn/util/JSON.hpp b/src/dawn/util/JSON.hpp new file mode 100644 index 00000000..1a591701 --- /dev/null +++ b/src/dawn/util/JSON.hpp @@ -0,0 +1,29 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "dawn.hpp" +#include "display/Color.hpp" + +namespace Dawn { + class JSON final { + public: + /** + * Parses the given JSON string as a vec3. + * + * @param j JSON obj to parse. + * @return Parsed vec3. + */ + static glm::vec3 vec3(const json &j); + + /** + * Parses the given JSON string as a color. + * + * @param j JSON obj to parse. + * @return Parsed color. + */ + static struct Color color(const json &j); + }; +} \ No newline at end of file diff --git a/src/dawnphysics/CMakeLists.txt b/src/dawnphysics/CMakeLists.txt new file mode 100644 index 00000000..eda889b4 --- /dev/null +++ b/src/dawnphysics/CMakeLists.txt @@ -0,0 +1,26 @@ +# Copyright (c) 2024 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +# Libraries +target_link_libraries(${DAWN_TARGET_NAME} + PUBLIC + Jolt +) + +# Includes +target_include_directories(${DAWN_TARGET_NAME} + PUBLIC + ${CMAKE_CURRENT_LIST_DIR} +) + +# Define for use in C pragmas +target_compile_definitions(${DAWN_TARGET_NAME} + PUBLIC + DAWN_ENABLE_PHYSICS="${DAWN_ENABLE_PHYSICS}" +) + +# Subdirs +add_subdirectory(physics) +add_subdirectory(component) \ No newline at end of file diff --git a/src/dawnphysics/component/CMakeLists.txt b/src/dawnphysics/component/CMakeLists.txt new file mode 100644 index 00000000..ea7fa3af --- /dev/null +++ b/src/dawnphysics/component/CMakeLists.txt @@ -0,0 +1,7 @@ +# Copyright (c) 2024 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +# Subdirs +add_subdirectory(physics) \ No newline at end of file diff --git a/src/dawn/component/physics/CMakeLists.txt b/src/dawnphysics/component/physics/CMakeLists.txt similarity index 100% rename from src/dawn/component/physics/CMakeLists.txt rename to src/dawnphysics/component/physics/CMakeLists.txt diff --git a/src/dawn/component/physics/Collider.cpp b/src/dawnphysics/component/physics/Collider.cpp similarity index 100% rename from src/dawn/component/physics/Collider.cpp rename to src/dawnphysics/component/physics/Collider.cpp diff --git a/src/dawn/component/physics/Collider.hpp b/src/dawnphysics/component/physics/Collider.hpp similarity index 100% rename from src/dawn/component/physics/Collider.hpp rename to src/dawnphysics/component/physics/Collider.hpp diff --git a/src/dawn/component/physics/CubeCollider.cpp b/src/dawnphysics/component/physics/CubeCollider.cpp similarity index 100% rename from src/dawn/component/physics/CubeCollider.cpp rename to src/dawnphysics/component/physics/CubeCollider.cpp diff --git a/src/dawn/component/physics/CubeCollider.hpp b/src/dawnphysics/component/physics/CubeCollider.hpp similarity index 100% rename from src/dawn/component/physics/CubeCollider.hpp rename to src/dawnphysics/component/physics/CubeCollider.hpp diff --git a/src/dawn/component/physics/SphereCollider.cpp b/src/dawnphysics/component/physics/SphereCollider.cpp similarity index 100% rename from src/dawn/component/physics/SphereCollider.cpp rename to src/dawnphysics/component/physics/SphereCollider.cpp diff --git a/src/dawn/component/physics/SphereCollider.hpp b/src/dawnphysics/component/physics/SphereCollider.hpp similarity index 100% rename from src/dawn/component/physics/SphereCollider.hpp rename to src/dawnphysics/component/physics/SphereCollider.hpp diff --git a/src/dawn/physics/CMakeLists.txt b/src/dawnphysics/physics/CMakeLists.txt similarity index 100% rename from src/dawn/physics/CMakeLists.txt rename to src/dawnphysics/physics/CMakeLists.txt diff --git a/src/dawn/physics/PhysicsManager.cpp b/src/dawnphysics/physics/PhysicsManager.cpp similarity index 100% rename from src/dawn/physics/PhysicsManager.cpp rename to src/dawnphysics/physics/PhysicsManager.cpp diff --git a/src/dawn/physics/PhysicsManager.hpp b/src/dawnphysics/physics/PhysicsManager.hpp similarity index 100% rename from src/dawn/physics/PhysicsManager.hpp rename to src/dawnphysics/physics/PhysicsManager.hpp diff --git a/src/dawnrpg/CMakeLists.txt b/src/dawnrpg/CMakeLists.txt index 17a91e7b..0b0627f9 100644 --- a/src/dawnrpg/CMakeLists.txt +++ b/src/dawnrpg/CMakeLists.txt @@ -15,5 +15,5 @@ add_subdirectory(game) add_subdirectory(scene) # Assets -tool_texture(rosa FILE=rosa.png) +tool_texture(rosa rosa.png) tool_copy(test_scene test_scene.json) \ No newline at end of file diff --git a/tools/assetstool/assetstool.py b/tools/assetstool/assetstool.py index 2b71bafe..06f62e81 100755 --- a/tools/assetstool/assetstool.py +++ b/tools/assetstool/assetstool.py @@ -27,8 +27,9 @@ if not os.path.exists(os.path.dirname(args.output)): filesInArchive = [] if os.path.exists(args.output) and False: +# if os.path.exists(args.output): # Yes, open it - archive = tarfile.open(args.output, 'r:') + archive = tarfile.open(args.output, 'a:') # Get all the files in the archive for member in archive.getmembers(): @@ -51,11 +52,16 @@ for foldername, subfolders, filenames in os.walk(args.input): relative_path = os.path.relpath(absolute_path, args.input) if relative_path in filesInArchive: - # Yes, skip it - continue - - # No, add it - print(f"Archiving asset {filename}...") + if relative_path.endswith('.texture'): + print(f"Skipping {filename}...") + continue + else: + print(f"Overwriting {filename}...") + # Does not work in python, throw error + exit (1) + else: + print(f"Archiving asset {filename}...") + archive.add(absolute_path, arcname=relative_path) # Close the archive diff --git a/tools/texturetool/CMakeLists.txt b/tools/texturetool/CMakeLists.txt index 4c16a725..d870cdc3 100644 --- a/tools/texturetool/CMakeLists.txt +++ b/tools/texturetool/CMakeLists.txt @@ -1,8 +1,7 @@ find_package(Python3 REQUIRED COMPONENTS Interpreter) -function(tool_texture target) +function(tool_texture target file) # Defaults - set(FILE "" ) set(FILTER_MIN "") set(FILTER_MAG "") set(WRAP_X "") @@ -23,13 +22,13 @@ function(tool_texture target) endforeach() # Check for missing args - if(NOT DEFINED FILE) + if(NOT DEFINED file) message(FATAL_ERROR "Missing FILE input") endif() add_custom_target(${target}_texture COMMAND ${DAWN_TOOLS_DIR}/texturetool/texturetool.py - --input="${DAWN_ASSETS_SOURCE_DIR}/${FILE}" + --input="${DAWN_ASSETS_SOURCE_DIR}/${file}" --output="${DAWN_ASSETS_BUILD_DIR}/${target}.texture" --wrap-x="${WRAP_X}" --wrap-y="${WRAP_Y}"