Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added regex filter field for TF display #1744

Merged
merged 11 commits into from
Mar 29, 2023
144 changes: 113 additions & 31 deletions src/rviz/default_plugin/tf_display.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -28,9 +28,14 @@
*/

#include <boost/bind.hpp>
#include <algorithm>
#include <regex>

#include <OgreSceneNode.h>
#include <OgreSceneManager.h>
#include <QValidator>
#include <QLineEdit>
#include <QToolTip>

#include <tf/transform_listener.h>

Expand Down Expand Up @@ -162,6 +167,75 @@ void FrameSelectionHandler::setOrientation(const Ogre::Quaternion& orientation)
}
}

class RegexValidator : public QValidator
{
QLineEdit* editor_;

public:
RegexValidator(QLineEdit* editor) : QValidator(editor), editor_(editor)
{
}
QValidator::State validate(QString& input, int& /*pos*/) const override
{
try
{
std::regex(input.toLocal8Bit().constData());
editor_->setStyleSheet(QString());
QToolTip::hideText();
return QValidator::Acceptable;
}
catch (const std::regex_error& e)
{
editor_->setStyleSheet("background: #ffe4e4");
QToolTip::showText(editor_->mapToGlobal(QPoint(0, 5)), tr(e.what()), editor_, QRect(), 5000);
return QValidator::Intermediate;
}
}
};

class RegexFilterProperty : public StringProperty
{
std::regex default_;
std::regex regex_;

void onValueChanged()
{
const auto& value = getString();
if (value.isEmpty())
regex_ = default_;
else
{
try
{
regex_.assign(value.toLocal8Bit().constData(), std::regex_constants::optimize);
}
catch (const std::regex_error& e)
{
regex_ = default_;
}
}
}

public:
RegexFilterProperty(const QString& name, const std::regex regex, Property* parent)
: StringProperty(name, "", "regular expression", parent), default_(regex), regex_(regex)
{
QObject::connect(this, &RegexFilterProperty::changed, this, [this]() { onValueChanged(); });
}
const std::regex& regex() const
{
return regex_;
}

QWidget* createEditor(QWidget* parent, const QStyleOptionViewItem& option) override
{
auto* editor = qobject_cast<QLineEdit*>(StringProperty::createEditor(parent, option));
if (editor)
editor->setValidator(new RegexValidator(editor));
return editor;
}
};

typedef std::set<FrameInfo*> S_FrameInfo;

TFDisplay::TFDisplay() : Display(), update_timer_(0.0f), changing_single_frame_enabled_state_(false)
Expand All @@ -181,6 +255,9 @@ TFDisplay::TFDisplay() : Display(), update_timer_(0.0f), changing_single_frame_e
scale_property_ =
new FloatProperty("Marker Scale", 1, "Scaling factor for all names, axes and arrows.", this);

filter_whitelist_property_ = new RegexFilterProperty("Filter (whitelist)", std::regex(""), this);
filter_blacklist_property_ = new RegexFilterProperty("Filter (blacklist)", std::regex(), this);

update_rate_property_ = new FloatProperty("Update Interval", 0,
"The interval, in seconds, at which to update the frame "
"transforms. 0 means to do so every update cycle.",
Expand Down Expand Up @@ -257,7 +334,7 @@ void TFDisplay::clear()

// Clear all frames
while (!frames_.empty())
deleteFrame(frames_.begin(), false);
deleteFrame(frames_.begin(), false, false);

update_timer_ = 0.0f;

Expand Down Expand Up @@ -377,42 +454,45 @@ void TFDisplay::updateFrames()
#ifndef _WIN32
#pragma GCC diagnostic pop
#endif
std::sort(frames.begin(), frames.end());

S_FrameInfo current_frames;

// filter frames according to white- and black-list regular expressions
auto it = frames.begin(), end = frames.end();
while (it != end)
{
for (const std::string& frame : frames)
{
if (frame.empty())
{
continue;
}
if (it->empty() || !std::regex_search(*it, filter_whitelist_property_->regex()) ||
std::regex_search(*it, filter_blacklist_property_->regex()))
std::swap(*it, *--end); // swap current to-be-dropped name with last one
else
++it;
}

FrameInfo* info = getFrameInfo(frame);
if (!info)
{
info = createFrame(frame);
}
else
{
updateFrame(info);
}
std::sort(frames.begin(), end);
std::sort(end, frames.end());

current_frames.insert(info);
S_FrameInfo current_frames;
for (it = frames.begin(); it != end; ++it)
{
FrameInfo* info = getFrameInfo(*it);
if (!info)
{
info = createFrame(*it);
}
else
{
updateFrame(info);
}

current_frames.insert(info);
}

it = end;
end = frames.end();
for (auto frame_it = frames_.begin(), frame_end = frames_.end(); frame_it != frame_end;)
{
M_FrameInfo::iterator frame_it = frames_.begin();
M_FrameInfo::iterator frame_end = frames_.end();
while (frame_it != frame_end)
{
if (current_frames.find(frame_it->second) == current_frames.end())
frame_it = deleteFrame(frame_it, true);
else
++frame_it;
}
if (current_frames.find(frame_it->second) == current_frames.end())
frame_it = deleteFrame(frame_it, true, std::lower_bound(it, end, frame_it->first) == end);
else
++frame_it;
}

context_->queueRender();
Expand Down Expand Up @@ -711,7 +791,8 @@ void TFDisplay::updateFrame(FrameInfo* frame)
frame->selection_handler_->setParentName(frame->parent_);
}

TFDisplay::M_FrameInfo::iterator TFDisplay::deleteFrame(M_FrameInfo::iterator it, bool delete_properties)
TFDisplay::M_FrameInfo::iterator
TFDisplay::deleteFrame(M_FrameInfo::iterator it, bool delete_properties, bool delete_tree_property)
{
FrameInfo* frame = it->second;
it = frames_.erase(it);
Expand All @@ -724,7 +805,8 @@ TFDisplay::M_FrameInfo::iterator TFDisplay::deleteFrame(M_FrameInfo::iterator it
if (delete_properties)
{
delete frame->enabled_property_;
delete frame->tree_property_;
if (delete_tree_property)
delete frame->tree_property_;
}
delete frame;
return it;
Expand Down
7 changes: 5 additions & 2 deletions src/rviz/default_plugin/tf_display.h
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@ class VectorProperty;
class FrameInfo;
class FrameSelectionHandler;
typedef boost::shared_ptr<FrameSelectionHandler> FrameSelectionHandlerPtr;
class RegexFilterProperty;

/** @brief Displays a visual representation of the TF hierarchy. */
class TFDisplay : public Display
Expand Down Expand Up @@ -89,7 +90,8 @@ private Q_SLOTS:
void updateFrames();
FrameInfo* createFrame(const std::string& frame);
void updateFrame(FrameInfo* frame);
M_FrameInfo::iterator deleteFrame(M_FrameInfo::iterator it, bool delete_properties);
M_FrameInfo::iterator
deleteFrame(M_FrameInfo::iterator it, bool delete_properties, bool delete_tree_property);

FrameInfo* getFrameInfo(const std::string& frame);

Expand Down Expand Up @@ -119,7 +121,8 @@ private Q_SLOTS:
BoolProperty* all_enabled_property_;

FloatProperty* scale_property_;

RegexFilterProperty* filter_whitelist_property_;
RegexFilterProperty* filter_blacklist_property_;
Property* frames_category_;
Property* tree_category_;

Expand Down