6 #include "../core/workspace.h" 7 #include "../render/mesh.h" 8 #include "../render/renderlist.h" 9 #include "../render/texture.h" 16 #include <QTextLayout> 18 #include <opencv2/core/eigen.hpp> 19 #include <opencv2/opencv.hpp> 21 #include <unordered_map> 22 #include <unordered_set> 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) {
36 int margin = oversampling * 2;
38 font.setPixelSize(font_size);
40 QRawFont raw_font = QRawFont::fromFont(font);
42 QPainterPath path = raw_font.pathForGlyph(glyph);
44 QRect rect = path.boundingRect()
45 .marginsAdded(QMarginsF(margin, margin, margin, margin))
49 _origin = -rect.topLeft();
51 rect = QRect(rect.x() * oversampling, rect.y() * oversampling,
52 rect.width() * oversampling, rect.height() * oversampling);
54 path = QTransform().scale(oversampling, oversampling).map(path);
56 path.translate(-rect.topLeft());
58 QPixmap pixmap(rect.width(), rect.height());
59 pixmap.fill(QColor(0, 0, 0, 255));
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));
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()));
74 cv::cvtColor(mat, mat, cv::COLOR_RGB2GRAY);
77 cv::Mat mat2 = mat * 0.5;
78 size_t n = oversampling * 2;
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));
91 cv::resize(mat, mat, cv::Size(), 1.0 / oversampling, 1.0 / oversampling,
96 static std::shared_ptr<Glyph> instance(
const QFont &font,
int font_size,
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>>
103 if (cache.size() > 1000) {
104 LOG_DEBUG(
"clearing glyph cache");
107 auto &v = cache[key];
109 LOG_DEBUG(
"rendering glyph " << glyph);
110 v = std::make_shared<Glyph>(font, font_size, glyph);
117 std::shared_ptr<Texture> _texture;
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;
125 static double pack(std::vector<QRect *> rr,
int width) {
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();
141 if (row_rects == 0) {
146 for (
int j = row_start; j < row_start + row_rects; j++) {
148 rect->translate(x - rect->x(), y - rect->y());
150 y2 = std::max(y2, y + rect->height());
153 row_start += row_rects;
158 static void pack(std::vector<QRect> &rects) {
159 std::vector<QRect *> rr;
160 for (
auto &rect : rects) {
163 std::sort(rr.begin(), rr.end(),
164 [&](QRect *a, QRect *b) {
return a->height() < b->height(); });
166 for (
auto &r : rects) {
167 width = std::max(width, r.width() + 1);
169 for (
int w = width; w < 1024 * 32; w *= 2) {
170 if (pack(rr, w) < pack(rr, width)) {
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;
186 for (
auto &p : instance->_glyph_indices) {
187 merged_glyph_indices.insert(p.first);
189 if (merged_glyph_indices.size() > 200) {
190 LOG_DEBUG(
"clearing glyph set");
191 merged_glyph_indices = new_glyph_indices;
195 if (instance->_font != font || instance->_font_size != font_size) {
200 for (
auto &i : new_glyph_indices) {
201 if (instance->_glyph_indices.find(i) ==
202 instance->_glyph_indices.end()) {
209 LOG_DEBUG(
"building font texture for " << merged_glyph_indices.size()
211 instance = std::make_shared<Font>(font, font_size, merged_glyph_indices);
213 LOG_DEBUG(
"re-using font texture");
218 Font(
const QFont &font,
int font_size,
219 const std::unordered_set<uint32_t> &glyph_indices) {
222 _font_size = font_size;
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);
231 std::vector<QRect> pack_rects;
233 for (
auto &glyph : glyphs) {
234 pack_rects.emplace_back(0, 0, glyph->image().cols, glyph->image().rows);
241 for (
auto &rect : pack_rects) {
242 width = std::max(width, rect.right() + 1);
243 height = std::max(height, rect.bottom() + 1);
245 LOG_DEBUG(
"font texture size " << width <<
" " << height);
246 cv::Mat image(height, width, CV_8UC1);
247 image.setTo(cv::Scalar(0));
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());
264 const std::shared_ptr<Texture> &texture() {
266 _texture = std::make_shared<Texture>(TextureType::Linear);
267 _texture->mipmap(
false);
268 _texture->update(_image);
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];
278 throw std::runtime_error(
"glyph error");
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];
287 throw std::runtime_error(
"glyph error");
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];
296 throw std::runtime_error(
"glyph error");
301 TextRenderer::TextRenderer(
const std::shared_ptr<Material> &material) {
303 _material = material;
306 _material = std::make_shared<Material>();
308 _material_renderer = std::make_shared<MaterialRenderer>(_material);
312 _material_renderer->renderSync(context);
313 _parent_pose = context.pose;
314 SceneNode::renderSync(context);
319 _material_renderer->renderAsync(context);
321 if (_watcher.changed(_view_facing, _text, _size, _offset)) {
334 font.setPixelSize(font_size);
336 QTextLayout text_layout;
337 text_layout.setFont(font);
338 text_layout.setText(text().c_str());
339 text_layout.beginLayout();
341 auto line = text_layout.createLine();
342 if (line.isValid()) {
343 line.setLineWidth(1000000000);
348 text_layout.endLayout();
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);
357 auto texture_font = Font::instance(font, font_size, glyph_indices);
359 _texture = texture_font->texture();
363 for (
auto &glyph_run : text_layout.glyphRuns()) {
364 for (
size_t i = 0; i < glyph_run.positions().size(); i++) {
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);
374 mesh_rect = QRectF(pos.x(), mesh_rect.y(), mesh_rect.width(),
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);
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());
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(),
398 mesh.scale(_size * 1.0 / font_size);
400 if (mesh.positions.size()) {
401 float xmin = mesh.positions.front().x();
402 float xmax = mesh.positions.front().x();
404 float ymin = mesh.positions.front().y();
405 float ymax = mesh.positions.front().y();
407 for (
auto &p : mesh.positions) {
408 xmin = std::min(xmin, p.x());
409 ymin = std::min(ymin, p.y());
411 xmax = std::max(xmax, p.x());
412 ymax = std::max(ymax, p.y());
415 for (
auto &p : mesh.positions) {
416 p.x() -= (xmin + xmax) * 0.5;
420 mesh.translate(Eigen::Vector3f(_offset.x(), _offset.y(), 0.0));
423 for (
auto &p : mesh.positions) {
424 mesh.extras.emplace_back(p.x(), p.y(), 3, 0);
425 p = Eigen::Vector3f(0, 0, 0);
429 _mesh = std::make_shared<Mesh>(mesh);
436 material.color_texture = _texture->id();
439 context.render_list->push(material);
443 options.double_sided =
true;
445 context.render_list->push(_mesh, options);
448 instance.setPose(_parent_pose.matrix().cast<
float>());
450 context.render_list->push(instance);
453 SceneNode::renderAsync(context);
456 bool TextRenderer::pick(uint32_t
id)
const {
457 return _material_renderer->id() == id;
460 TextDisplay::TextDisplay() {
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);