TAMSVIZ
Visualization and annotation tool for ROS
plotwindow.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "plotwindow.h"
5 
6 #include "../core/bagplayer.h"
7 #include "../core/log.h"
8 #include "../core/topic.h"
9 #include "../core/workspace.h"
10 
11 #include <QGraphicsScene>
12 #include <QGraphicsView>
13 
14 PlotWindow::PlotWindow() {
15 
16  LockScope()->modified.connect(this, [this]() {
17  LockScope ws;
18  auto plot_display = plot().resolve(ws.ws());
19  if (!plot_display) {
20  _renderer.reset();
21  } else {
22  if (!_renderer || plot_display != _renderer->plotDisplay()) {
23  LOG_DEBUG("creating plot renderer");
24  _renderer = std::make_shared<PlotRenderer>(plot_display);
25  }
26  }
27  GlobalEvents::instance()->redraw();
28  });
29 
30  {
31  auto *button = new FlatButton();
32  button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
33  QMenu *menu = new QMenu(this);
34  button->setMenu(menu);
35  button->setText("Select plot...");
36  addToolWidget(button);
37  connect(menu, &QMenu::aboutToShow, this, [this, menu]() {
38  menu->clear();
39  LockScope()->document()->display()->recurse(
40  [&](const std::shared_ptr<Display> &display) {
41  if (auto plot = std::dynamic_pointer_cast<PlotDisplay>(display)) {
42  connect(menu->addAction(plot->name().c_str()),
43  &QAction::triggered, this, [this, plot](bool) {
44  ActionScope ws("Plot");
45  this->plot() = plot;
46  LockScope()->selection() = plot;
47  ws->modified();
48  });
49  }
50  });
51  connect(menu->addAction("Create new plot"), &QAction::triggered, this,
52  [this](bool) {
53  ActionScope ws("Plot");
54  auto plot = std::make_shared<PlotDisplay>();
55  static size_t counter = 1;
56  plot->name() = "Plot " + std::to_string(counter++);
57  LockScope()->document()->display()->displays().push_back(plot);
58  LockScope()->selection() = plot;
59  this->plot() = plot;
60  ws->modified();
61  });
62  });
63  LockScope ws;
64  ws->modified.connect(button, [this, button]() {
65  LockScope ws;
66  if (auto plot = this->plot().resolve(ws.ws())) {
67  button->setText(plot->name().c_str());
68  } else {
69  button->setText("Select plot...");
70  }
71  });
72  }
73 }
74 
75 void PlotWindow::renderWindowSync(const RenderWindowSyncContext &context) {
76  _renderer_async = _renderer;
77  if (_renderer) {
78  _renderer->renderSync();
79  }
80  if (!LockScope()->player) {
81  GlobalEvents::instance()->redraw();
82  }
83 }
84 
85 void PlotWindow::renderWindowAsync(const RenderWindowAsyncContext &context) {
86 
87  auto *previous_context = QOpenGLContext::currentContext();
88  auto *previous_surface = previous_context->surface();
89 
90  if (!_offscreen_context) {
91  _offscreen_context = std::make_shared<QOpenGLContext>();
92  _offscreen_context->setShareContext(QOpenGLContext::globalShareContext());
93  if (!_offscreen_context->create()) {
94  throw std::runtime_error("failed to create opengl context");
95  }
96  }
97 
98  if (!_offscreen_surface) {
99  _offscreen_surface = std::make_shared<QOffscreenSurface>();
100  _offscreen_surface->create();
101  if (!_offscreen_surface->isValid()) {
102  throw std::runtime_error("failed to create offscreen surface");
103  }
104  }
105 
106  _offscreen_context->makeCurrent(_offscreen_surface.get());
107 
108  if (!_fbo_render || _fbo_render->width() != _width ||
109  _fbo_render->height() != _height) {
110  QOpenGLFramebufferObjectFormat format;
111  // format.setSamples(0);
112  format.setSamples(16);
113  format.setAttachment(QOpenGLFramebufferObject::CombinedDepthStencil);
114  _fbo_render =
115  std::make_shared<QOpenGLFramebufferObject>(_width, _height, format);
116  }
117 
118  if (!_paint_device) {
119  _paint_device =
120  std::make_shared<QOpenGLPaintDevice>(QSize(_width, _height));
121  }
122  _paint_device->setSize(QSize(_width, _height));
123 
124  if (!_fbo_render->bind()) {
125  throw std::runtime_error("failed to bind fbo");
126  }
127 
128  {
129  QPainter painter(_paint_device.get());
130  painter.endNativePainting();
131 
132  painter.setRenderHint(QPainter::Antialiasing);
133 
134  if (_renderer_async) {
135  _renderer_async->renderAsync(&painter);
136  }
137 
138  painter.beginNativePainting();
139  }
140 
141  _fbo_render->release();
142 
143  {
144  std::unique_lock<std::mutex> lock(_fbo_mutex);
145 
146  _fbo_buffer.update(_width, _height, GL_RGB8);
147 
148  _fbo_present.bind();
149  _fbo_present.attach(_fbo_buffer, GL_COLOR_ATTACHMENT0);
150 
151  V_GL(glBindFramebuffer(GL_READ_FRAMEBUFFER, _fbo_render->handle()));
152  V_GL(glBindFramebuffer(GL_DRAW_FRAMEBUFFER, _fbo_present.id()));
153  V_GL(glBlitFramebuffer(0, 0, _width, _height, 0, 0, _width, _height,
154  GL_COLOR_BUFFER_BIT, GL_NEAREST));
155 
156  _fbo_present.bind();
157  V_GL(glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0,
158  GL_RENDERBUFFER, 0));
159 
160  V_GL(glBindFramebuffer(GL_FRAMEBUFFER, 0));
161 
162  _fbo_initialized = true;
163  }
164 
165  previous_context->makeCurrent(previous_surface);
166 }
167 
168 void PlotWindow::paintHUD(QPainter *p) {}
169 
170 void PlotWindow::composite(int target) {
171  std::unique_lock<std::mutex> lock(_fbo_mutex);
172 
173  V_GL(glViewport(0, 0, _width, _height));
174  V_GL(glClearColor(0, 0, 0, 1));
175  V_GL(glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT));
176 
177  if (_fbo_initialized) {
178  _fbo_composite.bind();
179  _fbo_composite.attach(_fbo_buffer, GL_COLOR_ATTACHMENT0);
180 
181  V_GL(glBindFramebuffer(GL_READ_FRAMEBUFFER, _fbo_composite.id()));
182  V_GL(glBindFramebuffer(GL_DRAW_FRAMEBUFFER, target));
183  V_GL(glBlitFramebuffer(0, 0, _width, _height, 0, 0, _width, _height,
184  GL_COLOR_BUFFER_BIT, GL_NEAREST));
185 
186  _fbo_composite.bind();
187  V_GL(glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0,
188  GL_RENDERBUFFER, 0));
189 
190  V_GL(glBindFramebuffer(GL_FRAMEBUFFER, 0));
191  }
192 }
193 
194 void PlotWindow::handleEvent(QEvent *event) {
195  switch (event->type()) {
196  case QEvent::MouseButtonPress:
197  if (_renderer) {
198  ActionScope ws("Pick");
199  ws->selection() = _renderer->plotDisplay();
200  }
201  break;
202  }
203 }