2016-08-03 18:45:08 +08:00
|
|
|
/*
|
|
|
|
* Copyright 2016 The Cartographer Authors
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2016-10-12 23:50:37 +08:00
|
|
|
#include "cartographer_rviz/submaps_display.h"
|
2016-08-03 18:45:08 +08:00
|
|
|
|
2016-10-12 23:50:37 +08:00
|
|
|
#include "OgreResourceGroupManager.h"
|
|
|
|
#include "cartographer/common/make_unique.h"
|
|
|
|
#include "cartographer/common/mutex.h"
|
|
|
|
#include "cartographer_ros_msgs/SubmapList.h"
|
|
|
|
#include "cartographer_ros_msgs/SubmapQuery.h"
|
|
|
|
#include "geometry_msgs/TransformStamped.h"
|
|
|
|
#include "pluginlib/class_list_macros.h"
|
|
|
|
#include "ros/package.h"
|
|
|
|
#include "ros/ros.h"
|
|
|
|
#include "rviz/display_context.h"
|
|
|
|
#include "rviz/frame_manager.h"
|
2017-04-21 17:58:16 +08:00
|
|
|
#include "rviz/properties/bool_property.h"
|
2016-10-12 23:50:37 +08:00
|
|
|
#include "rviz/properties/string_property.h"
|
2016-08-03 18:45:08 +08:00
|
|
|
|
2016-09-20 16:30:21 +08:00
|
|
|
namespace cartographer_rviz {
|
2016-08-03 18:45:08 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2016-08-05 20:23:30 +08:00
|
|
|
constexpr int kMaxOnGoingRequestsPerTrajectory = 6;
|
2016-08-03 18:45:08 +08:00
|
|
|
constexpr char kMaterialsDirectory[] = "/ogre_media/materials";
|
|
|
|
constexpr char kGlsl120Directory[] = "/glsl120";
|
|
|
|
constexpr char kScriptsDirectory[] = "/scripts";
|
|
|
|
constexpr char kDefaultMapFrame[] = "map";
|
|
|
|
constexpr char kDefaultTrackingFrame[] = "base_link";
|
2016-09-20 16:30:21 +08:00
|
|
|
constexpr char kDefaultSubmapQueryServiceName[] = "/submap_query";
|
2016-08-03 18:45:08 +08:00
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2016-08-10 18:31:19 +08:00
|
|
|
SubmapsDisplay::SubmapsDisplay() : tf_listener_(tf_buffer_) {
|
2016-08-03 18:45:08 +08:00
|
|
|
submap_query_service_property_ = new ::rviz::StringProperty(
|
2016-09-20 16:30:21 +08:00
|
|
|
"Submap query service", kDefaultSubmapQueryServiceName,
|
2016-08-04 23:59:03 +08:00
|
|
|
"Submap query service to connect to.", this, SLOT(Reset()));
|
2016-08-03 18:45:08 +08:00
|
|
|
map_frame_property_ = new ::rviz::StringProperty(
|
|
|
|
"Map frame", kDefaultMapFrame, "Map frame, used for fading out submaps.",
|
|
|
|
this);
|
|
|
|
tracking_frame_property_ = new ::rviz::StringProperty(
|
|
|
|
"Tracking frame", kDefaultTrackingFrame,
|
|
|
|
"Tracking frame, used for fading out submaps.", this);
|
2016-08-03 20:54:45 +08:00
|
|
|
client_ = update_nh_.serviceClient<::cartographer_ros_msgs::SubmapQuery>("");
|
2017-04-21 17:58:16 +08:00
|
|
|
submaps_category_ = new ::rviz::Property(
|
|
|
|
"Submaps", QVariant(), "List of all submaps, organized by trajectories.",
|
|
|
|
this);
|
|
|
|
visibility_all_enabled_ = new ::rviz::BoolProperty(
|
|
|
|
"All Enabled", true,
|
|
|
|
"Whether all the submaps should be displayed or not.", submaps_category_,
|
|
|
|
SLOT(AllEnabledToggled()), this);
|
2016-08-03 18:45:08 +08:00
|
|
|
const std::string package_path = ::ros::package::getPath(ROS_PACKAGE_NAME);
|
|
|
|
Ogre::ResourceGroupManager::getSingleton().addResourceLocation(
|
|
|
|
package_path + kMaterialsDirectory, "FileSystem", ROS_PACKAGE_NAME);
|
|
|
|
Ogre::ResourceGroupManager::getSingleton().addResourceLocation(
|
|
|
|
package_path + kMaterialsDirectory + kGlsl120Directory, "FileSystem",
|
|
|
|
ROS_PACKAGE_NAME);
|
|
|
|
Ogre::ResourceGroupManager::getSingleton().addResourceLocation(
|
|
|
|
package_path + kMaterialsDirectory + kScriptsDirectory, "FileSystem",
|
|
|
|
ROS_PACKAGE_NAME);
|
|
|
|
Ogre::ResourceGroupManager::getSingleton().initialiseAllResourceGroups();
|
|
|
|
}
|
|
|
|
|
2016-08-04 21:58:41 +08:00
|
|
|
SubmapsDisplay::~SubmapsDisplay() { client_.shutdown(); }
|
2016-08-03 18:45:08 +08:00
|
|
|
|
2016-08-05 17:50:22 +08:00
|
|
|
void SubmapsDisplay::Reset() { reset(); }
|
|
|
|
|
|
|
|
void SubmapsDisplay::CreateClient() {
|
|
|
|
client_ = update_nh_.serviceClient<::cartographer_ros_msgs::SubmapQuery>(
|
|
|
|
submap_query_service_property_->getStdString());
|
|
|
|
}
|
|
|
|
|
2016-08-03 18:45:08 +08:00
|
|
|
void SubmapsDisplay::onInitialize() {
|
2016-08-04 21:58:41 +08:00
|
|
|
MFDClass::onInitialize();
|
|
|
|
CreateClient();
|
2016-08-03 18:45:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void SubmapsDisplay::reset() {
|
2016-08-04 21:58:41 +08:00
|
|
|
MFDClass::reset();
|
|
|
|
::cartographer::common::MutexLocker locker(&mutex_);
|
|
|
|
client_.shutdown();
|
|
|
|
trajectories_.clear();
|
|
|
|
CreateClient();
|
2016-08-03 18:45:08 +08:00
|
|
|
}
|
|
|
|
|
2016-08-04 21:58:41 +08:00
|
|
|
void SubmapsDisplay::processMessage(
|
2016-08-03 18:57:56 +08:00
|
|
|
const ::cartographer_ros_msgs::SubmapList::ConstPtr& msg) {
|
2016-08-03 18:45:08 +08:00
|
|
|
::cartographer::common::MutexLocker locker(&mutex_);
|
2017-04-21 17:58:16 +08:00
|
|
|
// In case Cartographer node is relaunched, destroy
|
|
|
|
// trajectories from the previous instance
|
|
|
|
if (msg->trajectory.size() < trajectories_.size()) {
|
|
|
|
trajectories_.clear();
|
|
|
|
}
|
2016-10-13 17:44:19 +08:00
|
|
|
for (size_t trajectory_id = 0; trajectory_id < msg->trajectory.size();
|
2016-08-03 18:45:08 +08:00
|
|
|
++trajectory_id) {
|
|
|
|
if (trajectory_id >= trajectories_.size()) {
|
2017-04-21 17:58:16 +08:00
|
|
|
// When a trajectory is destroyed, it also needs to delete its rviz
|
|
|
|
// Property object, so we use a unique_ptr for it
|
|
|
|
trajectories_.push_back(Trajectory(
|
|
|
|
::cartographer::common::make_unique<::rviz::Property>(
|
|
|
|
QString("Trajectory %1").arg(trajectory_id), QVariant(),
|
|
|
|
QString("List of all submaps in Trajectory %1.")
|
|
|
|
.arg(trajectory_id),
|
|
|
|
submaps_category_),
|
|
|
|
std::vector<std::unique_ptr<DrawableSubmap>>()));
|
2016-08-03 18:45:08 +08:00
|
|
|
}
|
2017-04-21 17:58:16 +08:00
|
|
|
auto& trajectory_category = trajectories_[trajectory_id].first;
|
|
|
|
auto& trajectory = trajectories_[trajectory_id].second;
|
2016-08-03 18:57:56 +08:00
|
|
|
const std::vector<::cartographer_ros_msgs::SubmapEntry>& submap_entries =
|
2016-08-05 17:50:22 +08:00
|
|
|
msg->trajectory[trajectory_id].submap;
|
2017-04-21 17:58:16 +08:00
|
|
|
// Same as above, destroy the whole trajectory if we detect that
|
|
|
|
// we have more submaps than we should
|
|
|
|
if (submap_entries.size() < trajectory.size()) {
|
|
|
|
trajectory.clear();
|
|
|
|
}
|
2016-10-27 21:36:15 +08:00
|
|
|
for (size_t submap_index = 0; submap_index < submap_entries.size();
|
|
|
|
++submap_index) {
|
|
|
|
if (submap_index >= trajectory.size()) {
|
2016-08-05 20:23:30 +08:00
|
|
|
trajectory.push_back(
|
|
|
|
::cartographer::common::make_unique<DrawableSubmap>(
|
2017-04-21 17:58:16 +08:00
|
|
|
trajectory_id, submap_index, context_->getSceneManager(),
|
|
|
|
trajectory_category.get(), visibility_all_enabled_->getBool()));
|
2016-08-03 18:45:08 +08:00
|
|
|
}
|
2016-10-27 21:36:15 +08:00
|
|
|
trajectory[submap_index]->Update(msg->header,
|
|
|
|
submap_entries[submap_index],
|
|
|
|
context_->getFrameManager());
|
2016-08-03 18:45:08 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-05 17:50:22 +08:00
|
|
|
void SubmapsDisplay::update(const float wall_dt, const float ros_dt) {
|
2017-03-31 22:36:47 +08:00
|
|
|
::cartographer::common::MutexLocker locker(&mutex_);
|
2016-08-05 20:23:30 +08:00
|
|
|
// Update the fading by z distance.
|
2016-08-05 17:50:22 +08:00
|
|
|
try {
|
|
|
|
const ::geometry_msgs::TransformStamped transform_stamped =
|
|
|
|
tf_buffer_.lookupTransform(map_frame_property_->getStdString(),
|
|
|
|
tracking_frame_property_->getStdString(),
|
|
|
|
ros::Time(0) /* latest */);
|
|
|
|
for (auto& trajectory : trajectories_) {
|
2017-04-21 17:58:16 +08:00
|
|
|
for (auto& submap : trajectory.second) {
|
2016-08-05 17:50:22 +08:00
|
|
|
submap->SetAlpha(transform_stamped.transform.translation.z);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (const tf2::TransformException& ex) {
|
|
|
|
ROS_WARN("Could not compute submap fading: %s", ex.what());
|
|
|
|
}
|
2016-08-05 20:23:30 +08:00
|
|
|
|
|
|
|
// Schedule fetching of new submap textures.
|
|
|
|
for (const auto& trajectory : trajectories_) {
|
|
|
|
int num_ongoing_requests = 0;
|
2017-04-21 17:58:16 +08:00
|
|
|
for (const auto& submap : trajectory.second) {
|
2016-08-05 20:23:30 +08:00
|
|
|
if (submap->QueryInProgress()) {
|
|
|
|
++num_ongoing_requests;
|
|
|
|
}
|
|
|
|
}
|
2017-04-21 17:58:16 +08:00
|
|
|
for (int submap_index = static_cast<int>(trajectory.second.size()) - 1;
|
2016-10-27 21:36:15 +08:00
|
|
|
submap_index >= 0 &&
|
2016-08-05 20:23:30 +08:00
|
|
|
num_ongoing_requests < kMaxOnGoingRequestsPerTrajectory;
|
2016-10-27 21:36:15 +08:00
|
|
|
--submap_index) {
|
2017-04-21 17:58:16 +08:00
|
|
|
if (trajectory.second[submap_index]->MaybeFetchTexture(&client_)) {
|
2016-08-05 20:23:30 +08:00
|
|
|
++num_ongoing_requests;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-08-05 17:50:22 +08:00
|
|
|
}
|
|
|
|
|
2017-04-21 17:58:16 +08:00
|
|
|
void SubmapsDisplay::AllEnabledToggled() {
|
|
|
|
::cartographer::common::MutexLocker locker(&mutex_);
|
|
|
|
const bool visibility = visibility_all_enabled_->getBool();
|
|
|
|
for (auto& trajectory : trajectories_) {
|
|
|
|
for (auto& submap : trajectory.second) {
|
|
|
|
submap->set_visibility(visibility);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-20 16:30:21 +08:00
|
|
|
} // namespace cartographer_rviz
|
2016-08-03 18:45:08 +08:00
|
|
|
|
2016-09-20 16:30:21 +08:00
|
|
|
PLUGINLIB_EXPORT_CLASS(cartographer_rviz::SubmapsDisplay, ::rviz::Display)
|