TAMSVIZ
Visualization and annotation tool for ROS
text.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "text.h"
5 
6 #include "../core/workspace.h"
7 #include "../render/mesh.h"
8 #include "../render/renderlist.h"
9 #include "../render/texture.h"
10 #include "shapes.h"
11 
12 #include <QFont>
13 #include <QPainter>
14 #include <QPixmap>
15 #include <QRawFont>
16 #include <QTextLayout>
17 
18 #include <opencv2/core/eigen.hpp>
19 #include <opencv2/opencv.hpp>
20 
21 #include <unordered_map>
22 #include <unordered_set>
23 
24 class Glyph {
25  cv::Mat _image;
26  QPointF _origin;
27  QRectF _rect;
28 
29 public:
30  const cv::Mat &image() const { return _image; }
31  const QPointF &origin() const { return _origin; }
32  const QRectF &rect() const { return _rect; }
33  Glyph(QFont font, int font_size, uint32_t glyph) {
34 
35  int oversampling = 4;
36  int margin = oversampling * 2;
37 
38  font.setPixelSize(font_size);
39 
40  QRawFont raw_font = QRawFont::fromFont(font);
41 
42  QPainterPath path = raw_font.pathForGlyph(glyph);
43 
44  QRect rect = path.boundingRect()
45  .marginsAdded(QMarginsF(margin, margin, margin, margin))
46  .toAlignedRect();
47 
48  _rect = rect;
49  _origin = -rect.topLeft();
50 
51  rect = QRect(rect.x() * oversampling, rect.y() * oversampling,
52  rect.width() * oversampling, rect.height() * oversampling);
53 
54  path = QTransform().scale(oversampling, oversampling).map(path);
55 
56  path.translate(-rect.topLeft());
57 
58  QPixmap pixmap(rect.width(), rect.height());
59  pixmap.fill(QColor(0, 0, 0, 255));
60  {
61  QPainter painter(&pixmap);
62  painter.setRenderHint(QPainter::Antialiasing, true);
63  painter.setFont(font);
64  painter.setPen(QPen(QBrush(Qt::white), 1));
65  painter.setBrush(QBrush(Qt::white));
66  painter.fillPath(path, QBrush(Qt::white));
67  }
68 
69  QImage image = pixmap.toImage();
70  cv::Mat mat(image.height(), image.width(), CV_8UC4,
71  const_cast<uint8_t *>(image.bits()),
72  static_cast<size_t>(image.bytesPerLine()));
73 
74  cv::cvtColor(mat, mat, cv::COLOR_RGB2GRAY);
75 
76  {
77  cv::Mat mat2 = mat * 0.5;
78  size_t n = oversampling * 2;
79  cv::Mat m;
80  for (size_t i = 1; i <= n; i++) {
81  cv::Mat kernel = cv::getStructuringElement(
82  cv::MORPH_ELLIPSE, cv::Point(i * 2 + 1, i * 2 + 1));
83  cv::erode(mat, m, kernel);
84  mat2 = cv::max(mat2, m * (0.5 + i * 0.5 / n));
85  cv::dilate(mat, m, kernel);
86  mat2 = cv::max(mat2, m * (0.5 - i * 0.5 / n));
87  }
88  mat = mat2;
89  }
90 
91  cv::resize(mat, mat, cv::Size(), 1.0 / oversampling, 1.0 / oversampling,
92  cv::INTER_AREA);
93 
94  _image = mat;
95  }
96  static std::shared_ptr<Glyph> instance(const QFont &font, int font_size,
97  uint32_t glyph) {
98  auto key = std::make_tuple(font, font_size, glyph);
99  static std::mutex mutex;
100  std::lock_guard<std::mutex> lock(mutex);
101  static std::map<std::tuple<QFont, int, uint32_t>, std::shared_ptr<Glyph>>
102  cache;
103  if (cache.size() > 1000) {
104  LOG_DEBUG("clearing glyph cache");
105  cache.clear();
106  }
107  auto &v = cache[key];
108  if (!v) {
109  LOG_DEBUG("rendering glyph " << glyph);
110  v = std::make_shared<Glyph>(font, font_size, glyph);
111  }
112  return v;
113  }
114 };
115 
116 class Font {
117  std::shared_ptr<Texture> _texture;
118  cv::Mat _image;
119  std::vector<QRectF> _texture_rects;
120  std::vector<QRectF> _mesh_rects;
121  std::vector<QPointF> _origins;
122  std::unordered_map<uint32_t, size_t> _glyph_indices;
123  QFont _font;
124  int _font_size = 0;
125  static double pack(std::vector<QRect *> rr, int width) {
126  // LOG_DEBUG("pack " << width);
127  int row_start = 0;
128  int y = 0;
129  while (true) {
130  int row_rects = 0;
131  int row_width = 0;
132  while (row_start + row_rects < rr.size()) {
133  auto *r = rr[row_start + row_rects];
134  if (row_width + r->width() < width) {
135  row_width += r->width();
136  row_rects++;
137  } else {
138  break;
139  }
140  }
141  if (row_rects == 0) {
142  break;
143  }
144  int y2 = y;
145  int x = 0;
146  for (int j = row_start; j < row_start + row_rects; j++) {
147  auto *rect = rr[j];
148  rect->translate(x - rect->x(), y - rect->y());
149  x += rect->width();
150  y2 = std::max(y2, y + rect->height());
151  }
152  y = y2;
153  row_start += row_rects;
154  }
155  // LOG_DEBUG("cost " << y + width);
156  return y + width;
157  }
158  static void pack(std::vector<QRect> &rects) {
159  std::vector<QRect *> rr;
160  for (auto &rect : rects) {
161  rr.push_back(&rect);
162  }
163  std::sort(rr.begin(), rr.end(),
164  [&](QRect *a, QRect *b) { return a->height() < b->height(); });
165  int width = 1;
166  for (auto &r : rects) {
167  width = std::max(width, r.width() + 1);
168  }
169  for (int w = width; w < 1024 * 32; w *= 2) {
170  if (pack(rr, w) < pack(rr, width)) {
171  width = w;
172  }
173  }
174  pack(rr, width);
175  }
176 
177 public:
178  static std::shared_ptr<Font>
179  instance(const QFont &font, int font_size,
180  const std::unordered_set<uint32_t> &new_glyph_indices) {
181  static std::mutex mutex;
182  std::lock_guard<std::mutex> lock(mutex);
183  static std::shared_ptr<Font> instance;
184  auto merged_glyph_indices = new_glyph_indices;
185  if (instance) {
186  for (auto &p : instance->_glyph_indices) {
187  merged_glyph_indices.insert(p.first);
188  }
189  if (merged_glyph_indices.size() > 200) {
190  LOG_DEBUG("clearing glyph set");
191  merged_glyph_indices = new_glyph_indices;
192  }
193  }
194  if (instance) {
195  if (instance->_font != font || instance->_font_size != font_size) {
196  instance = nullptr;
197  }
198  }
199  if (instance) {
200  for (auto &i : new_glyph_indices) {
201  if (instance->_glyph_indices.find(i) ==
202  instance->_glyph_indices.end()) {
203  instance = nullptr;
204  break;
205  }
206  }
207  }
208  if (!instance) {
209  LOG_DEBUG("building font texture for " << merged_glyph_indices.size()
210  << " glyphs");
211  instance = std::make_shared<Font>(font, font_size, merged_glyph_indices);
212  } else {
213  LOG_DEBUG("re-using font texture");
214  }
215  return instance;
216  }
217 
218  Font(const QFont &font, int font_size,
219  const std::unordered_set<uint32_t> &glyph_indices) {
220 
221  _font = font;
222  _font_size = font_size;
223 
224  std::vector<std::shared_ptr<Glyph>> glyphs;
225  for (auto &glyph_index : glyph_indices) {
226  auto glyph = Glyph::instance(font, font_size, glyph_index);
227  _glyph_indices[glyph_index] = glyphs.size();
228  glyphs.emplace_back(glyph);
229  }
230 
231  std::vector<QRect> pack_rects;
232  {
233  for (auto &glyph : glyphs) {
234  pack_rects.emplace_back(0, 0, glyph->image().cols, glyph->image().rows);
235  }
236  }
237  pack(pack_rects);
238 
239  int width = 0;
240  int height = 0;
241  for (auto &rect : pack_rects) {
242  width = std::max(width, rect.right() + 1);
243  height = std::max(height, rect.bottom() + 1);
244  }
245  LOG_DEBUG("font texture size " << width << " " << height);
246  cv::Mat image(height, width, CV_8UC1);
247  image.setTo(cv::Scalar(0));
248 
249  for (size_t i = 0; i < glyphs.size(); i++) {
250  auto &rect = pack_rects[i];
251  auto &glyph = glyphs[i];
252  glyph->image().copyTo(
253  image(cv::Rect(rect.x(), rect.y(), rect.width(), rect.height())));
254  _mesh_rects.emplace_back(glyph->rect());
255  _texture_rects.emplace_back(
256  rect.x() * 1.0 / width, rect.y() * 1.0 / height,
257  rect.width() * 1.0 / width, rect.height() * 1.0 / height);
258  _origins.emplace_back(glyph->origin());
259  }
260 
261  _image = image;
262  }
263 
264  const std::shared_ptr<Texture> &texture() {
265  if (!_texture) {
266  _texture = std::make_shared<Texture>(TextureType::Linear);
267  _texture->mipmap(false);
268  _texture->update(_image);
269  }
270  return _texture;
271  }
272 
273  const QRectF &textureRect(uint32_t glyph) const {
274  auto iter = _glyph_indices.find(glyph);
275  if (iter != _glyph_indices.end()) {
276  return _texture_rects[iter->second];
277  } else {
278  throw std::runtime_error("glyph error");
279  }
280  }
281 
282  const QRectF &meshRect(uint32_t glyph) const {
283  auto iter = _glyph_indices.find(glyph);
284  if (iter != _glyph_indices.end()) {
285  return _mesh_rects[iter->second];
286  } else {
287  throw std::runtime_error("glyph error");
288  }
289  }
290 
291  const QPointF &origin(uint32_t glyph) const {
292  auto iter = _glyph_indices.find(glyph);
293  if (iter != _glyph_indices.end()) {
294  return _origins[iter->second];
295  } else {
296  throw std::runtime_error("glyph error");
297  }
298  }
299 };
300 
301 TextRenderer::TextRenderer(const std::shared_ptr<Material> &material) {
302  if (material) {
303  _material = material;
304  } else {
305  ObjectScope ws;
306  _material = std::make_shared<Material>();
307  }
308  _material_renderer = std::make_shared<MaterialRenderer>(_material);
309 }
310 
311 void TextRenderer::renderSync(const RenderSyncContext &context) {
312  _material_renderer->renderSync(context);
313  _parent_pose = context.pose;
314  SceneNode::renderSync(context);
315 }
316 
317 void TextRenderer::renderAsync(const RenderAsyncContext &context) {
318 
319  _material_renderer->renderAsync(context);
320 
321  if (_watcher.changed(_view_facing, _text, _size, _offset)) {
322 
323  if (_text.empty()) {
324 
325  _mesh = nullptr;
326  _texture = nullptr;
327 
328  } else {
329 
330  int font_size = 64;
331 
332  QFont font;
333  font.setBold(true);
334  font.setPixelSize(font_size);
335 
336  QTextLayout text_layout;
337  text_layout.setFont(font);
338  text_layout.setText(text().c_str());
339  text_layout.beginLayout();
340  while (true) {
341  auto line = text_layout.createLine();
342  if (line.isValid()) {
343  line.setLineWidth(1000000000);
344  } else {
345  break;
346  }
347  }
348  text_layout.endLayout();
349 
350  std::unordered_set<uint32_t> glyph_indices;
351  for (auto &glyph_run : text_layout.glyphRuns()) {
352  for (auto &glyph_index : glyph_run.glyphIndexes()) {
353  glyph_indices.insert(glyph_index);
354  }
355  }
356 
357  auto texture_font = Font::instance(font, font_size, glyph_indices);
358 
359  _texture = texture_font->texture();
360 
361  MeshData mesh;
362 
363  for (auto &glyph_run : text_layout.glyphRuns()) {
364  for (size_t i = 0; i < glyph_run.positions().size(); i++) {
365 
366  auto pos = glyph_run.positions()[i];
367  auto index = glyph_run.glyphIndexes()[i];
368  pos -= texture_font->origin(index);
369  auto tex_rect = texture_font->textureRect(index);
370  auto mesh_rect = texture_font->meshRect(index);
371  auto bounds = glyph_run.rawFont().boundingRect(index);
372  auto origin = texture_font->origin(index);
373 
374  mesh_rect = QRectF(pos.x(), mesh_rect.y(), mesh_rect.width(),
375  mesh_rect.height());
376 
377  mesh.indices.emplace_back(mesh.positions.size() + 0);
378  mesh.indices.emplace_back(mesh.positions.size() + 2);
379  mesh.indices.emplace_back(mesh.positions.size() + 1);
380  mesh.indices.emplace_back(mesh.positions.size() + 0);
381  mesh.indices.emplace_back(mesh.positions.size() + 3);
382  mesh.indices.emplace_back(mesh.positions.size() + 2);
383 
384  mesh.texcoords.emplace_back(tex_rect.left(), 1.0 - tex_rect.bottom());
385  mesh.texcoords.emplace_back(tex_rect.left(), 1.0 - tex_rect.top());
386  mesh.texcoords.emplace_back(tex_rect.right(), 1.0 - tex_rect.top());
387  mesh.texcoords.emplace_back(tex_rect.right(),
388  1.0 - tex_rect.bottom());
389 
390  mesh.positions.emplace_back(mesh_rect.left(), -mesh_rect.bottom(), 0);
391  mesh.positions.emplace_back(mesh_rect.left(), -mesh_rect.top(), 0);
392  mesh.positions.emplace_back(mesh_rect.right(), -mesh_rect.top(), 0);
393  mesh.positions.emplace_back(mesh_rect.right(), -mesh_rect.bottom(),
394  0);
395  }
396  }
397 
398  mesh.scale(_size * 1.0 / font_size);
399 
400  if (mesh.positions.size()) {
401  float xmin = mesh.positions.front().x();
402  float xmax = mesh.positions.front().x();
403 
404  float ymin = mesh.positions.front().y();
405  float ymax = mesh.positions.front().y();
406 
407  for (auto &p : mesh.positions) {
408  xmin = std::min(xmin, p.x());
409  ymin = std::min(ymin, p.y());
410 
411  xmax = std::max(xmax, p.x());
412  ymax = std::max(ymax, p.y());
413  }
414 
415  for (auto &p : mesh.positions) {
416  p.x() -= (xmin + xmax) * 0.5;
417  }
418  }
419 
420  mesh.translate(Eigen::Vector3f(_offset.x(), _offset.y(), 0.0));
421 
422  if (_view_facing) {
423  for (auto &p : mesh.positions) {
424  mesh.extras.emplace_back(p.x(), p.y(), 3, 0);
425  p = Eigen::Vector3f(0, 0, 0);
426  }
427  }
428 
429  _mesh = std::make_shared<Mesh>(mesh);
430  }
431  }
432 
433  if (_mesh) {
434 
435  MaterialBlock material = _material_renderer->block();
436  material.color_texture = _texture->id();
437  material.flags |= 1;
438  material.flags |= 2;
439  context.render_list->push(material);
440 
441  RenderOptions options;
442  if (!_view_facing) {
443  options.double_sided = true;
444  }
445  context.render_list->push(_mesh, options);
446 
447  InstanceBlock instance;
448  instance.setPose(_parent_pose.matrix().cast<float>());
449 
450  context.render_list->push(instance);
451  }
452 
453  SceneNode::renderAsync(context);
454 }
455 
456 bool TextRenderer::pick(uint32_t id) const {
457  return _material_renderer->id() == id;
458 }
459 
460 TextDisplay::TextDisplay() {
461  _renderer = node()->create<TextRenderer>(_material);
462 }
463 
464 void TextDisplay::renderSync(const RenderSyncContext &context) {
465  _material->color() = color();
466  _material->opacity() = opacity();
467  _renderer->text(text());
468  _renderer->offset(offset());
469  _renderer->size(size());
470  _renderer->viewFacing(viewFacing());
471  MeshDisplayBase::renderSync(context);
472 }
Definition: text.cpp:116
Definition: text.cpp:24
Definition: mesh.h:11