TAMSVIZ
Visualization and annotation tool for ROS
texture.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "texture.h"
5 
6 #include "../core/log.h"
7 
8 #include <opencv2/opencv.hpp>
9 
10 #include <chrono>
11 #include <thread>
12 
13 struct TextureData {
14  cv::Mat image;
15  TextureData(const std::string &url) {
16  if (url.empty()) {
17  return;
18  }
19  std::vector<uint8_t> data;
20  try {
21  loadResource(url, data);
22  } catch (std::exception &ex) {
23  LOG_ERROR("failed to read texture file " << url);
24  return;
25  }
26  if (data.empty()) {
27  LOG_ERROR("texture file is empty " << url);
28  return;
29  }
30  auto img = cv::imdecode(data, cv::IMREAD_COLOR);
31  if (img.data == nullptr) {
32  LOG_ERROR("failed to decode texture " << url);
33  return;
34  }
35  image = img;
36  }
37 };
38 
39 Texture::Texture(const std::string &url, TextureType type)
40  : _url(url), _type(type), _loader(url) {}
41 
42 void TextureBase::destroy() {
43  if (_id) {
44  GLuint texture = _id;
45  cleanup([texture]() { V_GL(glDeleteTextures(1, &texture)); });
46  _id = 0;
47  }
48 }
49 
50 void TextureBase::create() {
51  if (!_id) {
52  V_GL(glGenTextures(1, &_id));
53  }
54 }
55 
56 GLuint Texture::update(const cv::Mat &img) {
57  cv::Mat image = img;
58  cv::flip(image, image, 0);
59  create();
60  V_GL(glActiveTexture(GL_TEXTURE0));
61  V_GL(glBindTexture(GL_TEXTURE_2D, _id));
62  switch (image.elemSize()) {
63  case 1:
64  V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 1));
65  break;
66  case 2:
67  V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 2));
68  break;
69  case 3:
70  V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 1));
71  break;
72  default:
73  V_GL(glPixelStorei(GL_UNPACK_ALIGNMENT, 4));
74  break;
75  }
76  V_GL(glPixelStorei(GL_UNPACK_ROW_LENGTH, image.step / image.elemSize()));
77  if (_mipmap) {
78  V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER,
79  GL_LINEAR_MIPMAP_LINEAR));
80  } else {
81  V_GL(glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR));
82  }
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()) {
88  case 3:
89  V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_SRGB8, image.cols, image.rows, 0,
90  GL_BGR, GL_UNSIGNED_BYTE, image.data));
91  break;
92  case 4:
93  V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_SRGB8_ALPHA8, image.cols,
94  image.rows, 0, GL_BGRA, GL_UNSIGNED_BYTE, image.data));
95  break;
96  default:
97  LOG_ERROR("invalid channel count " << image.channels()
98  << " for color image " << _url);
99  }
100  }
101  if (_type == TextureType::Normal) {
102  switch (image.channels()) {
103  case 3:
104  V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB8, image.cols, image.rows, 0,
105  GL_BGR, GL_UNSIGNED_BYTE, image.data));
106  break;
107  case 4:
108  V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, image.cols, image.rows, 0,
109  GL_BGRA, GL_UNSIGNED_BYTE, image.data));
110  break;
111  default:
112  LOG_ERROR("invalid channel count " << image.channels()
113  << " for normal map " << _url);
114  }
115  }
116  if (_type == TextureType::Linear) {
117  switch (image.channels()) {
118  case 1:
119  V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_R8, image.cols, image.rows, 0,
120  GL_RED, GL_UNSIGNED_BYTE, image.data));
121  break;
122  case 3:
123  V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB8, image.cols, image.rows, 0,
124  GL_BGR, GL_UNSIGNED_BYTE, image.data));
125  break;
126  case 4:
127  V_GL(glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, image.cols, image.rows, 0,
128  GL_BGRA, GL_UNSIGNED_BYTE, image.data));
129  break;
130  default:
131  LOG_ERROR("invalid channel count " << image.channels()
132  << " for color image " << _url);
133  }
134  }
135  if (_mipmap) {
136  V_GL(glGenerateMipmap(GL_TEXTURE_2D));
137  }
138  V_GL(glBindTexture(GL_TEXTURE_2D, 0));
139  return _id;
140 }
141 
142 GLuint Texture::update(int width, int height, int format, int samples) {
143  create();
144  if (_watcher.changed(width, height, format, samples)) {
145  V_GL(glActiveTexture(GL_TEXTURE0));
146  if (samples <= 0) {
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));
154  } else {
155  V_GL(glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, _id));
156  V_GL(glTexImage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, samples, format,
157  width, height, true));
158  }
159  }
160  return _id;
161 }
162 
163 GLuint Texture::update() {
164  if (!_url.empty() && invalidated()) {
165  _loaded = false;
166  _loader.clear();
167  destroy();
168  }
169  if (!_loaded) {
170  if (_url.empty()) {
171  _loaded = true;
172  } else {
173  if (_loader.load()) {
174  _loaded = true;
175  if (auto image = _loader.load()) {
176  if (image->image.rows) {
177  update(image->image);
178  }
179  }
180  _loader.clear();
181  }
182  }
183  }
184  return _id;
185 }
186 
187 TextureManager &Texture::manager() {
188  static TextureManager instance;
189  return instance;
190 }