#pragma once #include // IWYU pragma: keep #include #include #include #include // NOLINTNEXTLINE(readability-identifier-naming) template class glArraysBase { static_assert(N > 0); public: ~glArraysBase() = default; NO_COPY(glArraysBase); CUSTOM_MOVE(glArraysBase); // NOLINTNEXTLINE(hicpp-explicit-conversions) operator GLuint() const requires(N == 1) { return ids.front(); } GLuint operator*() const requires(N == 1) { return ids.front(); } const auto & operator[](size_t n) const { return ids[n]; } constexpr static auto size() { return N; } protected: glArraysBase() noexcept = default; std::array ids {}; }; template inline glArraysBase::glArraysBase(glArraysBase && src) noexcept : ids {src.ids} { std::fill(src.ids.begin(), src.ids.end(), 0); } template inline glArraysBase & glArraysBase::operator=(glArraysBase && src) noexcept { ids = src.ids; std::fill(src.ids.begin(), src.ids.end(), 0); return *this; } // NOLINTNEXTLINE(readability-identifier-naming) template class glArrays : public glArraysBase { public: using glArraysBase::glArraysBase; using glArraysBase::operator=; DEFAULT_MOVE_COPY(glArrays); glArrays() noexcept { (*Gen)(N, this->ids.data()); } ~glArrays() noexcept { if (this->ids.front()) { (*Del)(N, this->ids.data()); } } }; // NOLINTBEGIN(readability-identifier-naming) template using glVertexArrays = glArrays; using glVertexArray = glVertexArrays<1>; template using glBuffers = glArrays; using glBuffer = glBuffers<1>; template using glTextures = glArrays; using glTexture = glTextures<1>; template using glFrameBuffers = glArrays; using glFrameBuffer = glFrameBuffers<1>; template using glRenderBuffers = glArrays; using glRenderBuffer = glRenderBuffers<1>; // NOLINTEND(readability-identifier-naming)