Skip to content

Commit a9e5382

Browse files
committed
Backends: Win32, SDL2, SDL3, GLFW: prioritize scancodes instead of translated keycodes when dealing with OEM keys + Added ImGuiKey_Oem102. (#7136, #7201, #7206, #7306, #7670, #7672, #8468)
1 parent 88d4827 commit a9e5382

7 files changed

+105
-30
lines changed

backends/imgui_impl_glfw.cpp

+3
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828

2929
// CHANGELOG
3030
// (minor and older changes stripped away, please see git history for details)
31+
// 2025-03-10: Map GLFW_KEY_WORLD_1 and GLFW_KEY_WORLD_2 into ImGuiKey_Oem102.
3132
// 2025-03-03: Fixed clipboard handler assertion when using GLFW <= 3.2.1 compiled with asserts enabled.
3233
// 2024-08-22: Moved some OS/backend related function pointers from ImGuiIO to ImGuiPlatformIO:
3334
// - io.GetClipboardTextFn -> platform_io.Platform_GetClipboardTextFn
@@ -221,6 +222,8 @@ ImGuiKey ImGui_ImplGlfw_KeyToImGuiKey(int keycode, int scancode)
221222
case GLFW_KEY_EQUAL: return ImGuiKey_Equal;
222223
case GLFW_KEY_LEFT_BRACKET: return ImGuiKey_LeftBracket;
223224
case GLFW_KEY_BACKSLASH: return ImGuiKey_Backslash;
225+
case GLFW_KEY_WORLD_1: return ImGuiKey_Oem102;
226+
case GLFW_KEY_WORLD_2: return ImGuiKey_Oem102;
224227
case GLFW_KEY_RIGHT_BRACKET: return ImGuiKey_RightBracket;
225228
case GLFW_KEY_GRAVE_ACCENT: return ImGuiKey_GraveAccent;
226229
case GLFW_KEY_CAPS_LOCK: return ImGuiKey_CapsLock;

backends/imgui_impl_sdl2.cpp

+26-8
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121

2222
// CHANGELOG
2323
// (minor and older changes stripped away, please see git history for details)
24+
// 2025-03-10: When dealing with OEM keys, use scancodes instead of translated keycodes to choose ImGuiKey values. (#7136, #7201, #7206, #7306, #7670, #7672, #8468)
2425
// 2025-02-26: Only start SDL_CaptureMouse() when mouse is being dragged, to mitigate issues with e.g.Linux debuggers not claiming capture back. (#6410, #3650)
2526
// 2025-02-24: Avoid calling SDL_GetGlobalMouseState() when mouse is in relative mode.
2627
// 2025-02-18: Added ImGuiMouseCursor_Wait and ImGuiMouseCursor_Progress mouse cursor support.
@@ -209,17 +210,17 @@ ImGuiKey ImGui_ImplSDL2_KeyEventToImGuiKey(SDL_Keycode keycode, SDL_Scancode sca
209210
case SDLK_SPACE: return ImGuiKey_Space;
210211
case SDLK_RETURN: return ImGuiKey_Enter;
211212
case SDLK_ESCAPE: return ImGuiKey_Escape;
212-
case SDLK_QUOTE: return ImGuiKey_Apostrophe;
213+
//case SDLK_QUOTE: return ImGuiKey_Apostrophe;
213214
case SDLK_COMMA: return ImGuiKey_Comma;
214-
case SDLK_MINUS: return ImGuiKey_Minus;
215+
//case SDLK_MINUS: return ImGuiKey_Minus;
215216
case SDLK_PERIOD: return ImGuiKey_Period;
216-
case SDLK_SLASH: return ImGuiKey_Slash;
217+
//case SDLK_SLASH: return ImGuiKey_Slash;
217218
case SDLK_SEMICOLON: return ImGuiKey_Semicolon;
218-
case SDLK_EQUALS: return ImGuiKey_Equal;
219-
case SDLK_LEFTBRACKET: return ImGuiKey_LeftBracket;
220-
case SDLK_BACKSLASH: return ImGuiKey_Backslash;
221-
case SDLK_RIGHTBRACKET: return ImGuiKey_RightBracket;
222-
case SDLK_BACKQUOTE: return ImGuiKey_GraveAccent;
219+
//case SDLK_EQUALS: return ImGuiKey_Equal;
220+
//case SDLK_LEFTBRACKET: return ImGuiKey_LeftBracket;
221+
//case SDLK_BACKSLASH: return ImGuiKey_Backslash;
222+
//case SDLK_RIGHTBRACKET: return ImGuiKey_RightBracket;
223+
//case SDLK_BACKQUOTE: return ImGuiKey_GraveAccent;
223224
case SDLK_CAPSLOCK: return ImGuiKey_CapsLock;
224225
case SDLK_SCROLLLOCK: return ImGuiKey_ScrollLock;
225226
case SDLK_NUMLOCKCLEAR: return ImGuiKey_NumLock;
@@ -315,6 +316,23 @@ ImGuiKey ImGui_ImplSDL2_KeyEventToImGuiKey(SDL_Keycode keycode, SDL_Scancode sca
315316
case SDLK_AC_FORWARD: return ImGuiKey_AppForward;
316317
default: break;
317318
}
319+
320+
// Fallback to scancode
321+
switch (scancode)
322+
{
323+
case SDL_SCANCODE_GRAVE: return ImGuiKey_GraveAccent;
324+
case SDL_SCANCODE_MINUS: return ImGuiKey_Minus;
325+
case SDL_SCANCODE_EQUALS: return ImGuiKey_Equal;
326+
case SDL_SCANCODE_LEFTBRACKET: return ImGuiKey_LeftBracket;
327+
case SDL_SCANCODE_RIGHTBRACKET: return ImGuiKey_RightBracket;
328+
case SDL_SCANCODE_NONUSBACKSLASH: return ImGuiKey_Oem102;
329+
case SDL_SCANCODE_BACKSLASH: return ImGuiKey_Backslash;
330+
case SDL_SCANCODE_SEMICOLON: return ImGuiKey_Semicolon;
331+
case SDL_SCANCODE_APOSTROPHE: return ImGuiKey_Apostrophe;
332+
case SDL_SCANCODE_COMMA: return ImGuiKey_Comma;
333+
case SDL_SCANCODE_PERIOD: return ImGuiKey_Period;
334+
case SDL_SCANCODE_SLASH: return ImGuiKey_Slash;
335+
}
318336
return ImGuiKey_None;
319337
}
320338

backends/imgui_impl_sdl3.cpp

+26-8
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020

2121
// CHANGELOG
2222
// (minor and older changes stripped away, please see git history for details)
23+
// 2025-03-10: When dealing with OEM keys, use scancodes instead of translated keycodes to choose ImGuiKey values. (#7136, #7201, #7206, #7306, #7670, #7672, #8468)
2324
// 2025-02-26: Only start SDL_CaptureMouse() when mouse is being dragged, to mitigate issues with e.g.Linux debuggers not claiming capture back. (#6410, #3650)
2425
// 2025-02-24: Avoid calling SDL_GetGlobalMouseState() when mouse is in relative mode.
2526
// 2025-02-18: Added ImGuiMouseCursor_Wait and ImGuiMouseCursor_Progress mouse cursor support.
@@ -207,17 +208,17 @@ ImGuiKey ImGui_ImplSDL3_KeyEventToImGuiKey(SDL_Keycode keycode, SDL_Scancode sca
207208
case SDLK_SPACE: return ImGuiKey_Space;
208209
case SDLK_RETURN: return ImGuiKey_Enter;
209210
case SDLK_ESCAPE: return ImGuiKey_Escape;
210-
case SDLK_APOSTROPHE: return ImGuiKey_Apostrophe;
211+
//case SDLK_APOSTROPHE: return ImGuiKey_Apostrophe;
211212
case SDLK_COMMA: return ImGuiKey_Comma;
212-
case SDLK_MINUS: return ImGuiKey_Minus;
213+
//case SDLK_MINUS: return ImGuiKey_Minus;
213214
case SDLK_PERIOD: return ImGuiKey_Period;
214-
case SDLK_SLASH: return ImGuiKey_Slash;
215+
//case SDLK_SLASH: return ImGuiKey_Slash;
215216
case SDLK_SEMICOLON: return ImGuiKey_Semicolon;
216-
case SDLK_EQUALS: return ImGuiKey_Equal;
217-
case SDLK_LEFTBRACKET: return ImGuiKey_LeftBracket;
218-
case SDLK_BACKSLASH: return ImGuiKey_Backslash;
219-
case SDLK_RIGHTBRACKET: return ImGuiKey_RightBracket;
220-
case SDLK_GRAVE: return ImGuiKey_GraveAccent;
217+
//case SDLK_EQUALS: return ImGuiKey_Equal;
218+
//case SDLK_LEFTBRACKET: return ImGuiKey_LeftBracket;
219+
//case SDLK_BACKSLASH: return ImGuiKey_Backslash;
220+
//case SDLK_RIGHTBRACKET: return ImGuiKey_RightBracket;
221+
//case SDLK_GRAVE: return ImGuiKey_GraveAccent;
221222
case SDLK_CAPSLOCK: return ImGuiKey_CapsLock;
222223
case SDLK_SCROLLLOCK: return ImGuiKey_ScrollLock;
223224
case SDLK_NUMLOCKCLEAR: return ImGuiKey_NumLock;
@@ -296,6 +297,23 @@ ImGuiKey ImGui_ImplSDL3_KeyEventToImGuiKey(SDL_Keycode keycode, SDL_Scancode sca
296297
case SDLK_AC_FORWARD: return ImGuiKey_AppForward;
297298
default: break;
298299
}
300+
301+
// Fallback to scancode
302+
switch (scancode)
303+
{
304+
case SDL_SCANCODE_GRAVE: return ImGuiKey_GraveAccent;
305+
case SDL_SCANCODE_MINUS: return ImGuiKey_Minus;
306+
case SDL_SCANCODE_EQUALS: return ImGuiKey_Equal;
307+
case SDL_SCANCODE_LEFTBRACKET: return ImGuiKey_LeftBracket;
308+
case SDL_SCANCODE_RIGHTBRACKET: return ImGuiKey_RightBracket;
309+
case SDL_SCANCODE_NONUSBACKSLASH: return ImGuiKey_Oem102;
310+
case SDL_SCANCODE_BACKSLASH: return ImGuiKey_Backslash;
311+
case SDL_SCANCODE_SEMICOLON: return ImGuiKey_Semicolon;
312+
case SDL_SCANCODE_APOSTROPHE: return ImGuiKey_Apostrophe;
313+
case SDL_SCANCODE_COMMA: return ImGuiKey_Comma;
314+
case SDL_SCANCODE_PERIOD: return ImGuiKey_Period;
315+
case SDL_SCANCODE_SLASH: return ImGuiKey_Slash;
316+
}
299317
return ImGuiKey_None;
300318
}
301319

backends/imgui_impl_win32.cpp

+33-10
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121

2222
// CHANGELOG
2323
// (minor and older changes stripped away, please see git history for details)
24+
// 2025-03-10: When dealing with OEM keys, use scancodes instead of translated keycodes to choose ImGuiKey values. (#7136, #7201, #7206, #7306, #7670, #7672, #8468)
2425
// 2025-02-18: Added ImGuiMouseCursor_Wait and ImGuiMouseCursor_Progress mouse cursor support.
2526
// 2024-07-08: Inputs: Fixed ImGuiMod_Super being mapped to VK_APPS instead of VK_LWIN||VK_RWIN. (#7768)
2627
// 2023-10-05: Inputs: Added support for extra ImGuiKey values: F13 to F24 function keys, app back/forward keys.
@@ -431,6 +432,8 @@ ImGuiKey ImGui_ImplWin32_KeyEventToImGuiKey(WPARAM wParam, LPARAM lParam)
431432
if ((wParam == VK_RETURN) && (HIWORD(lParam) & KF_EXTENDED))
432433
return ImGuiKey_KeypadEnter;
433434

435+
const int scancode = (int)LOBYTE(HIWORD(lParam));
436+
//IMGUI_DEBUG_LOG("scancode %3d, keycode = 0x%02X\n", scancode, wParam);
434437
switch (wParam)
435438
{
436439
case VK_TAB: return ImGuiKey_Tab;
@@ -448,17 +451,17 @@ ImGuiKey ImGui_ImplWin32_KeyEventToImGuiKey(WPARAM wParam, LPARAM lParam)
448451
case VK_SPACE: return ImGuiKey_Space;
449452
case VK_RETURN: return ImGuiKey_Enter;
450453
case VK_ESCAPE: return ImGuiKey_Escape;
451-
case VK_OEM_7: return ImGuiKey_Apostrophe;
454+
//case VK_OEM_7: return ImGuiKey_Apostrophe;
452455
case VK_OEM_COMMA: return ImGuiKey_Comma;
453-
case VK_OEM_MINUS: return ImGuiKey_Minus;
456+
//case VK_OEM_MINUS: return ImGuiKey_Minus;
454457
case VK_OEM_PERIOD: return ImGuiKey_Period;
455-
case VK_OEM_2: return ImGuiKey_Slash;
456-
case VK_OEM_1: return ImGuiKey_Semicolon;
457-
case VK_OEM_PLUS: return ImGuiKey_Equal;
458-
case VK_OEM_4: return ImGuiKey_LeftBracket;
459-
case VK_OEM_5: return ImGuiKey_Backslash;
460-
case VK_OEM_6: return ImGuiKey_RightBracket;
461-
case VK_OEM_3: return ImGuiKey_GraveAccent;
458+
//case VK_OEM_2: return ImGuiKey_Slash;
459+
//case VK_OEM_1: return ImGuiKey_Semicolon;
460+
//case VK_OEM_PLUS: return ImGuiKey_Equal;
461+
//case VK_OEM_4: return ImGuiKey_LeftBracket;
462+
//case VK_OEM_5: return ImGuiKey_Backslash;
463+
//case VK_OEM_6: return ImGuiKey_RightBracket;
464+
//case VK_OEM_3: return ImGuiKey_GraveAccent;
462465
case VK_CAPITAL: return ImGuiKey_CapsLock;
463466
case VK_SCROLL: return ImGuiKey_ScrollLock;
464467
case VK_NUMLOCK: return ImGuiKey_NumLock;
@@ -550,8 +553,28 @@ ImGuiKey ImGui_ImplWin32_KeyEventToImGuiKey(WPARAM wParam, LPARAM lParam)
550553
case VK_F24: return ImGuiKey_F24;
551554
case VK_BROWSER_BACK: return ImGuiKey_AppBack;
552555
case VK_BROWSER_FORWARD: return ImGuiKey_AppForward;
553-
default: return ImGuiKey_None;
556+
default: break;
554557
}
558+
559+
// Fallback to scancode
560+
// https://handmade.network/forums/t/2011-keyboard_inputs_-_scancodes,_raw_input,_text_input,_key_names
561+
switch (scancode)
562+
{
563+
case 41: return ImGuiKey_GraveAccent; // VK_OEM_8 in EN-UK, VK_OEM_3 in EN-US, VK_OEM_7 in FR, VK_OEM_5 in DE, etc.
564+
case 12: return ImGuiKey_Minus;
565+
case 13: return ImGuiKey_Equal;
566+
case 26: return ImGuiKey_LeftBracket;
567+
case 27: return ImGuiKey_RightBracket;
568+
case 86: return ImGuiKey_Oem102;
569+
case 43: return ImGuiKey_Backslash;
570+
case 39: return ImGuiKey_Semicolon;
571+
case 40: return ImGuiKey_Apostrophe;
572+
case 51: return ImGuiKey_Comma;
573+
case 52: return ImGuiKey_Period;
574+
case 53: return ImGuiKey_Slash;
575+
}
576+
577+
return ImGuiKey_None;
555578
}
556579

557580
// Allow compilation with old Windows SDK. MinGW doesn't have default _WIN32_WINNT/WINVER versions.

docs/CHANGELOG.txt

+12
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,18 @@ Other changes:
121121
by showing the filter inside the combo contents. (#718)
122122
- Examples: SDL3: Added comments to clarify setup for users of the unfortunate
123123
SDL_MAIN_USE_CALLBACKS feature. (#8455)
124+
- IO: Added ImGuiKey_Oem102 to ImGuiKey enum. (#7136, #7201, #7206, #7306, #8468)
125+
- Backends: reworked key handlers to use/prioritize untranslated scancodes instead of
126+
translated keycodes when dealing with OEM keys. (#7136, #7201, #7206, #7306, #7670, #7672, #8468)
127+
Not that only ImGuiKey value (NOT the characters used for text input) are affected.
128+
- Win32, SDL2, SDL3: Use scancodes for OEM keys.
129+
- GLFW: GLFW_KEY_WORLD_1 and GLFW_KEY_WORLD_2 are emitting ImGuiKey_Oem102.
130+
- Fixes many cases of keys not emitting a ImGuiKey value with certain keyboad layouts.
131+
- Makes emitted ImGuiKey values more consistent regardless of keyboard mapping.
132+
- OEM keys are: ImGuiKey_Apostrophe, ImGuiKey_Comma, ImGuiKey_Minus, ImGuiKey_Period,
133+
ImGuiKey_Slash, ImGuiKey_Semicolon, ImGuiKey_Equal, ImGuiKey_LeftBracket,
134+
ImGuiKey_RightBracket, ImGuiKey_Backslash, ImGuiKey_GraveAccent, which are difficult
135+
to find a reliable translated mapping on all keyboard layouts.
124136
- Backends: GLFW: Fixed clipboard handler assertion when using GLFW <= 3.2.1 compiled
125137
with asserts enabled. (#8452)
126138
- Backends: SDL2, SDL3: Using SDL_OpenURL() in platform_io.Platform_OpenInShellFn

imgui.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -8772,7 +8772,7 @@ ImGuiKeyData* ImGui::GetKeyData(ImGuiContext* ctx, ImGuiKey key)
87728772
return &g.IO.KeysData[key - ImGuiKey_NamedKey_BEGIN];
87738773
}
87748774

8775-
// Those names a provided for debugging purpose and are not meant to be saved persistently not compared.
8775+
// Those names are provided for debugging purpose and are not meant to be saved persistently nor compared.
87768776
static const char* const GKeyNames[] =
87778777
{
87788778
"Tab", "LeftArrow", "RightArrow", "UpArrow", "DownArrow", "PageUp", "PageDown",
@@ -8787,7 +8787,7 @@ static const char* const GKeyNames[] =
87878787
"Pause", "Keypad0", "Keypad1", "Keypad2", "Keypad3", "Keypad4", "Keypad5", "Keypad6",
87888788
"Keypad7", "Keypad8", "Keypad9", "KeypadDecimal", "KeypadDivide", "KeypadMultiply",
87898789
"KeypadSubtract", "KeypadAdd", "KeypadEnter", "KeypadEqual",
8790-
"AppBack", "AppForward",
8790+
"AppBack", "AppForward", "Oem102",
87918791
"GamepadStart", "GamepadBack",
87928792
"GamepadFaceLeft", "GamepadFaceRight", "GamepadFaceUp", "GamepadFaceDown",
87938793
"GamepadDpadLeft", "GamepadDpadRight", "GamepadDpadUp", "GamepadDpadDown",

imgui.h

+3-2
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@
2929
// Library Version
3030
// (Integer encoded as XYYZZ for use in #if preprocessor conditionals, e.g. '#if IMGUI_VERSION_NUM >= 12345')
3131
#define IMGUI_VERSION "1.91.9 WIP"
32-
#define IMGUI_VERSION_NUM 19187
32+
#define IMGUI_VERSION_NUM 19188
3333
#define IMGUI_HAS_TABLE
3434

3535
/*
@@ -975,7 +975,7 @@ namespace ImGui
975975
IMGUI_API bool IsKeyReleased(ImGuiKey key); // was key released (went from Down to !Down)?
976976
IMGUI_API bool IsKeyChordPressed(ImGuiKeyChord key_chord); // was key chord (mods + key) pressed, e.g. you can pass 'ImGuiMod_Ctrl | ImGuiKey_S' as a key-chord. This doesn't do any routing or focus check, please consider using Shortcut() function instead.
977977
IMGUI_API int GetKeyPressedAmount(ImGuiKey key, float repeat_delay, float rate); // uses provided repeat rate/delay. return a count, most often 0 or 1 but might be >1 if RepeatRate is small enough that DeltaTime > RepeatRate
978-
IMGUI_API const char* GetKeyName(ImGuiKey key); // [DEBUG] returns English name of the key. Those names a provided for debugging purpose and are not meant to be saved persistently not compared.
978+
IMGUI_API const char* GetKeyName(ImGuiKey key); // [DEBUG] returns English name of the key. Those names are provided for debugging purpose and are not meant to be saved persistently nor compared.
979979
IMGUI_API void SetNextFrameWantCaptureKeyboard(bool want_capture_keyboard); // Override io.WantCaptureKeyboard flag next frame (said flag is left for your application to handle, typically when true it instructs your app to ignore inputs). e.g. force capture keyboard when your widget is being hovered. This is equivalent to setting "io.WantCaptureKeyboard = want_capture_keyboard"; after the next NewFrame() call.
980980

981981
// Inputs Utilities: Shortcut Testing & Routing [BETA]
@@ -1493,6 +1493,7 @@ enum ImGuiKey : int
14931493
ImGuiKey_KeypadEqual,
14941494
ImGuiKey_AppBack, // Available on some keyboard/mouses. Often referred as "Browser Back"
14951495
ImGuiKey_AppForward,
1496+
ImGuiKey_Oem102, // Non-US backslash.
14961497

14971498
// Gamepad (some of those are analog values, 0.0f to 1.0f) // NAVIGATION ACTION
14981499
// (download controller mapping PNG/PSD at http://dearimgui.com/controls_sheets)

0 commit comments

Comments
 (0)