4 #include "interactive.h" 6 #include "../core/interaction.h" 7 #include "../core/profiler.h" 8 #include "../core/workspace.h" 10 #include <eigen_conversions/eigen_msg.h> 12 #include <interactive_markers/tools.h> 14 InteractiveMarkerControl::InteractiveMarkerControl(
15 const visualization_msgs::InteractiveMarkerControl &message,
16 const std::shared_ptr<InteractiveMarkerParameters> ¶ms,
18 : _parent(parent), _params(params) {
22 void InteractiveMarkerControl::update(
23 const visualization_msgs::InteractiveMarkerControl &message) {
24 std::lock_guard<std::mutex> lock(_mutex);
26 tf::quaternionMsgToEigen(message.orientation, _orientation);
27 if (_orientation.norm() == 0) {
28 _orientation = Eigen::Quaterniond::Identity();
30 _orientation.normalize();
33 _interaction_mode = message.interaction_mode;
34 _markers.resize(message.markers.size());
35 for (
size_t i = 0; i < message.markers.size(); i++) {
36 if (!_markers.at(i)) {
37 _markers.at(i) = create<VisualizationMarker>(message.markers[i]);
39 _markers.at(i)->update(message.markers[i]);
44 bool InteractiveMarkerControl::interact(
const Interaction &interaction) {
45 std::lock_guard<std::mutex> lock(_mutex);
47 if (pick(interaction.id)) {
48 LOG_DEBUG(
"interactive marker interaction mode " << _interaction_mode);
49 if (_interaction_mode != 0) {
50 Eigen::Affine3d p = _parent->pose();
51 Eigen::Affine3d control_pose =
52 _parent->renderPose() * Eigen::AngleAxisd(_orientation);
53 Eigen::Matrix3d control_orientation = control_pose.linear();
54 switch (_interaction_mode) {
55 case visualization_msgs::InteractiveMarkerControl::MOVE_3D: {
57 _parent->framePose().linear().inverse() *
58 (interaction.current.point - interaction.previous.point);
61 case visualization_msgs::InteractiveMarkerControl::MOVE_PLANE: {
62 p.translation() += _parent->framePose().linear().inverse() *
63 interaction.projectPlane(control_orientation *
64 Eigen::Vector3d::UnitX());
67 case visualization_msgs::InteractiveMarkerControl::MOVE_AXIS: {
68 p.translation() += _parent->framePose().linear().inverse() *
69 interaction.projectToAxis(control_orientation *
70 Eigen::Vector3d::UnitX());
73 case visualization_msgs::InteractiveMarkerControl::ROTATE_AXIS:
74 case visualization_msgs::InteractiveMarkerControl::MOVE_ROTATE: {
75 Eigen::Vector3d a = control_pose.inverse() *
76 interaction.previous.projectPlane(
77 control_pose.translation(),
78 control_orientation * Eigen::Vector3d::UnitX());
79 Eigen::Vector3d b = control_pose.inverse() *
80 interaction.current.projectPlane(
81 control_pose.translation(),
82 control_orientation * Eigen::Vector3d::UnitX());
83 Eigen::Matrix3d rotation =
85 std::atan2(a.y(), a.z()) - std::atan2(b.y(), b.z()),
86 Eigen::AngleAxisd(_orientation) * Eigen::Vector3d::UnitX())
88 p.linear() = (p.linear() * rotation).eval();
89 if (_interaction_mode ==
90 visualization_msgs::InteractiveMarkerControl::MOVE_ROTATE) {
92 _parent->framePose().linear().inverse() *
93 (control_orientation * (b.normalized() * (b.norm() - a.norm())));
97 case visualization_msgs::InteractiveMarkerControl::ROTATE_3D:
98 case visualization_msgs::InteractiveMarkerControl::MOVE_ROTATE_3D: {
100 _parent->renderPose().inverse() * interaction.previous.point;
102 _parent->renderPose().inverse() * interaction.current.point;
103 Eigen::Quaterniond quat = Eigen::Quaterniond::Identity();
104 quat.setFromTwoVectors(a, b);
105 p *= Eigen::AngleAxisd(quat);
106 if (_interaction_mode ==
107 visualization_msgs::InteractiveMarkerControl::MOVE_ROTATE_3D) {
108 p.translation() += _parent->framePose().linear().inverse() *
109 (_parent->renderPose().linear() *
110 (b.normalized() * (b.norm() - a.norm())));
117 visualization_msgs::InteractiveMarkerFeedback feedback;
118 feedback.marker_name = parentInteractiveMarker()->markerName();
119 feedback.control_name = controlName();
120 if (interaction.pressed) {
121 feedback.event_type =
122 visualization_msgs::InteractiveMarkerFeedback::MOUSE_DOWN;
123 }
else if (interaction.finished) {
124 feedback.event_type =
125 visualization_msgs::InteractiveMarkerFeedback::MOUSE_UP;
127 feedback.event_type =
128 visualization_msgs::InteractiveMarkerFeedback::POSE_UPDATE;
130 feedback.header.frame_id = _parent->frame().name();
131 feedback.client_id = ros::this_node::getNamespace();
132 tf::pointEigenToMsg(_parent->framePose().inverse() *
133 interaction.current.point,
134 feedback.mouse_point);
135 feedback.mouse_point_valid =
true;
136 tf::poseEigenToMsg(_parent->pose(), feedback.pose);
137 parentInteractiveMarker()->parentInteractiveMarkerArray()->feedback(
146 InteractiveMarker::InteractiveMarker(
147 const visualization_msgs::InteractiveMarker &message,
148 const std::shared_ptr<InteractiveMarkerParameters> ¶ms,
150 : _parent(parent), _params(params) {
151 _text = create<TextRenderer>(_params->description_material);
155 void InteractiveMarker::update(
const visualization_msgs::InteractiveMarker &m) {
158 interactive_markers::autoComplete(marker,
false);
159 std::lock_guard<std::mutex> lock(_mutex);
164 _text->text(marker.description);
166 Eigen::Affine3d p = Eigen::Affine3d::Identity();
167 tf::poseMsgToEigen(marker.pose, p);
171 frame(marker.header.frame_id);
172 _controls.resize(marker.controls.size());
173 for (
size_t i = 0; i < marker.controls.size(); i++) {
174 auto &m = _controls.at(i);
177 m = create<InteractiveMarkerControl>(marker.controls.at(i), _params,
180 m->update(marker.controls.at(i));
186 SceneNode::renderSync(context);
187 _text->size(_scale * _params->description_size);
188 _text->offset(_params->description_offset * _scale);
189 _text->visible(_params->show_descriptions && _params->description_size > 0);
192 bool InteractiveMarker::interact(
const Interaction &interaction) {
193 std::lock_guard<std::mutex> lock(_mutex);
195 bool x = SceneNode::interact(interaction);
197 _dragged = !interaction.finished;
202 void InteractiveMarker::update(
203 const visualization_msgs::InteractiveMarkerPose &message) {
204 std::lock_guard<std::mutex> lock(_mutex);
209 frame(message.header.frame_id);
210 Eigen::Affine3d p = Eigen::Affine3d::Identity();
211 tf::poseMsgToEigen(message.pose, p);
215 InteractiveMarkerArray::InteractiveMarkerArray(
216 const std::shared_ptr<InteractiveMarkerParameters> ¶ms)
220 SceneNode::renderSync(context);
223 void InteractiveMarkerArray::init(
224 const visualization_msgs::InteractiveMarkerInit &init) {
225 std::lock_guard<std::mutex> lock(_mutex);
227 auto previous_markers = _markers;
229 for (
auto &marker : init.markers) {
230 auto &m = _markers[marker.name];
231 m = previous_markers[marker.name];
233 m = create<InteractiveMarker>(marker, _params,
this);
240 void InteractiveMarkerArray::update(
241 const visualization_msgs::InteractiveMarkerUpdate &update) {
242 std::lock_guard<std::mutex> lock(_mutex);
244 for (
auto &erase : update.erases) {
245 _markers.erase(erase);
247 for (
auto &pose : update.poses) {
248 auto iter = _markers.find(pose.name);
249 if (iter != _markers.end()) {
250 iter->second->update(pose);
253 for (
auto &marker : update.markers) {
254 auto &m = _markers[marker.name];
256 m = create<InteractiveMarker>(marker, _params,
this);
263 std::shared_ptr<InteractiveMarker>
264 InteractiveMarkerArray::marker(
const std::string &name) {
265 auto iter = _markers.find(name);
266 if (iter != _markers.end()) {
273 InteractiveMarkerDisplayBase::InteractiveMarkerDisplayBase() {
274 _params = std::make_shared<InteractiveMarkerParameters>();
275 _params->description_material = std::make_shared<Material>();
279 void InteractiveMarkerDisplayBase::renderSync(
281 MeshDisplayBase::renderSync(context);
284 InteractiveMarkerDisplay::InteractiveMarkerDisplay() {
285 _publisher = std::make_shared<
287 auto *pub_ptr = _publisher.get();
288 _markers->feedback.connect(
290 [pub_ptr](
const visualization_msgs::InteractiveMarkerFeedback &feedback) {
291 pub_ptr->publish(feedback);
296 _params->show_descriptions = showDescriptions();
297 _params->description_size = descriptionSize();
298 _params->description_offset = descriptionOffset();
299 _params->description_material->color(descriptionColor());
300 _params->description_material->opacity(descriptionOpacity());
301 if (_watcher.changed(topicNamespace())) {
302 auto markers = _markers;
303 _update_subscriber = std::make_shared<
305 topicNamespace() +
"/update", markers,
306 [markers](
const std::shared_ptr<
307 const visualization_msgs::InteractiveMarkerUpdate> &update) {
310 markers->update(*update);
314 std::make_shared<Subscriber<visualization_msgs::InteractiveMarkerInit>>(
315 topicNamespace() +
"/update_full", markers,
316 [markers](
const std::shared_ptr<
317 const visualization_msgs::InteractiveMarkerInit> &init) {
320 markers->init(*init);
323 _publisher->topic(topicNamespace() +
"/feedback");
325 InteractiveMarkerDisplayBase::renderSync(context);
328 std::vector<std::string> InteractiveMarkerDisplay::listTopicNamespaces() {
329 std::string postfix =
"/update";
330 std::vector<std::string> all = TopicManager::instance()->listTopics(
331 ros::message_traits::DataType<
332 visualization_msgs::InteractiveMarkerUpdate>::value());
333 std::vector<std::string> ret;
334 for (
auto &n : all) {
335 if (n.size() > postfix.size()) {
336 if (n.substr(n.size() - postfix.size()) == postfix) {
337 ret.push_back(n.substr(0, n.size() - postfix.size()));
344 InteractivePoseDisplayBase::InteractivePoseDisplayBase() {}
346 void InteractivePoseDisplayBase::refresh() {
347 InteractiveMarkerDisplayBase::refresh();
348 if (!_publish_thread_data) {
349 auto publish_thread_data = _publish_thread_data =
350 std::make_shared<PublishThreadData>();
351 std::weak_ptr<InteractivePoseDisplayBase> _this =
354 std::thread([_this, publish_thread_data]() {
355 LOG_DEBUG(
"start publisher thread");
356 auto t = std::chrono::steady_clock::now();
358 t += std::chrono::seconds(1);
360 std::unique_lock<std::mutex> lock(
361 publish_thread_data->_publish_mutex);
363 if (publish_thread_data->_stop_flag) {
364 LOG_DEBUG(
"exit publisher thread");
367 if (std::chrono::steady_clock::now() >= t) {
370 publish_thread_data->_stop_condition.wait_until(lock, t);
373 if (
auto me = _this.lock()) {
378 LOG_DEBUG(
"exit publisher thread");
384 InteractivePoseDisplayBase::~InteractivePoseDisplayBase() {
385 if (_publish_thread_data) {
387 std::unique_lock<std::mutex> lock(_publish_thread_data->_publish_mutex);
388 _publish_thread_data->_stop_flag =
true;
389 _publish_thread_data->_stop_condition.notify_all();
391 _publish_thread_data =
nullptr;
395 void InteractivePoseDisplayBase::publish() {
396 if (
auto m = _markers->marker(
"")) {
397 auto pose = m->pose();
398 pose.linear().col(0).normalize();
399 pose.linear().col(1).normalize();
400 pose.linear().col(2).normalize();
401 publish(frame().empty()
402 ?
LockScope()->document()->display()->transformer->root()
404 Eigen::Isometry3d(pose.matrix()));
409 if (
auto m = _markers->marker(
"")) {
412 auto p = transform().toIsometry3d();
413 p.linear() *= scale();
417 InteractiveMarkerDisplayBase::renderSync(context);
421 bool InteractivePoseDisplayBase::interact(
const Interaction &interaction) {
422 if (InteractiveMarkerDisplayBase::interact(interaction)) {
423 if (
auto m = _markers->marker(
"")) {
424 if (interaction.finished) {
426 transform().fromIsometry3d(Eigen::Isometry3d(m->pose().matrix()));
430 transform().fromIsometry3d(Eigen::Isometry3d(m->pose().matrix()));
440 visualization_msgs::InteractiveMarker
441 InteractiveMarkerDisplayBase::makePoseMarker() {
442 visualization_msgs::InteractiveMarker marker;
444 visualization_msgs::InteractiveMarkerControl control;
445 control.interaction_mode =
446 visualization_msgs::InteractiveMarkerControl::MOVE_3D;
447 control.always_visible =
true;
449 visualization_msgs::Marker marker;
450 marker.type = visualization_msgs::Marker::CUBE;
451 marker.scale.x = 0.45;
452 marker.scale.y = 0.45;
453 marker.scale.z = 0.45;
454 marker.color.r = 0.5;
455 marker.color.g = 0.5;
456 marker.color.b = 0.5;
457 marker.color.a = 1.0;
458 control.markers.push_back(marker);
460 marker.controls.push_back(control);
463 visualization_msgs::InteractiveMarkerControl control;
464 control.orientation.w = 1;
465 control.orientation.x = 1;
466 control.orientation.y = 0;
467 control.orientation.z = 0;
468 control.name =
"rotate_x";
469 control.interaction_mode =
470 visualization_msgs::InteractiveMarkerControl::ROTATE_AXIS;
471 marker.controls.push_back(control);
472 control.name =
"move_x";
473 control.interaction_mode =
474 visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
475 marker.controls.push_back(control);
478 visualization_msgs::InteractiveMarkerControl control;
479 control.orientation.w = 1;
480 control.orientation.x = 0;
481 control.orientation.y = 0;
482 control.orientation.z = 1;
483 control.name =
"rotate_y";
484 control.interaction_mode =
485 visualization_msgs::InteractiveMarkerControl::ROTATE_AXIS;
486 marker.controls.push_back(control);
487 control.name =
"move_y";
488 control.interaction_mode =
489 visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
490 marker.controls.push_back(control);
493 visualization_msgs::InteractiveMarkerControl control;
494 control.orientation.w = 1;
495 control.orientation.x = 0;
496 control.orientation.y = 1;
497 control.orientation.z = 0;
498 control.name =
"rotate_z";
499 control.interaction_mode =
500 visualization_msgs::InteractiveMarkerControl::ROTATE_AXIS;
501 marker.controls.push_back(control);
502 control.name =
"move_z";
503 control.interaction_mode =
504 visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
505 marker.controls.push_back(control);
510 visualization_msgs::InteractiveMarker
511 InteractiveMarkerDisplayBase::makeRotationMarker() {
512 visualization_msgs::InteractiveMarker marker;
514 visualization_msgs::InteractiveMarkerControl control;
515 control.interaction_mode =
516 visualization_msgs::InteractiveMarkerControl::ROTATE_3D;
517 control.always_visible =
true;
519 visualization_msgs::Marker marker;
520 marker.type = visualization_msgs::Marker::CUBE;
521 marker.scale.x = 0.45;
522 marker.scale.y = 0.45;
523 marker.scale.z = 0.45;
524 marker.color.r = 0.5;
525 marker.color.g = 0.5;
526 marker.color.b = 0.5;
527 marker.color.a = 1.0;
528 control.markers.push_back(marker);
530 marker.controls.push_back(control);
533 visualization_msgs::InteractiveMarkerControl control;
534 control.orientation.w = 1;
535 control.orientation.x = 1;
536 control.orientation.y = 0;
537 control.orientation.z = 0;
538 control.name =
"rotate_x";
539 control.interaction_mode =
540 visualization_msgs::InteractiveMarkerControl::ROTATE_AXIS;
541 marker.controls.push_back(control);
544 visualization_msgs::InteractiveMarkerControl control;
545 control.orientation.w = 1;
546 control.orientation.x = 0;
547 control.orientation.y = 0;
548 control.orientation.z = 1;
549 control.name =
"rotate_y";
550 control.interaction_mode =
551 visualization_msgs::InteractiveMarkerControl::ROTATE_AXIS;
552 marker.controls.push_back(control);
555 visualization_msgs::InteractiveMarkerControl control;
556 control.orientation.w = 1;
557 control.orientation.x = 0;
558 control.orientation.y = 1;
559 control.orientation.z = 0;
560 control.name =
"rotate_z";
561 control.interaction_mode =
562 visualization_msgs::InteractiveMarkerControl::ROTATE_AXIS;
563 marker.controls.push_back(control);
568 visualization_msgs::InteractiveMarker
569 InteractiveMarkerDisplayBase::makePointMarker() {
570 visualization_msgs::InteractiveMarker marker;
572 visualization_msgs::InteractiveMarkerControl control;
573 control.interaction_mode =
574 visualization_msgs::InteractiveMarkerControl::MOVE_3D;
575 control.always_visible =
true;
577 visualization_msgs::Marker marker;
578 marker.type = visualization_msgs::Marker::CUBE;
579 marker.scale.x = 0.45;
580 marker.scale.y = 0.45;
581 marker.scale.z = 0.45;
582 marker.color.r = 0.5;
583 marker.color.g = 0.5;
584 marker.color.b = 0.5;
585 marker.color.a = 1.0;
586 control.markers.push_back(marker);
588 marker.controls.push_back(control);
591 visualization_msgs::InteractiveMarkerControl control;
592 control.orientation.w = 1;
593 control.orientation.x = 1;
594 control.orientation.y = 0;
595 control.orientation.z = 0;
596 control.name =
"move_x";
597 control.interaction_mode =
598 visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
599 marker.controls.push_back(control);
602 visualization_msgs::InteractiveMarkerControl control;
603 control.orientation.w = 1;
604 control.orientation.x = 0;
605 control.orientation.y = 0;
606 control.orientation.z = 1;
607 control.name =
"move_y";
608 control.interaction_mode =
609 visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
610 marker.controls.push_back(control);
613 visualization_msgs::InteractiveMarkerControl control;
614 control.orientation.w = 1;
615 control.orientation.x = 0;
616 control.orientation.y = 1;
617 control.orientation.z = 0;
618 control.name =
"move_z";
619 control.interaction_mode =
620 visualization_msgs::InteractiveMarkerControl::MOVE_AXIS;
621 marker.controls.push_back(control);
626 TransformPublisherDisplay::TransformPublisherDisplay() {
627 visualization_msgs::InteractiveMarkerInit init;
628 init.markers.push_back(makePoseMarker());
629 _markers->init(init);
632 void TransformPublisherDisplay::publish(
const std::string &frame,
633 const Eigen::Isometry3d &pose) {
634 if (_publish_watcher.changed(childFrame(), frame, poseToArray(pose),
636 tf2_msgs::TFMessage m;
637 m.transforms.emplace_back();
638 m.transforms.back().header.frame_id = frame;
639 m.transforms.back().header.stamp = ros::Time::now();
640 m.transforms.back().child_frame_id = childFrame();
641 tf::transformEigenToMsg(pose, m.transforms.back().transform);
642 _publisher.publish(m);