Radium Engine
1.5.0
|
KeyMapping is defined around two concepts, context (Ra::Gui::KeyMappingManager::Context) and action (Ra::Gui::KeyMappingManager::KeyMappingAction). An action is associated to an event binding (Ra::Gui::KeyMappingManager::EventBinding), this binding represents the input (mouse, keys, modifiers) that are associated with the action. The context allows to use same binding for different actions depending of the considered context. The first widget that receive input event needs to manage contexts and determine which context to use to request corresponding action. Typically the viewer is this first widget, and can decide which context to use depending on what is under mouse, if gizmo edit is enable, or if an active camera manipulator is active. Action, context and binding are managed with Ra::Gui::KeyMappingManager singleton.
Context and action are simply indices (handle) stored in the key mapping manager, they are associated with std::string name, to ease user key mapping management. Typical usage is to have one context per class that may have its specific key mapping actions.
On the user side, one can test which action corresponds to a given binding (the one that triggers the event for instance), or use Ra::Gui::KeyMappingCallbackManager to trigger registered callback.
First step is to add the context, the actions and associated bindings to the manager. To this end you can either use configuration file or call Ra::Gui::KeyMappingManager::addContext and Ra::Gui::KeyMappingManager::addAction.
This can be done with a xml configuration file. Viewer and camera manipulator basic configuration file is bundled with radium install (located in the install_dir/Configs/
folder). You can extends edit this file or copy it somewhere else, and load it with Ra::Gui::KeyMappingManager::loadConfiguration.
This file as an only keymaps
node, with one keymap
node per binding.
Default values are in [ ]
ACTION
mandatory represents the KeyMappingAction enum's value you want to trigger.CONTEXT
[AppContext] represents the context, for instance the class that will try to catch this keymappingKEY
[-1] represents the key that need to be pressed to trigger the event (ie Key_Z
, for example), "" or "-1" or absent correspond to no key needed.MODIFIERS
[NoModifiers] represents the modifier used along with key or mouse button (needs to be a Qt::Modifier
enum value) to trigger the action. Multiples modifiers can be specified, separated by commas as in "ControlModifier,ShiftModifier"
.BUTTONS
[NoButtons] represents the button to trigger the event (i.e. LeftButton
, for example).WHEEL
[false] if true, it's a wheel event ! (anything else is false).If only a key is defined then it's a keyPressedEvent
. If buttons is defined, then it's a mouse[move/press/release]Event
, that optionally take modifiers and key into account. If wheel is true, then it's a wheel event, that optionally take modifiers, buttons and key into account.
On the implementation side, your class C
need (could) derive from KeyMappingManageable<C>
, it defines the static member variable m_keyMappingContext
, and the static function configureKeyMapping()
, that calls configureKeyMapping_impl()
, which have to be implemented in your class C
.
Then you can store your specific actions as static member of your class, to ease checking which action is triggered.
Define the static action variable in .cpp
And configureKeyMapping_impl
which will be called by KeyMappingManager on configuration changes.
And also add a listener so key mapping context and action handles are updated in case of new configuration load. This can be done typically in application or viewer constructor, but after KeyMappingManager instance is created.
To ease the initialization of Action handles, you can use macro as shown in Viewer.hpp and Viewer.cpp (KeyMappingViewer and KMA_VALUE).
The viewer is the main entry point to dispatch key and mouse event. The idea is that at a key press or mouse press event, the viewer is capable of determining which class will receive the events.
For instance see Viewer.cpp
Viewer::mousePressEvent
:
If you want to define a derived class that inherits a base class with key mapping, and you want to have key mapping management in this derived, you have to consider the following implementation details:
setupKeymappingcallbacks
should call base setupKeymappingcallbacks
, hence derived configureKeymapping_impl
should only care about its own KeyMappingAction
.Here is an example snippets.
An easy way to manage action callback is to use KeyMappingCallbackManager. This manager is a class member, initialized with Context. One can register callback (std::function<void(QEvent*)>) that is triggered if the input event corresponds to the event binding.
The callback is responsible to test if the event corresponds to an event it can handle.
Example usage