#pragma once #include "drape/drape_global.hpp" #include "drape/graphics_context.hpp" #include "drape/pointers.hpp" #include "drape/render_state.hpp" #include #include #include #include namespace dp { class GpuProgram; class GraphicsContext; class MeshObjectImpl; namespace metal { class MetalMeshObjectImpl; } // namespace metal namespace vulkan { class VulkanMeshObjectImpl; } // namespace vulkan // This class implements a simple mesh object which does not use an index buffer. // Use this class only for simple geometry. class MeshObject { friend class MeshObjectImpl; friend class GLMeshObjectImpl; friend class metal::MetalMeshObjectImpl; friend class vulkan::VulkanMeshObjectImpl; public: enum class DrawPrimitive: uint8_t { Triangles, TriangleStrip, LineStrip }; MeshObject(ref_ptr context, DrawPrimitive drawPrimitive, std::string const & debugName = ""); virtual ~MeshObject(); template void SetBuffer(uint32_t bufferInd, std::vector && vertices, uint32_t stride = 0) { CHECK_LESS_OR_EQUAL(bufferInd, GetNextBufferIndex(), ()); if (bufferInd == GetNextBufferIndex()) m_buffers.emplace_back(make_unique_dp>(std::move(vertices), stride)); else m_buffers[bufferInd] = make_unique_dp>(std::move(vertices), stride); Reset(); } void SetAttribute(std::string const & attributeName, uint32_t bufferInd, uint32_t offset, uint32_t componentsCount, glConst type = gl_const::GLFloatType); template void UpdateBuffer(ref_ptr context, uint32_t bufferInd, std::vector const & vertices) { CHECK(m_initialized, ()); CHECK_LESS(bufferInd, static_cast(m_buffers.size()), ()); CHECK(!vertices.empty(), ()); auto & buffer = m_buffers[bufferInd]; CHECK_LESS_OR_EQUAL(static_cast(vertices.size() * sizeof(T)), buffer->GetSizeInBytes(), ()); memcpy(buffer->GetData(), vertices.data(), vertices.size() * sizeof(T)); UpdateImpl(context, bufferInd); } void SetIndexBuffer(std::vector && indices); void UpdateIndexBuffer(ref_ptr context, std::vector const & indices); template void Render(ref_ptr context, ref_ptr program, dp::RenderState const & state, ref_ptr paramsSetter, TParams const & params) { Bind(context, program); ApplyState(context, program, state); paramsSetter->Apply(context, program, params); DrawPrimitives(context); Unbind(program); } template void Render(ref_ptr context, ref_ptr program, dp::RenderState const & state, ref_ptr paramsSetter, TParams const & params, std::function && drawCallback) { Bind(context, program); ApplyState(context, program, state); paramsSetter->Apply(context, program, params); CHECK(drawCallback, ()); drawCallback(); Unbind(program); } uint32_t GetNextBufferIndex() const { return static_cast(m_buffers.size()); } bool IsInitialized() const { return m_initialized; } void Build(ref_ptr context, ref_ptr program); void Reset(); // Should be called inside draw callback in Render() method void DrawPrimitivesSubset(ref_ptr context, uint32_t vertexCount, uint32_t startVertex); void DrawPrimitivesSubsetIndexed(ref_ptr context, uint32_t indexCount, uint32_t startIndex); static std::vector GenerateNormalsForTriangles(std::vector const & vertices, size_t componentsCount); private: struct AttributeMapping { AttributeMapping() = default; AttributeMapping(std::string const & attributeName, uint32_t offset, uint32_t componentsCount, glConst type = gl_const::GLFloatType) : m_attributeName(attributeName) , m_offset(offset) , m_componentsCount(componentsCount) , m_type(type) {} std::string m_attributeName; uint32_t m_offset = 0; uint32_t m_componentsCount = 0; glConst m_type = gl_const::GLFloatType; }; class VertexBufferBase { public: virtual ~VertexBufferBase() = default; virtual void * GetData() = 0; virtual uint32_t GetSizeInBytes() const = 0; virtual uint32_t GetStrideInBytes() const = 0; uint32_t m_bufferId = 0; std::vector m_attributes; }; template class VertexBuffer : public VertexBufferBase { public: VertexBuffer() = default; VertexBuffer(std::vector && data, uint32_t stride = 0) : m_data(std::move(data)) , m_stride(stride == 0 ? sizeof(T) : stride) { CHECK_GREATER_OR_EQUAL(m_stride, sizeof(T), ()); } void * GetData() override { return m_data.data(); } uint32_t GetSizeInBytes() const override { return static_cast(m_data.size() * sizeof(T)); } uint32_t GetStrideInBytes() const override { return m_stride; } private: std::vector m_data; uint32_t m_stride = 0; }; void InitForOpenGL(); void InitForVulkan(ref_ptr context); #if defined(OMIM_METAL_AVAILABLE) // Definition of this method is in a .mm-file. void InitForMetal(); #endif void UpdateImpl(ref_ptr context, uint32_t bufferInd); void Bind(ref_ptr context, ref_ptr program); void Unbind(ref_ptr program); void DrawPrimitives(ref_ptr context); std::vector> m_buffers; std::vector m_indices; DrawPrimitive m_drawPrimitive = DrawPrimitive::Triangles; drape_ptr m_impl; bool m_initialized = false; std::string m_debugName; }; class MeshObjectImpl { public: virtual ~MeshObjectImpl() = default; virtual void Build(ref_ptr context, ref_ptr program) = 0; virtual void Reset() = 0; virtual void UpdateBuffer(ref_ptr context, uint32_t bufferInd) = 0; virtual void UpdateIndexBuffer(ref_ptr context) = 0; virtual void Bind(ref_ptr program) = 0; virtual void Unbind() = 0; virtual void DrawPrimitives(ref_ptr context, uint32_t verticesCount, uint32_t startVertex) = 0; virtual void DrawPrimitivesIndexed(ref_ptr context, uint32_t indexCount, uint32_t startIndex) = 0; }; } // namespace dp