On Wayland, fix coordinates in mouse events when scale factor isn't 1 (#1385)

* On Wayland, fix coordinates in mouse events when scale factor isn't 1

* Refactor mouse_focus to be a surface instead of WindowId
This commit is contained in:
Kirill Chibisov 2020-01-11 11:45:52 +03:00 committed by Freya Gentz
parent 1ddceeb063
commit a6d180cefb
2 changed files with 21 additions and 8 deletions

View file

@ -6,6 +6,7 @@
- On X11, fix deadlock on window state when handling certain window events. - On X11, fix deadlock on window state when handling certain window events.
- `WindowBuilder` now implements `Default`. - `WindowBuilder` now implements `Default`.
- **Breaking:** `WindowEvent::CursorMoved` changed to `f64` units, preserving high-precision data supplied by most backends - **Breaking:** `WindowEvent::CursorMoved` changed to `f64` units, preserving high-precision data supplied by most backends
- On Wayland, fix coordinates in mouse events when scale factor isn't 1
# 0.20.0 (2020-01-05) # 0.20.0 (2020-01-05)

View file

@ -7,10 +7,13 @@ use crate::event::{
use super::{ use super::{
event_loop::{CursorManager, EventsSink}, event_loop::{CursorManager, EventsSink},
make_wid,
window::WindowStore, window::WindowStore,
DeviceId, DeviceId,
}; };
use smithay_client_toolkit::surface;
use smithay_client_toolkit::reexports::client::protocol::{ use smithay_client_toolkit::reexports::client::protocol::{
wl_pointer::{self, Event as PtrEvent, WlPointer}, wl_pointer::{self, Event as PtrEvent, WlPointer},
wl_seat, wl_seat,
@ -36,6 +39,7 @@ pub fn implement_pointer(
cursor_manager: Arc<Mutex<CursorManager>>, cursor_manager: Arc<Mutex<CursorManager>>,
) -> WlPointer { ) -> WlPointer {
seat.get_pointer(|pointer| { seat.get_pointer(|pointer| {
// Currently focused winit surface
let mut mouse_focus = None; let mut mouse_focus = None;
let mut axis_buffer = None; let mut axis_buffer = None;
let mut axis_discrete_buffer = None; let mut axis_discrete_buffer = None;
@ -53,8 +57,10 @@ pub fn implement_pointer(
.. ..
} => { } => {
let wid = store.find_wid(&surface); let wid = store.find_wid(&surface);
if let Some(wid) = wid { if let Some(wid) = wid {
mouse_focus = Some(wid); let scale_factor = surface::get_dpi_factor(&surface) as f64;
mouse_focus = Some(surface);
// Reload cursor style only when we enter winit's surface. Calling // Reload cursor style only when we enter winit's surface. Calling
// this function every time on `PtrEvent::Enter` could interfere with // this function every time on `PtrEvent::Enter` could interfere with
@ -75,7 +81,8 @@ pub fn implement_pointer(
device_id: crate::event::DeviceId( device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId), crate::platform_impl::DeviceId::Wayland(DeviceId),
), ),
position: (surface_x, surface_y).into(), position: (surface_x * scale_factor, surface_y * scale_factor)
.into(),
modifiers: modifiers_tracker.lock().unwrap().clone(), modifiers: modifiers_tracker.lock().unwrap().clone(),
}, },
wid, wid,
@ -101,13 +108,16 @@ pub fn implement_pointer(
surface_y, surface_y,
.. ..
} => { } => {
if let Some(wid) = mouse_focus { if let Some(surface) = mouse_focus.as_ref() {
let scale_factor = surface::get_dpi_factor(&surface) as f64;
let wid = make_wid(surface);
sink.send_window_event( sink.send_window_event(
WindowEvent::CursorMoved { WindowEvent::CursorMoved {
device_id: crate::event::DeviceId( device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId), crate::platform_impl::DeviceId::Wayland(DeviceId),
), ),
position: (surface_x, surface_y).into(), position: (surface_x * scale_factor, surface_y * scale_factor)
.into(),
modifiers: modifiers_tracker.lock().unwrap().clone(), modifiers: modifiers_tracker.lock().unwrap().clone(),
}, },
wid, wid,
@ -115,7 +125,7 @@ pub fn implement_pointer(
} }
} }
PtrEvent::Button { button, state, .. } => { PtrEvent::Button { button, state, .. } => {
if let Some(wid) = mouse_focus { if let Some(surface) = mouse_focus.as_ref() {
let state = match state { let state = match state {
wl_pointer::ButtonState::Pressed => ElementState::Pressed, wl_pointer::ButtonState::Pressed => ElementState::Pressed,
wl_pointer::ButtonState::Released => ElementState::Released, wl_pointer::ButtonState::Released => ElementState::Released,
@ -137,12 +147,13 @@ pub fn implement_pointer(
button, button,
modifiers: modifiers_tracker.lock().unwrap().clone(), modifiers: modifiers_tracker.lock().unwrap().clone(),
}, },
wid, make_wid(surface),
); );
} }
} }
PtrEvent::Axis { axis, value, .. } => { PtrEvent::Axis { axis, value, .. } => {
if let Some(wid) = mouse_focus { if let Some(surface) = mouse_focus.as_ref() {
let wid = make_wid(surface);
if pointer.as_ref().version() < 5 { if pointer.as_ref().version() < 5 {
let (mut x, mut y) = (0.0, 0.0); let (mut x, mut y) = (0.0, 0.0);
// old seat compatibility // old seat compatibility
@ -184,7 +195,8 @@ pub fn implement_pointer(
PtrEvent::Frame => { PtrEvent::Frame => {
let axis_buffer = axis_buffer.take(); let axis_buffer = axis_buffer.take();
let axis_discrete_buffer = axis_discrete_buffer.take(); let axis_discrete_buffer = axis_discrete_buffer.take();
if let Some(wid) = mouse_focus { if let Some(surface) = mouse_focus.as_ref() {
let wid = make_wid(surface);
if let Some((x, y)) = axis_discrete_buffer { if let Some((x, y)) = axis_discrete_buffer {
sink.send_window_event( sink.send_window_event(
WindowEvent::MouseWheel { WindowEvent::MouseWheel {