6 #include "../core/log.h" 8 #include <opencv2/opencv.hpp> 19 std::vector<uint8_t> data;
21 loadResource(url, data);
22 }
catch (std::exception &ex) {
23 LOG_ERROR(
"failed to read texture file " << url);
27 LOG_ERROR(
"texture file is empty " << url);
30 auto img = cv::imdecode(data, cv::IMREAD_COLOR);
31 if (img.data ==
nullptr) {
32 LOG_ERROR(
"failed to decode texture " << url);
39 Texture::Texture(
const std::string &url, TextureType type)
40 : _url(url), _type(type), _loader(url) {}
42 void TextureBase::destroy() {
45 cleanup([texture]() { V_GL(glDeleteTextures(1, &texture)); });
50 void TextureBase::create() {
52 V_GL(glGenTextures(1, &_id));
56 GLuint Texture::update(
const cv::Mat &img) {
58 cv::flip(image, image, 0);
60 V_GL(glActiveTexture(GL_TEXTURE0));
61 V_GL(glBindTexture(GL_TEXTURE_2D, _id));
62 switch (image.elemSize()) {
64 V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 1));
67 V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 2));
70 V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 1));
73 V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 4));
76 V_GL(glPixelStorei(GL_UNPACK_ROW_LENGTH, image.step / image.elemSize()));
78 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER,
79 GL_LINEAR_MIPMAP_LINEAR));
81 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR));
83 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR));
84 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT));
85 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT));
86 if (_type == TextureType::Color) {
87 switch (image.channels()) {
89 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_SRGB8, image.cols, image.rows, 0,
90 GL_BGR, GL_UNSIGNED_BYTE, image.data));
93 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_SRGB8_ALPHA8, image.cols,
94 image.rows, 0, GL_BGRA, GL_UNSIGNED_BYTE, image.data));
97 LOG_ERROR(
"invalid channel count " << image.channels()
98 <<
" for color image " << _url);
101 if (_type == TextureType::Normal) {
102 switch (image.channels()) {
104 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB8, image.cols, image.rows, 0,
105 GL_BGR, GL_UNSIGNED_BYTE, image.data));
108 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, image.cols, image.rows, 0,
109 GL_BGRA, GL_UNSIGNED_BYTE, image.data));
112 LOG_ERROR(
"invalid channel count " << image.channels()
113 <<
" for normal map " << _url);
116 if (_type == TextureType::Linear) {
117 switch (image.channels()) {
119 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_R8, image.cols, image.rows, 0,
120 GL_RED, GL_UNSIGNED_BYTE, image.data));
123 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB8, image.cols, image.rows, 0,
124 GL_BGR, GL_UNSIGNED_BYTE, image.data));
127 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, image.cols, image.rows, 0,
128 GL_BGRA, GL_UNSIGNED_BYTE, image.data));
131 LOG_ERROR(
"invalid channel count " << image.channels()
132 <<
" for color image " << _url);
136 V_GL(glGenerateMipmap(GL_TEXTURE_2D));
138 V_GL(glBindTexture(GL_TEXTURE_2D, 0));
142 GLuint Texture::update(
int width,
int height,
int format,
int samples) {
144 if (_watcher.changed(width, height, format, samples)) {
145 V_GL(glActiveTexture(GL_TEXTURE0));
147 V_GL(glBindTexture(GL_TEXTURE_2D, _id));
148 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE));
149 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE));
150 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST));
151 V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST));
152 V_GL(glTexImage2D(GL_TEXTURE_2D, 0, format, width, height, 0, GL_BGRA,
153 GL_UNSIGNED_BYTE,
nullptr));
155 V_GL(glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, _id));
156 V_GL(glTexImage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, samples, format,
157 width, height,
true));
163 GLuint Texture::update() {
164 if (!_url.empty() && invalidated()) {
173 if (_loader.load()) {
175 if (
auto image = _loader.load()) {
176 if (image->image.rows) {
177 update(image->image);