#pragma once #include "../containers.hpp" #include #include #define STB_IMAGE_IMPLEMENTATION #include #ifndef __gl_h_ #include #endif namespace hpr::gpu { class Texture { public: using ustring = std::basic_string; public: enum class Format { RGB = GL_RGB, RGBA = GL_RGBA }; protected: unsigned int p_index; std::string p_filename; ustring p_source; int p_width; int p_height; Format p_internalFormat; Format p_imageFormat; public: inline Texture() : p_index {0}, p_filename {}, p_source {}, p_width {}, p_height {}, p_internalFormat {Format::RGBA}, p_imageFormat {Format::RGBA} {} inline Texture(const std::string& filename) : p_index {0}, p_filename {filename}, p_source {}, p_width {}, p_height {}, p_internalFormat {Format::RGB}, p_imageFormat {Format::RGB} {} virtual ~Texture() = default; [[nodiscard]] unsigned int index() const { return p_index; } [[nodiscard]] unsigned int width() const { return p_width; } [[nodiscard]] unsigned int height() const { return p_height; } void active() { glActiveTexture(GL_TEXTURE0 + p_index); } void bind() { glBindTexture(GL_TEXTURE_2D, p_index); } void unbind() { glBindTexture(GL_TEXTURE_2D, 0); } void alphaChannel(bool enabled) { if (enabled) { p_internalFormat = Format::RGBA; p_imageFormat = Format::RGBA; } else { p_internalFormat = Format::RGB; p_imageFormat = Format::RGB; } } void load(const std::string& filename) { auto filepath = std::filesystem::canonical(std::filesystem::path(filename)).string(); stbi_set_flip_vertically_on_load(true); int channelsCount; unsigned char* source = stbi_load(filepath.c_str(), &p_width, &p_height, &channelsCount, 0); if (!source) throw std::runtime_error("Failed to load texture source"); else create(source); stbi_image_free(source); } void load() { load(p_filename); } void create() { glGenTextures(1, &p_index); bind(); glTexImage2D(GL_TEXTURE_2D, 0, (GLint)p_internalFormat, p_width, p_height, 0, (GLint)p_imageFormat, GL_UNSIGNED_BYTE, nullptr); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); unbind(); } void create(const ustring& source) { glGenTextures(1, &p_index); bind(); glTexImage2D(GL_TEXTURE_2D, 0, (GLint)p_internalFormat, p_width, p_height, 0, (GLint)p_imageFormat, GL_UNSIGNED_BYTE, source.data()); p_source = source; glGenerateMipmap(GL_TEXTURE_2D); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); unbind(); } void rescale(int width, int height) { bind(); glTexImage2D(GL_TEXTURE_2D, 0, (GLint)p_internalFormat, p_width, p_height, 0, (GLint)p_imageFormat, GL_UNSIGNED_BYTE, !p_source.empty() ? p_source.data() : nullptr); unbind(); } void destroy() { glDeleteTextures(1, &p_index); } bool valid() const { return p_index != 0; } }; }