diff --git a/src/os/unix.rs b/src/os/unix.rs index 9d897a76..bcf2dcd1 100644 --- a/src/os/unix.rs +++ b/src/os/unix.rs @@ -42,6 +42,8 @@ pub trait WindowExt { fn get_xlib_screen_id(&self) -> Option<*mut libc::c_void>; fn get_xlib_xconnection(&self) -> Option>; + + fn send_xim_spot(&self, x: i16, y: i16); /// This function returns the underlying `xcb_connection_t` of an xlib `Display`. /// @@ -121,6 +123,12 @@ impl WindowExt for Window { } } + fn send_xim_spot(&self, x: i16, y: i16) { + if let LinuxWindow::X(ref w) = self.window { + w.send_xim_spot(x, y); + } + } + #[inline] fn get_wayland_surface(&self) -> Option<*mut libc::c_void> { use wayland_client::Proxy; diff --git a/src/platform/linux/x11/mod.rs b/src/platform/linux/x11/mod.rs index c046ac25..398a7cfd 100644 --- a/src/platform/linux/x11/mod.rs +++ b/src/platform/linux/x11/mod.rs @@ -685,6 +685,7 @@ impl Window2 { x_events_loop.windows.lock().unwrap().insert(win.id(), WindowData { im: im, ic: ic, + spot: ffi::XPoint {x: 0, y: 0}, config: None, multitouch: window.multitouch, cursor_pos: None, @@ -701,6 +702,26 @@ impl Window2 { pub fn id(&self) -> WindowId { self.window.id() } + + #[inline] + pub fn send_xim_spot(&self, x: i16, y: i16) { + if let (Some(windows), Some(display)) = (self.windows.upgrade(), self.display.upgrade()) { + let nspot = ffi::XPoint{x: x, y: y}; + let mut windows = windows.lock().unwrap(); + let mut w = windows.get_mut(&self.window.id()).unwrap(); + if w.spot.x == x && w.spot.y == y { + return + } + w.spot = nspot; + unsafe { + let preedit_attr = (display.xlib.XVaCreateNestedList) + (0, b"spotLocation\0", &nspot, ptr::null::<()>()); + (display.xlib.XSetICValues)(w.ic, b"preeditAttributes\0", + preedit_attr, ptr::null::<()>()); + (display.xlib.XFree)(preedit_attr); + } + } + } } impl Drop for Window2 { @@ -722,6 +743,7 @@ struct WindowData { config: Option, im: ffi::XIM, ic: ffi::XIC, + spot: ffi::XPoint, multitouch: bool, cursor_pos: Option<(f64, f64)>, }