From 0e81251f3a768f51cf1c3a7b3c78d08a9c0547d8 Mon Sep 17 00:00:00 2001 From: Joe Wilm Date: Mon, 22 Jan 2018 09:36:26 -0800 Subject: [PATCH] Implement hidpi factor for x11 MonitorId (#387) This was previously hardcoded to 1.0. The values for physical size in millimeters and pixel counts on each axis are used to compute the dpi per monitor. --- src/platform/linux/x11/monitor.rs | 28 +++++++++++++++++++++++++++- 1 file changed, 27 insertions(+), 1 deletion(-) diff --git a/src/platform/linux/x11/monitor.rs b/src/platform/linux/x11/monitor.rs index 66a81bfd..c8db06b0 100644 --- a/src/platform/linux/x11/monitor.rs +++ b/src/platform/linux/x11/monitor.rs @@ -15,6 +15,8 @@ pub struct MonitorId { position: (i32, i32), /// If the monitor is the primary one primary: bool, + /// The DPI scaling factor + hidpi_factor: f32, } pub fn get_available_monitors(x: &Arc) -> Vec { @@ -35,10 +37,21 @@ pub fn get_available_monitors(x: &Arc) -> Vec { let output = (x.xrandr.XRRGetOutputInfo)(x.display, resources, *(monitor.outputs.offset(0))); let nameslice = slice::from_raw_parts((*output).name as *mut u8, (*output).nameLen as usize); let name = String::from_utf8_lossy(nameslice).into_owned(); + let hidpi_factor = { + let x_mm = (*output).mm_width as f32; + let y_mm = (*output).mm_height as f32; + let x_px = monitor.width as f32; + let y_px = monitor.height as f32; + let ppmm = ((x_px * y_px) / (x_mm * y_mm)).sqrt(); + + // Quantize 1/12 step size + ((ppmm * (12.0 * 25.4 / 96.0)).round() / 12.0).max(1.0) + }; (x.xrandr.XRRFreeOutputInfo)(output); available.push(MonitorId{ id: i as u32, name, + hidpi_factor, dimensions: (monitor.width as u32, monitor.height as u32), position: (monitor.x as i32, monitor.y as i32), primary: (monitor.primary != 0), @@ -56,10 +69,23 @@ pub fn get_available_monitors(x: &Arc) -> Vec { let output = (x.xrandr.XRRGetOutputInfo)(x.display, resources, *((*crtc).outputs.offset(0))); let nameslice = slice::from_raw_parts((*output).name as *mut u8, (*output).nameLen as usize); let name = String::from_utf8_lossy(nameslice).into_owned(); + + let hidpi_factor = { + let x_mm = (*output).mm_width as f32; + let y_mm = (*output).mm_height as f32; + let x_px = (*crtc).width as f32; + let y_px = (*crtc).height as f32; + let ppmm = ((x_px * y_px) / (x_mm * y_mm)).sqrt(); + + // Quantize 1/12 step size + ((ppmm * (12.0 * 25.4 / 96.0)).round() / 12.0).max(1.0) + }; + (x.xrandr.XRRFreeOutputInfo)(output); available.push(MonitorId{ id: crtcid as u32, name, + hidpi_factor, dimensions: ((*crtc).width as u32, (*crtc).height as u32), position: ((*crtc).x as i32, (*crtc).y as i32), primary: true, @@ -101,6 +127,6 @@ impl MonitorId { #[inline] pub fn get_hidpi_factor(&self) -> f32 { - 1.0 + self.hidpi_factor } }