From 3ef59028155ba4cab0f734aecb6bdfc2aa90cbf4 Mon Sep 17 00:00:00 2001 From: yw05 <37980625+yw05@users.noreply.github.com> Date: Tue, 30 Mar 2021 13:52:45 +0200 Subject: [PATCH] Add support for IMEs on Linux --- source/Irrlicht/CIrrDeviceLinux.cpp | 49 ++++++++++++++++++++++++----- 1 file changed, 42 insertions(+), 7 deletions(-) diff --git a/source/Irrlicht/CIrrDeviceLinux.cpp b/source/Irrlicht/CIrrDeviceLinux.cpp index 7f43c649..192dd39c 100644 --- a/source/Irrlicht/CIrrDeviceLinux.cpp +++ b/source/Irrlicht/CIrrDeviceLinux.cpp @@ -13,6 +13,7 @@ #include #include "IEventReceiver.h" #include "ISceneManager.h" +#include "IGUIElement.h" #include "IGUIEnvironment.h" #include "os.h" #include "CTimer.h" @@ -705,6 +706,14 @@ bool CIrrDeviceLinux::createInputContext() return false; } + // Load XMODIFIERS (e.g. for IMEs) + if (!XSetLocaleModifiers("")) + { + setlocale(LC_CTYPE, oldLocale.c_str()); + os::Printer::log("XSetLocaleModifiers failed. Falling back to non-i18n input.", ELL_WARNING); + return false; + } + XInputMethod = XOpenIM(XDisplay, NULL, NULL, NULL); if ( !XInputMethod ) { @@ -820,7 +829,7 @@ bool CIrrDeviceLinux::run() { XEvent event; XNextEvent(XDisplay, &event); - if (XFilterEvent(&event, XWindow)) + if (acceptsIME() && XFilterEvent(&event, None)) continue; switch (event.type) @@ -998,18 +1007,29 @@ bool CIrrDeviceLinux::run() SKeyMap mp; if ( XInputContext ) { - wchar_t buf[8]={0}; + wchar_t buf[64]={0}; Status status; - int strLen = XwcLookupString(XInputContext, &event.xkey, buf, sizeof(buf), &mp.X11Key, &status); + int strLen = XwcLookupString(XInputContext, &event.xkey, buf, sizeof(buf)/sizeof(wchar_t)-1, &mp.X11Key, &status); if ( status == XBufferOverflow ) { os::Printer::log("XwcLookupString needs a larger buffer", ELL_INFORMATION); } if ( strLen > 0 && (status == XLookupChars || status == XLookupBoth) ) { - if ( strLen > 1 ) - os::Printer::log("Additional returned characters dropped", ELL_INFORMATION); - irrevent.KeyInput.Char = buf[0]; + if (strLen > 1) + { + // Multiple characters: send string event + irrevent.EventType = irr::EET_STRING_INPUT_EVENT; + irrevent.StringInput.Str = new core::stringw(buf); + postEventFromUser(irrevent); + delete irrevent.StringInput.Str; + irrevent.StringInput.Str = NULL; + break; + } + else + { + irrevent.KeyInput.Char = buf[0]; + } } else { @@ -1044,7 +1064,6 @@ bool CIrrDeviceLinux::run() irrevent.KeyInput.Control = (event.xkey.state & ControlMask) != 0; irrevent.KeyInput.Shift = (event.xkey.state & ShiftMask) != 0; irrevent.KeyInput.Key = getKeyCode(event); - postEventFromUser(irrevent); } break; @@ -1138,6 +1157,22 @@ bool CIrrDeviceLinux::run() } // end switch } // end while + + // Update IME information + if (GUIEnvironment) + { + gui::IGUIElement *elem = GUIEnvironment->getFocus(); + if (elem && elem->acceptsIME()) + { + core::rect r = elem->getAbsolutePosition(); + XPoint p; + p.x = (short)r.UpperLeftCorner.X; + p.y = (short)r.LowerRightCorner.Y; + XVaNestedList l = XVaCreateNestedList(0, XNSpotLocation, &p, NULL); + XSetICValues(XInputContext, XNPreeditAttributes, l, NULL); + XFree(l); + } + } } #endif //_IRR_COMPILE_WITH_X11_