/* * ===================================================================================== * * OpenMiner * * Copyright (C) 2018-2020 Unarelith, Quentin Bazin * Copyright (C) 2019-2020 the OpenMiner contributors (see CONTRIBUTORS.md) * * This file is part of OpenMiner. * * OpenMiner is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * OpenMiner is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with OpenMiner; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * ===================================================================================== */ #include #include #include #include "ComponentType.hpp" #include "Network.hpp" #include "NetworkUtils.hpp" #include "Scene.hpp" static bool areComponentsRegistered = false; Scene::Scene() { if (!areComponentsRegistered) { registerComponents(); areComponentsRegistered = true; } } void Scene::update() { for (auto &controller : m_controllers) controller->update(m_registry); } // Opaque entity stamper: https://github.com/skypjack/entt/issues/481#issuecomment-627614706 entt::entity Scene::createEntityFromModel(entt::registry &modelRegistry, entt::entity modelEntity) { if (!modelRegistry.valid(modelEntity)) { gkError() << "Failed to create entity from model: Model entity with ID" << (u32)modelEntity << "doesn't exist"; return entt::null; } auto other = m_registry.create(); modelRegistry.visit(modelEntity, [&](const auto &component) { const auto type = entt::resolve_type(component); const auto any = type.func("get"_hs).invoke({}, std::ref(modelRegistry), modelEntity); type.func("set"_hs).invoke({}, std::ref(m_registry), other, any); }); return other; } template T &get(entt::registry ®istry, entt::entity entity) { return registry.get_or_emplace(entity); } template T &set(entt::registry ®istry, entt::entity entity, const T &instance) { return registry.emplace_or_replace(entity, instance); } template Network::Packet serialize(entt::entity entity, T &component, bool useUpdateStatus) { Network::Packet packet; if constexpr(std::is_base_of_v>) { if (!useUpdateStatus || component.isUpdated) { packet << component.packetType << entity << component; if (useUpdateStatus) component.isUpdated = false; } } return packet; } template Network::Packet save(T &component) { Network::Packet packet; packet << component; return packet; } template void extend_meta_type(const std::string &name, ComponentType type, bool isSerializable, bool isSavable) { entt::meta().type() .prop("name"_hs, name) .prop("type"_hs, type) .prop("is_serializable"_hs, isSerializable) .prop("is_savable"_hs, isSavable) .template func<&get, entt::as_ref_t>("get"_hs) .template func<&set>("set"_hs) .template func<&serialize>("serialize"_hs) .template func<&save>("save"_hs); } #include #include "AnimationComponent.hpp" #include "DrawableDef.hpp" #include "ItemStack.hpp" #include "NetworkComponent.hpp" #include "PositionComponent.hpp" #include "RotationComponent.hpp" void Scene::registerComponents() { extend_meta_type ("gk::DoubleBox", ComponentType::Hitbox, false, true); extend_meta_type ("AnimationComponent", ComponentType::Animation, true, true); extend_meta_type ("DrawableDef", ComponentType::Drawable, true, true); extend_meta_type ("ItemStack", ComponentType::ItemStack, false, true); extend_meta_type ("NetworkComponent", ComponentType::Network, false, false); extend_meta_type ("PositionComponent", ComponentType::Position, true, true); extend_meta_type ("RotationComponent", ComponentType::Rotation, true, true); extend_meta_type ("EntityID", ComponentType::EntityID, false, true); }