Skip to content

Commit

Permalink
Fix imgui key mapping
Browse files Browse the repository at this point in the history
  • Loading branch information
jslee02 committed Nov 10, 2024
1 parent 1093809 commit 3a45dbe
Show file tree
Hide file tree
Showing 3 changed files with 641 additions and 688 deletions.
190 changes: 59 additions & 131 deletions dart/gui/osg/ImGuiHandler.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -46,95 +46,77 @@
#include <osg/RenderInfo>

#include <algorithm>
#include <iostream>

namespace dart {
namespace gui {
namespace osg {

//==============================================================================
// Special keys that are usually greater than 512 in osgGA
//
// Imporant Note: Dear ImGui expects the control Keys indices not to be greater
// thant 511. It actually uses an array of 512 elements. However, OSG has
// indices greater than that. So here I do a conversion for special keys between
// ImGui and OSG.
enum ConvertedKey : int
{
ConvertedKey_Tab = 257,
ConvertedKey_Left,
ConvertedKey_Right,
ConvertedKey_Up,
ConvertedKey_Down,
ConvertedKey_PageUp,
ConvertedKey_PageDown,
ConvertedKey_Home,
ConvertedKey_End,
ConvertedKey_Delete,
ConvertedKey_BackSpace,
ConvertedKey_Enter,
ConvertedKey_Escape,
ConvertedKey_LeftControl,
ConvertedKey_RightControl,
ConvertedKey_LeftShift,
ConvertedKey_RightShift,
ConvertedKey_LeftAlt,
ConvertedKey_RightAlt,
ConvertedKey_LeftSuper,
ConvertedKey_RightSuper,
};

//==============================================================================
// Check for a special key and return the converted code (range [257, 511]) if
// so. Otherwise returns -1
int convertFromOSGKey(int key)

// The ConvertedKey enum and convertFromOSGKey() function can be adjusted to map
// directly to ImGuiKey values. Since Dear ImGui now provides ImGuiKey enums for
// all keys, you can map OSG keys directly to these enums.
ImGuiKey convertFromOSGKey(int key)
{
using KeySymbol = osgGA::GUIEventAdapter::KeySymbol;

switch (key) {
case KeySymbol::KEY_Tab:
return ConvertedKey_Tab;
return ImGuiKey_Tab;
case KeySymbol::KEY_Left:
return ConvertedKey_Left;
return ImGuiKey_LeftArrow;
case KeySymbol::KEY_Right:
return ConvertedKey_Right;
return ImGuiKey_RightArrow;
case KeySymbol::KEY_Up:
return ConvertedKey_Up;
return ImGuiKey_UpArrow;
case KeySymbol::KEY_Down:
return ConvertedKey_Down;
return ImGuiKey_DownArrow;
case KeySymbol::KEY_Page_Up:
return ConvertedKey_PageUp;
return ImGuiKey_PageUp;
case KeySymbol::KEY_Page_Down:
return ConvertedKey_PageDown;
return ImGuiKey_PageDown;
case KeySymbol::KEY_Home:
return ConvertedKey_Home;
return ImGuiKey_Home;
case KeySymbol::KEY_End:
return ConvertedKey_End;
return ImGuiKey_End;
case KeySymbol::KEY_Delete:
return ConvertedKey_Delete;
return ImGuiKey_Delete;
case KeySymbol::KEY_BackSpace:
return ConvertedKey_BackSpace;
return ImGuiKey_Backspace;
case KeySymbol::KEY_Return:
return ConvertedKey_Enter;
return ImGuiKey_Enter;
case KeySymbol::KEY_Escape:
return ConvertedKey_Escape;
return ImGuiKey_Escape;
case KeySymbol::KEY_Control_L:
return ConvertedKey_LeftControl;
case KeySymbol::KEY_Control_R:
return ConvertedKey_RightControl;
return ImGuiKey_ModCtrl;
case KeySymbol::KEY_Shift_L:
return ConvertedKey_LeftShift;
case KeySymbol::KEY_Shift_R:
return ConvertedKey_RightShift;
return ImGuiKey_ModShift;
case KeySymbol::KEY_Alt_L:
return ConvertedKey_LeftAlt;
case KeySymbol::KEY_Alt_R:
return ConvertedKey_RightAlt;
return ImGuiKey_ModAlt;
case KeySymbol::KEY_Super_L:
return ConvertedKey_LeftSuper;
case KeySymbol::KEY_Super_R:
return ConvertedKey_RightSuper;
return ImGuiKey_ModSuper;
case KeySymbol::KEY_A:
return ImGuiKey_A;
case KeySymbol::KEY_C:
return ImGuiKey_C;
case KeySymbol::KEY_V:
return ImGuiKey_V;
case KeySymbol::KEY_X:
return ImGuiKey_X;
case KeySymbol::KEY_Y:
return ImGuiKey_Y;
case KeySymbol::KEY_Z:
return ImGuiKey_Z;
// Add other key mappings as needed
default:
return -1;
return ImGuiKey_None;
}
}

Expand Down Expand Up @@ -181,29 +163,6 @@ ImGuiHandler::ImGuiHandler()
ImGui::StyleColorsDark();

ImGui_ImplOpenGL2_Init();

// Keyboard mapping. ImGui will use those indices to peek into the
// io.KeyDown[] array.
ImGuiIO& io = ImGui::GetIO();
io.KeyMap[ImGuiKey_Tab] = ConvertedKey_Tab;
io.KeyMap[ImGuiKey_LeftArrow] = ConvertedKey_Left;
io.KeyMap[ImGuiKey_RightArrow] = ConvertedKey_Right;
io.KeyMap[ImGuiKey_UpArrow] = ConvertedKey_Up;
io.KeyMap[ImGuiKey_DownArrow] = ConvertedKey_Down;
io.KeyMap[ImGuiKey_PageUp] = ConvertedKey_PageUp;
io.KeyMap[ImGuiKey_PageDown] = ConvertedKey_PageDown;
io.KeyMap[ImGuiKey_Home] = ConvertedKey_Home;
io.KeyMap[ImGuiKey_End] = ConvertedKey_End;
io.KeyMap[ImGuiKey_Delete] = ConvertedKey_Delete;
io.KeyMap[ImGuiKey_Backspace] = ConvertedKey_BackSpace;
io.KeyMap[ImGuiKey_Enter] = ConvertedKey_Enter;
io.KeyMap[ImGuiKey_Escape] = ConvertedKey_Escape;
io.KeyMap[ImGuiKey_A] = osgGA::GUIEventAdapter::KeySymbol::KEY_A;
io.KeyMap[ImGuiKey_C] = osgGA::GUIEventAdapter::KeySymbol::KEY_C;
io.KeyMap[ImGuiKey_V] = osgGA::GUIEventAdapter::KeySymbol::KEY_V;
io.KeyMap[ImGuiKey_X] = osgGA::GUIEventAdapter::KeySymbol::KEY_X;
io.KeyMap[ImGuiKey_Y] = osgGA::GUIEventAdapter::KeySymbol::KEY_Y;
io.KeyMap[ImGuiKey_Z] = osgGA::GUIEventAdapter::KeySymbol::KEY_Z;
}

//==============================================================================
Expand Down Expand Up @@ -272,64 +231,33 @@ bool ImGuiHandler::handle(
::osg::Object* /*object*/,
::osg::NodeVisitor* /*nodeVisitor*/)
{
auto& io = ImGui::GetIO();
const auto wantCapureMouse = io.WantCaptureMouse;
const auto wantCapureKeyboard = io.WantCaptureKeyboard;
ImGuiIO& io = ImGui::GetIO();
const bool wantCaptureMouse = io.WantCaptureMouse;
const bool wantCaptureKeyboard = io.WantCaptureKeyboard;

switch (eventAdapter.getEventType()) {
case osgGA::GUIEventAdapter::KEYDOWN: {
const auto c = eventAdapter.getUnmodifiedKey();
const auto special_key = convertFromOSGKey(c);

if (special_key > 0) {
assert(special_key < 512 && "ImGui KeysDown is an array of 512");
assert(
special_key > 256
&& "ASCII stop at 127, but we use the range [257, 511]");

io.KeysDown[special_key] = true;

io.KeyCtrl = io.KeysDown[ConvertedKey_LeftControl]
|| io.KeysDown[ConvertedKey_RightControl];
io.KeyShift = io.KeysDown[ConvertedKey_LeftShift]
|| io.KeysDown[ConvertedKey_RightShift];
io.KeyAlt = io.KeysDown[ConvertedKey_LeftAlt]
|| io.KeysDown[ConvertedKey_RightAlt];
io.KeySuper = io.KeysDown[ConvertedKey_LeftSuper]
|| io.KeysDown[ConvertedKey_RightSuper];
} else if (0 < c && c < 0x10000) {
io.KeysDown[c] = true;
io.AddInputCharacter(static_cast<ImWchar>(c));
const int key = eventAdapter.getUnmodifiedKey();
const ImGuiKey imguiKey = convertFromOSGKey(key);

if (imguiKey != ImGuiKey_None) {
io.AddKeyEvent(imguiKey, true);
} else if (key != 0 && key < 0x10000) {
const ImWchar c = static_cast<ImWchar>(eventAdapter.getKey());
io.AddInputCharacter(c);
}

return wantCapureKeyboard;
return wantCaptureKeyboard;
}
case osgGA::GUIEventAdapter::KEYUP: {
const auto c = eventAdapter.getUnmodifiedKey();
const auto special_key = convertFromOSGKey(c);

if (special_key > 0) {
assert(special_key < 512 && "ImGui KeysDown is an array of 512");
assert(
special_key > 256
&& "ASCII stop at 127, but we use the range [257, 511]");

io.KeysDown[special_key] = false;

io.KeyCtrl = io.KeysDown[ConvertedKey_LeftControl]
|| io.KeysDown[ConvertedKey_RightControl];
io.KeyShift = io.KeysDown[ConvertedKey_LeftShift]
|| io.KeysDown[ConvertedKey_RightShift];
io.KeyAlt = io.KeysDown[ConvertedKey_LeftAlt]
|| io.KeysDown[ConvertedKey_RightAlt];
io.KeySuper = io.KeysDown[ConvertedKey_LeftSuper]
|| io.KeysDown[ConvertedKey_RightSuper];
} else if (0 < c && c < 0x10000) {
io.KeysDown[c] = false;
io.AddInputCharacter(static_cast<ImWchar>(c));
const int key = eventAdapter.getUnmodifiedKey();
const ImGuiKey imguiKey = convertFromOSGKey(key);

if (imguiKey != ImGuiKey_None) {
io.AddKeyEvent(imguiKey, false);
}

return wantCapureKeyboard;
return wantCaptureKeyboard;
}
case osgGA::GUIEventAdapter::PUSH: {
io.MousePos
Expand All @@ -351,14 +279,14 @@ bool ImGuiHandler::handle(
mMousePressed[0] = true;
}

return wantCapureMouse;
return wantCaptureMouse;
}
case osgGA::GUIEventAdapter::DRAG:
case osgGA::GUIEventAdapter::MOVE: {
io.MousePos
= ImVec2(eventAdapter.getX(), io.DisplaySize.y - eventAdapter.getY());

return wantCapureMouse;
return wantCaptureMouse;
}
case osgGA::GUIEventAdapter::RELEASE: {
// When a mouse button is released no button mask is set. So we mark all
Expand All @@ -367,7 +295,7 @@ bool ImGuiHandler::handle(
mMousePressed[1] = false;
mMousePressed[2] = false;

return wantCapureMouse;
return wantCaptureMouse;
}
case osgGA::GUIEventAdapter::SCROLL: {
constexpr float increment = 0.1f;
Expand All @@ -390,7 +318,7 @@ bool ImGuiHandler::handle(
break;
}

return wantCapureMouse;
return wantCaptureMouse;
}
default: {
return false;
Expand Down
Loading

0 comments on commit 3a45dbe

Please sign in to comment.