This is intended as a guide for people wishing to develop custom plugins.
Plugins can extend RViz at different extension points:
plugin type | base type |
---|---|
Display | rviz_common::Display |
Panel | rviz_common::Panel |
Tool | rviz_common::Tool |
Frames transformation library | rviz_common::transformation::FrameTransformer |
View Controller | rviz_common::ViewController |
Every plugin must be of the corresponding base type in order to be recognized by RViz. Refer to the documentation of the relevant base class for a detailed API description.
- In order to write your own plugin, set up a regular ROS 2 workspace (see the official documentation for further help).
- You will need to link your program against
rviz_common
and probably also againstrviz_rendering
andrviz_default_plugins
. - To let the plugin loader find your plugin, you will need to invoke the
PLUGINLIB_EXPORT_CLASS
macro. For instance, if your plugin is called "myplugin::Plugin" with base classrviz_common::Display
, you will need to include the following into your.cpp
file:
#include <pluginlib/class_list_macros.hpp>
PLUGINLIB_EXPORT_CLASS(myplugin::Plugin, rviz_common::Display)
- You need to link and compile against the
pluginlib
package. Further CMake options that might be relevant (seerviz_default_plugins
CMakeLists):
# prevent pluginlib from using boost
target_compile_definitions(rviz_default_plugins PUBLIC "PLUGINLIB__DISABLE_BOOST_FUNCTIONS")
# Causes the visibility macros to use dllexport rather than dllimport (for Windows, when your plugin should be used as library)
target_compile_definitions(rviz_default_plugins PRIVATE "RVIZ_DEFAULT_PLUGINS_BUILDING_LIBRARY")
- You need to write a
plugin_description.xml
file which contains the information necessary for the pluginlib to load your plugin at runtime. Seerviz_default_plugins/plugins_description.xml
for an example (the syntax is the same as for the old RViz) - Export the plugin description file via
pluginlib_export_plugin_description_file(rviz_common plugins_description.xml)
This should make sure that your plugins are found by rviz_common at runtime as described here
- In order to write a display plugin, derive from either
rviz_common::Display
orrviz_common::RosTopicDisplay<MessageType>
.
When writing a display without subscription to ros, derive from rviz_common::Display
.
- The Display class provides an
Ogre::SceneNode
, which can be used to add visual objects. - It provides convenience methods to set and delete Status messages
- When you need any Ogre or SceneNode setup, override the
onInitilize()
method.
When writing a display for a topic type, derive from rviz_common::RosTopicDisplay<MessageType>
.
The RosTopicDisplay is templated on the message type of this display:
- RosTopicDisplay derives from Display and thereby contains all of the functionality described in the previous section
- It handles subscribing and unsubscribing to topics of its template type
- It provides two properties: a property to choose the topic and a property to choose the QoS profile ("Quality of Service", currently "reliable" or "unreliable")
- It provides convenient methods to receive messages and do initial checking. Override the
processMessage
to add behaviour on receiving a new message
It is possible to add your own meshes, scripts, etc. to RViz.
To make them available at runtime, use the cmake macro register_rviz_ogre_media_exports
defined in rviz_rendering
For instance, if you want to register a folder test_folder/scripts
which resides in your project's source folder, write
register_rviz_ogre_media_exports(DIRECTORIES "test_folder/scripts")
Note: If you want to export folder hierarchies, each folder needs to be exported separately. Folders within exported folders are not automatically available.
Some of the default displays (e.g. TF display or LaserScan display) can only work with tf2 as transformation framework.
Similarly, as a developer, you may have written your own transformation plugin which cannot work with every display, or your own display which cannot work with every transformation plugin.
The class rviz_default_plugins::transformation::TransformerGuard
helps you to handle these possibilities.
As explained below, every transformation plugin implements the base class rviz_common::transformation::FrameTransformer
.
The TransformerGuard
class is templated on the type of this implementation and will make sure that the display by which it is owned will be disabled if the transformer currently in use is of a different type.
When writing a display that is supposed to be working only with a specific transformation plugin, you can add a templated instance of TransformerGuard
as a display member.
The constructor of this class takes two argument: a raw pointer to the display that owns it, and the name of the transformer it is supposed to work with (needed to correctly set the error status).
Ideally the TransformerGuard
object is initialized in the constructor of the display.
In order to make the TransformerGuard
object work correctly, you need to call the method TransformerGuard::initialize(rviz_common::DisplayContext * context)
inside the display onInitialize()
method, passing as parameter the context_
member of the display.
Doing this will make sure that every time the current transformer plugin is of the wrong type, the display will be disabled and an appropriate error message will be shown in the display status.
In addition to this, the TransformerGuard
class also provides the checkTransformer()
method, which returns true
if the currently usd transformer is of the allowed type, and false
otherwise.
This method can be used by the display every time you want it to behave differently according to what kind of transformation plugin is used at the moment.
For a concrete example of how all this is done, you can have a look, for example, at the TF or the LaserScan display, in rviz_default_plugins
which, as said, only work with tf2.
To write a custom panel, derive from rviz_common::Panel
.
- When writing a tool, derive from
rviz_common::Tool
- Hotkeys for activating and deactivating tools can be set by setting the member variable
shortcut_key_
to the char of your choice - Adding properties to the "Tool Properties" panel is possible by passing the return value of the function
getPropertyContainer()
as parent to the property in question (seerviz_default_plugins::tools::MeasureTool
for an example) - Custom cursors, icons, etc. for use in your plugins are available through
rviz_common/load_resource
- In order to write a custom view controller which should be independent of tf frames, derive from
rviz_common::ViewController
- If your view controller should be able to track tf frames in a scene, derive from
rviz_common::FramePositionTrackingViewController
, which already contains convenience functionality for tracking target frames - If your custom view controller orbits a focal point, it might also be beneficial to derive from
rviz_default_plugins::OrbitViewController
- To write a transformer plugin, you must implement the
rviz_common::transformation::FrameTransformer
class (refer to the API documentation contained in the header file). - If for your plugin you also need extra functionality, or you want to offer direct access to some parts of the library itself, you can also implement a subclass of
rviz_common::transformation::TransformerLibraryConnector
(see, for example,rviz_default_plugins::transformation::TFWrapper
).
The rviz_rendering
package is supposed to contain all functionality referring to rendering:
- Visuals and objects to be added to the scene graph such as arrows, shapes or text objects in the subfolder
objects
(many of those objects were ported from the folderogre_helpers
) - The render window, including functions exposing some of its internals (RenderWindowOgreAdapter). If possible, refrain from using the RenderWindowOgreAdapter, as it might be deprecated and deleted in the future
- Convenience functions to work with materials (
material_manager.hpp
) or other ogre related functionality (e.g. ray tracing inviewport_projection_finder.hpp
) - Convenience classes for testing, allowing to set up a working ogre environment and helpers for scene graph introspection
The rviz_common
package contains the bulk of RViz useful for doing plugin development:
- Main application and rendering queues (not exposed)
- Main entry points for plugin development - base classes for panels, view controllers, displays and tools
- Convenience classes to work with properties in the various views (such as the display panel) located in
rviz_common/properties
- Main classes for selectable types, the SelectionHandlers (located in
rviz_common/interaction
) - Access points to ROS 2.
Currently, RViz uses only one node, which can be accessed via
ros_integration
. In the future, further changes may be necessary to fully abstract ROS 2 access intoros_integration
The rviz_default_plugins
contains all plugins (view controllers, tools, displays and in the future, panels) shipped with RViz (most of them ported from the default_plugins
folder of RViz).
- When developing simple plugins it is not necessary to use anything in this package.
- When developing more complex plugins similar to existing plugins it might be beneficial to use or even derive from classes contained in this package to simplify your development.
The rviz_visual_testing_framework
contains the backbone to writing visual tests for plugins.
It will only ever be necessary to use this package as a test dependency if you want to write automated screenshot tests.
Please see the documentation in the package for further help.
This package contains the main program startup and entry point for nodes (not exposed).