X11: Fix misreporting DPI factor at startup (#1252)

* X11: Fix misreporting DPI factor at startup

* Add a CHANGELOG entry
This commit is contained in:
Murarth 2019-11-10 13:55:29 -07:00 committed by Hal Gentz
parent 1ed15c7ec7
commit 9828f368d6
4 changed files with 46 additions and 72 deletions

View file

@ -1,6 +1,7 @@
# Unreleased # Unreleased
- On macOS, fix application termination on `ControlFlow::Exit` - On macOS, fix application termination on `ControlFlow::Exit`
- On X11, fix misreporting DPI factor at startup.
- On X11, fix events not being reported when using `run_return`. - On X11, fix events not being reported when using `run_return`.
- On X11, fix key modifiers being incorrectly reported. - On X11, fix key modifiers being incorrectly reported.

View file

@ -382,14 +382,8 @@ impl<T: 'static> EventProcessor<T> {
let (width, height) = shared_state_lock let (width, height) = shared_state_lock
.dpi_adjusted .dpi_adjusted
.unwrap_or_else(|| (xev.width as f64, xev.height as f64)); .unwrap_or_else(|| (xev.width as f64, xev.height as f64));
let last_hidpi_factor =
shared_state_lock.guessed_dpi.take().unwrap_or_else(|| { let last_hidpi_factor = shared_state_lock.last_monitor.hidpi_factor;
shared_state_lock
.last_monitor
.as_ref()
.map(|last_monitor| last_monitor.hidpi_factor)
.unwrap_or(1.0)
});
let new_hidpi_factor = { let new_hidpi_factor = {
let window_rect = util::AaRect::new(new_outer_position, new_inner_size); let window_rect = util::AaRect::new(new_outer_position, new_inner_size);
monitor = wt.xconn.get_monitor_for_window(Some(window_rect)); monitor = wt.xconn.get_monitor_for_window(Some(window_rect));
@ -397,7 +391,7 @@ impl<T: 'static> EventProcessor<T> {
// Avoid caching an invalid dummy monitor handle // Avoid caching an invalid dummy monitor handle
if monitor.id != 0 { if monitor.id != 0 {
shared_state_lock.last_monitor = Some(monitor.clone()); shared_state_lock.last_monitor = monitor.clone();
} }
new_hidpi_factor new_hidpi_factor
}; };

View file

@ -130,7 +130,7 @@ impl MonitorHandle {
}) })
} }
fn dummy() -> Self { pub fn dummy() -> Self {
MonitorHandle { MonitorHandle {
id: 0, id: 0,
name: "<dummy monitor>".into(), name: "<dummy monitor>".into(),

View file

@ -38,15 +38,14 @@ unsafe extern "C" fn visibility_predicate(
(event.window == window && event.type_ == ffi::VisibilityNotify) as _ (event.window == window && event.type_ == ffi::VisibilityNotify) as _
} }
#[derive(Debug, Default)] #[derive(Debug)]
pub struct SharedState { pub struct SharedState {
pub cursor_pos: Option<(f64, f64)>, pub cursor_pos: Option<(f64, f64)>,
pub size: Option<(u32, u32)>, pub size: Option<(u32, u32)>,
pub position: Option<(i32, i32)>, pub position: Option<(i32, i32)>,
pub inner_position: Option<(i32, i32)>, pub inner_position: Option<(i32, i32)>,
pub inner_position_rel_parent: Option<(i32, i32)>, pub inner_position_rel_parent: Option<(i32, i32)>,
pub guessed_dpi: Option<f64>, pub last_monitor: X11MonitorHandle,
pub last_monitor: Option<X11MonitorHandle>,
pub dpi_adjusted: Option<(f64, f64)>, pub dpi_adjusted: Option<(f64, f64)>,
pub fullscreen: Option<Fullscreen>, pub fullscreen: Option<Fullscreen>,
// Used to restore position after exiting fullscreen // Used to restore position after exiting fullscreen
@ -60,11 +59,24 @@ pub struct SharedState {
} }
impl SharedState { impl SharedState {
fn new(dpi_factor: f64, is_visible: bool) -> Mutex<Self> { fn new(last_monitor: X11MonitorHandle, is_visible: bool) -> Mutex<Self> {
let mut shared_state = SharedState::default(); Mutex::new(SharedState {
shared_state.guessed_dpi = Some(dpi_factor); last_monitor,
shared_state.is_visible = is_visible; is_visible,
Mutex::new(shared_state)
cursor_pos: None,
size: None,
position: None,
inner_position: None,
inner_position_rel_parent: None,
dpi_adjusted: None,
fullscreen: None,
restore_position: None,
desktop_video_mode: None,
frame_extents: None,
min_inner_size: None,
max_inner_size: None,
})
} }
} }
@ -93,34 +105,27 @@ impl UnownedWindow {
let xconn = &event_loop.xconn; let xconn = &event_loop.xconn;
let root = event_loop.root; let root = event_loop.root;
let monitors = xconn.available_monitors(); let mut monitors = xconn.available_monitors();
let dpi_factor = if !monitors.is_empty() { let guessed_monitor = if monitors.is_empty() {
let mut dpi_factor = Some(monitors[0].hidpi_factor()); X11MonitorHandle::dummy()
for monitor in &monitors {
if Some(monitor.hidpi_factor()) != dpi_factor {
dpi_factor = None;
}
}
dpi_factor.unwrap_or_else(|| {
xconn
.query_pointer(root, util::VIRTUAL_CORE_POINTER)
.ok()
.and_then(|pointer_state| {
let (x, y) = (pointer_state.root_x as i64, pointer_state.root_y as i64);
let mut dpi_factor = None;
for monitor in &monitors {
if monitor.rect.contains_point(x, y) {
dpi_factor = Some(monitor.hidpi_factor());
break;
}
}
dpi_factor
})
.unwrap_or(1.0)
})
} else { } else {
1.0 xconn
.query_pointer(root, util::VIRTUAL_CORE_POINTER)
.ok()
.and_then(|pointer_state| {
let (x, y) = (pointer_state.root_x as i64, pointer_state.root_y as i64);
for i in 0..monitors.len() {
if monitors[i].rect.contains_point(x, y) {
return Some(monitors.swap_remove(i));
}
}
None
})
.unwrap_or_else(|| monitors.swap_remove(0))
}; };
let dpi_factor = guessed_monitor.hidpi_factor();
info!("Guessed window DPI factor: {}", dpi_factor); info!("Guessed window DPI factor: {}", dpi_factor);
@ -232,7 +237,7 @@ impl UnownedWindow {
cursor_grabbed: Mutex::new(false), cursor_grabbed: Mutex::new(false),
cursor_visible: Mutex::new(true), cursor_visible: Mutex::new(true),
ime_sender: Mutex::new(event_loop.ime_sender.clone()), ime_sender: Mutex::new(event_loop.ime_sender.clone()),
shared_state: SharedState::new(dpi_factor, window_attrs.visible), shared_state: SharedState::new(guessed_monitor, window_attrs.visible),
pending_redraws: event_loop.pending_redraws.clone(), pending_redraws: event_loop.pending_redraws.clone(),
}; };
@ -698,24 +703,9 @@ impl UnownedWindow {
} }
} }
fn get_rect(&self) -> util::AaRect {
// TODO: This might round-trip more times than needed.
let position = self.outer_position_physical();
let size = self.outer_size_physical();
util::AaRect::new(position, size)
}
#[inline] #[inline]
pub fn current_monitor(&self) -> X11MonitorHandle { pub fn current_monitor(&self) -> X11MonitorHandle {
let monitor = self.shared_state.lock().last_monitor.as_ref().cloned(); self.shared_state.lock().last_monitor.clone()
monitor.unwrap_or_else(|| {
let monitor = self.xconn.get_monitor_for_window(Some(self.get_rect()));
// Avoid caching an invalid dummy monitor handle
if monitor.id != 0 {
self.shared_state.lock().last_monitor = Some(monitor.clone());
}
monitor
})
} }
pub fn available_monitors(&self) -> Vec<X11MonitorHandle> { pub fn available_monitors(&self) -> Vec<X11MonitorHandle> {
@ -982,17 +972,6 @@ impl UnownedWindow {
self.logicalize_size(self.inner_size_physical()) self.logicalize_size(self.inner_size_physical())
} }
pub(crate) fn outer_size_physical(&self) -> (u32, u32) {
let extents = self.shared_state.lock().frame_extents.clone();
if let Some(extents) = extents {
let (w, h) = self.inner_size_physical();
extents.inner_size_to_outer(w, h)
} else {
self.update_cached_frame_extents();
self.outer_size_physical()
}
}
#[inline] #[inline]
pub fn outer_size(&self) -> LogicalSize { pub fn outer_size(&self) -> LogicalSize {
let extents = self.shared_state.lock().frame_extents.clone(); let extents = self.shared_state.lock().frame_extents.clone();