The main goal of this MEP is to make it easier to modify (add, change, remove) the way the user interacts with the figures.
The user interaction with the figure is deeply integrated within the Canvas and Toolbar. Making extremely difficult to do any modification.
This MEP proposes the separation of this interaction into Toolbar, Navigation and Tools to provide independent access and reconfiguration.
This approach will make easier to create and share tools among
users. In the far future, we can even foresee a kind of Marketplace
for Tools
where the most popular can be added into the main
distribution.
The reconfiguration of the Toolbar is complex, most of the time it requires a custom backend.
The creation of custom Tools sometimes interferes with the Toolbar, as example see https://github.com/matplotlib/matplotlib/issues/2694 also the shortcuts are hardcoded and again not easily modifiable https://github.com/matplotlib/matplotlib/issues/2699
The proposed solution is to take the actions out of the Toolbar
and
the shortcuts out of the Canvas
. This actions and shortcuts will be
in the form of Tools
.
A new class Navigation
will be the bridge between the events from
the Canvas
and Toolbar
and redirect them to the appropriate Tool
.
At the end the user interaction will be divided into three classes:
- NavigationBase: This class is instantiated for each FigureManager and connect the all user interactions with the Tools
- ToolbarBase: This existing class is relegated only as a GUI access to Tools.
- ToolBase: Is the basic definition of Tools.
Tools can have a graphical representation as the SubplotTool
or not even be present in the Toolbar as Quit
The ToolBase
has the following class attributes for configuration at definition time
- keymap = None: Key(s) to be used to trigger the tool
- description = ‘’: Small description of the tool
- image = None: Image that is used in the toolbar
destroy(self, *args)
: Destroy the Tool
graphical interface (if exists)The ToolToggleBase
has the following class attributes for
configuration at definition time
- radio_group = None: Attribute to group ‘radio’ like tools (mutually exclusive)
- cursor = None: Cursor to use when the tool is active
The Toggleable Tools, can capture keypress, mouse moves, and mouse button press
ToolToggleBase.trigger
methodtool_added_event
(emitted by navigation)Toolbar
For backward compatibility added a ‘navigation’ key to
rcsetup.validate_toolbar
, that is used for Navigation classes
instantiation instead of the NavigationToolbar classes
With this parameter, it makes it transparent to anyone using the existing backends.
[@pelson comment: This also gives us an opportunity to avoid needing to implement all of this in the same PR - some backends can potentially exist without the new functionality for a short while (but it must be done at some point).]