TAMSVIZ
Visualization and annotation tool for ROS
scene.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "scene.h"
5 
6 #include "../core/interaction.h"
7 #include "../core/log.h"
8 #include "../core/workspace.h"
9 #include "../displays/interactive.h"
10 #include "../displays/shapes.h"
11 #include "../render/mesh.h"
12 
13 #include <QColor>
14 
15 SceneAnnotationBase::SceneAnnotationBase() {}
16 
17 bool SceneAnnotationBase::selected() {
18  return LockScope()->selection().contains(shared_from_this());
19 }
20 
21 std::shared_ptr<SceneNode> SceneAnnotationBase::node() {
22  if (_reset) {
23  _reset = false;
24  _node = nullptr;
25  }
26  if (!_node) {
27  _node = std::make_shared<SceneNode>();
28  update(_node);
29  bool s = selected();
30  std::weak_ptr<Object> me = shared_from_this();
31  LockScope()->modified.connect(_node, [this, me, s]() {
32  if (auto me__ = me.lock()) {
33  bool v = selected();
34  if (v != s) {
35  LOG_DEBUG("selection " << int(s) << " " << int(v));
36  _reset = true;
37  GlobalEvents::instance()->redraw();
38  }
39  }
40  });
41  }
42  return _node;
43 }
44 
45 void SceneAnnotationBase::renderSync(
46  const RenderSyncContext &context, const std::shared_ptr<TrackBase> &track,
47  const std::shared_ptr<AnnotationSpan> &span) {
48  if (!_context) {
49  _context = std::make_shared<SceneContext>();
50  }
51  _context->nodes.clear();
52  node()->renderSyncRecursive(context, *_context);
53 }
54 
55 void SceneAnnotationBase::renderAsync(const RenderAsyncContext &context) {
56  // LOG_DEBUG("render");
57  if (_context) {
58  for (auto &node : _context->nodes) {
59  // LOG_DEBUG("async");
60  node->renderAsync(context);
61  }
62  }
63 }
64 
65 bool SceneAnnotationBase::pick(uint32_t id) const {
66  if (_node) {
67  return _node->pick(id);
68  } else {
69  return false;
70  }
71 }
72 
73 bool SceneAnnotationBase::interact(const Interaction &interaction) {
74  if (_node) {
75  bool rs = _node->interact(interaction);
76  // LOG_DEBUG("SceneAnnotationBase::interact " << (int)rs);
77  return rs;
78  } else {
79  return false;
80  }
81 }
82 
83 /*
84 void SceneAnnotationBase::clear() {
85  LOG_ERROR("clear scene annotation");
86  //std::lock_guard<std::mutex> lock(_mutex);
87  //_node.reset();
88  //_context.reset();
89 }
90 */
91 
93 private:
94  std::shared_ptr<InteractiveMarkerArray> _markers;
95 
96 protected:
97  std::shared_ptr<SceneNode> _visual;
98  std::shared_ptr<Material> _material;
100 
101 public:
102  PROPERTY(double, scale, 1.0, min = 0.0);
103  // PROPERTY(Pose, transform);
104  const std::shared_ptr<InteractiveMarkerArray> &
105  markers(const std::shared_ptr<SceneNode> &node) {
106  if (!_markers) {
107  _markers = node->create<InteractiveMarkerArray>(
108  std::make_shared<InteractiveMarkerParameters>());
109  }
110  return _markers;
111  }
112  virtual void update(const std::shared_ptr<SceneNode> &node) override {
113  SceneAnnotationBase::update(node);
114  _markers = nullptr;
115  }
116  virtual Eigen::Isometry3d virtualPose() const = 0;
117  virtual void virtualPose(const Eigen::Isometry3d &pose) = 0;
118  virtual void
119  renderSync(const RenderSyncContext &context,
120  const std::shared_ptr<TrackBase> &track,
121  const std::shared_ptr<AnnotationSpan> &span) override {
122  node();
123  auto p = virtualPose();
124  p.linear() *= scale();
125  if (_visual) {
126  _visual->pose(p);
127  }
128  if (_material) {
129  auto color = QColor::fromHsvF(track->color(), 0.8, 0.8);
130  _material->color() = Color3(color.redF(), color.greenF(), color.blueF());
131  }
132  if (_markers) {
133  if (auto m = _markers->marker("")) {
134  m->pose(p);
135  }
136  }
137  SceneAnnotationBase::renderSync(context, track, span);
138  }
139  virtual bool interact(const Interaction &interaction) override {
140  if (SceneAnnotationBase::interact(interaction)) {
141  if (_markers) {
142  if (auto m = _markers->marker("")) {
143  if (interaction.finished) {
144  ActionScope ws("Move handle");
145  virtualPose(Eigen::Isometry3d(m->pose().matrix()));
146  ws->modified();
147  } else {
148  LockScope ws;
149  virtualPose(Eigen::Isometry3d(m->pose().matrix()));
150  ws->modified();
151  }
152  }
153  }
154  return true;
155  } else {
156  return false;
157  }
158  }
159 };
161 
163  PROPERTY(Eigen::Vector3d, position);
164  virtual Eigen::Isometry3d virtualPose() const override {
165  Eigen::Isometry3d pose = Eigen::Isometry3d::Identity();
166  pose.translation() = position();
167  return pose;
168  }
169  virtual void virtualPose(const Eigen::Isometry3d &pose) override {
170  position() = pose.translation();
171  }
172  // transform().toIsometry3d()
173  // transform().fromIsometry3d
174  virtual void update(const std::shared_ptr<SceneNode> &node) override {
175  InteractiveSceneAnnotationBase::update(node);
176  if (!_material) {
177  _material = std::make_shared<Material>();
178  }
179  // if (!selected())
180  {
181  static auto mesh = std::make_shared<Mesh>(makeSphere().scale(0.22));
182  _visual = node->create<MeshRenderer>(
183  mesh, _material /*, [this](const Interaction &interaction) {
184  position() +=
185  (interaction.current.point - interaction.previous.point);
186  }*/);
187  }
188  if (selected()) {
189  visualization_msgs::InteractiveMarker marker;
190  {
191  visualization_msgs::InteractiveMarkerControl control;
192  control.interaction_mode =
193  visualization_msgs::InteractiveMarkerControl::MOVE_3D;
194  control.always_visible = true;
195  {
196  visualization_msgs::Marker marker;
197  marker.type = visualization_msgs::Marker::SPHERE;
198  marker.scale.x = 0.45;
199  marker.scale.y = 0.45;
200  marker.scale.z = 0.45;
201  marker.color.r = 0.5;
202  marker.color.g = 0.5;
203  marker.color.b = 0.5;
204  marker.color.a = 0.2;
205  control.markers.push_back(marker);
206  }
207  marker.controls.push_back(control);
208  }
209  {
210  visualization_msgs::InteractiveMarkerControl control;
211  control.orientation.w = 1;
212  control.orientation.x = 1;
213  control.orientation.y = 0;
214  control.orientation.z = 0;
215  control.name = "move_x";
216  control.interaction_mode =
217  visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
218  marker.controls.push_back(control);
219  }
220  {
221  visualization_msgs::InteractiveMarkerControl control;
222  control.orientation.w = 1;
223  control.orientation.x = 0;
224  control.orientation.y = 1;
225  control.orientation.z = 0;
226  control.name = "move_z";
227  control.interaction_mode =
228  visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
229  marker.controls.push_back(control);
230  }
231  {
232  visualization_msgs::InteractiveMarkerControl control;
233  control.orientation.w = 1;
234  control.orientation.x = 0;
235  control.orientation.y = 0;
236  control.orientation.z = 1;
237  control.name = "move_y";
238  control.interaction_mode =
239  visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
240  marker.controls.push_back(control);
241  }
242  visualization_msgs::InteractiveMarkerInit init;
243  init.markers.push_back(marker);
244  markers(node)->init(init);
245  }
246  }
247 };