// Minetest // SPDX-License-Identifier: LGPL-2.1-or-later #pragma once #include "SkinnedMesh.h" #include "IMeshLoader.h" #include "IReadFile.h" #include "irrTypes.h" #include "path.h" #include "S3DVertex.h" #include "tiniergltf.hpp" #include #include #include #include namespace irr { namespace scene { class CGLTFMeshFileLoader : public IMeshLoader { public: CGLTFMeshFileLoader() noexcept {}; bool isALoadableFileExtension(const io::path &filename) const override; IAnimatedMesh *createMesh(io::IReadFile *file) override; private: template static T rawget(const char *ptr); template class Accessor { struct BufferSource { const char *ptr; std::size_t byteStride; }; using Source = std::variant, std::tuple<>>; public: static Accessor sparseIndices( const tiniergltf::GlTF &model, const tiniergltf::AccessorSparseIndices &indices, const std::size_t count); static Accessor sparseValues( const tiniergltf::GlTF &model, const tiniergltf::AccessorSparseValues &values, const std::size_t count, const std::size_t defaultByteStride); static Accessor base( const tiniergltf::GlTF &model, std::size_t accessorIdx); static Accessor make(const tiniergltf::GlTF &model, std::size_t accessorIdx); static constexpr tiniergltf::Accessor::Type getType(); static constexpr tiniergltf::Accessor::ComponentType getComponentType(); std::size_t getCount() const { return count; } T get(std::size_t i) const; private: Accessor(const char *ptr, std::size_t byteStride, std::size_t count) : source(BufferSource{ptr, byteStride}), count(count) {} Accessor(std::vector vec, std::size_t count) : source(vec), count(count) {} Accessor(std::size_t count) : source(std::make_tuple()), count(count) {} // Directly from buffer, sparse, or default-initialized const Source source; const std::size_t count; }; template using AccessorVariant = std::variant...>; template using ArrayAccessorVariant = std::variant>...>; template using NormalizedValuesAccessor = ArrayAccessorVariant; template static NormalizedValuesAccessor createNormalizedValuesAccessor( const tiniergltf::GlTF &model, const std::size_t accessorIdx); template static std::array getNormalizedValues( const NormalizedValuesAccessor &accessor, const std::size_t i); class MeshExtractor { public: MeshExtractor(tiniergltf::GlTF &&model, SkinnedMeshBuilder *mesh) noexcept : m_gltf_model(std::move(model)), m_irr_model(mesh) {}; /* Gets indices for the given mesh/primitive. * * Values are return in Irrlicht winding order. */ std::optional> getIndices( const tiniergltf::MeshPrimitive &primitive) const; std::optional> getVertices( const tiniergltf::MeshPrimitive &primitive) const; std::size_t getMeshCount() const; std::size_t getPrimitiveCount(const std::size_t meshIdx) const; void load(); const std::unordered_set &getWarnings() { return warnings; } private: const tiniergltf::GlTF m_gltf_model; SkinnedMeshBuilder *m_irr_model; std::vector> m_mesh_loaders; std::vector m_loaded_nodes; std::unordered_set warnings; void warn(const std::string &warning) { warnings.insert(warning); } void copyPositions(const std::size_t accessorIdx, std::vector& vertices) const; void copyNormals(const std::size_t accessorIdx, std::vector& vertices) const; void copyTCoords(const std::size_t accessorIdx, std::vector& vertices) const; void addPrimitive(const tiniergltf::MeshPrimitive &primitive, const std::optional skinIdx, SkinnedMesh::SJoint *parent); void deferAddMesh(const std::size_t meshIdx, const std::optional skinIdx, SkinnedMesh::SJoint *parentJoint); void loadNode(const std::size_t nodeIdx, SkinnedMesh::SJoint *parentJoint); void loadNodes(); void loadSkins(); void loadAnimation(const std::size_t animIdx); }; tiniergltf::GlTF parseGLTF(io::IReadFile *file); }; } // namespace scene } // namespace irr