use std::collections::VecDeque; use std::sync::{Arc, Mutex, Weak}; use {CreationError, CursorState, MouseCursor, WindowAttributes, LogicalPosition, LogicalSize}; use platform::MonitorId as PlatformMonitorId; use window::MonitorId as RootMonitorId; use sctk::window::{BasicFrame, Event as WEvent, Window as SWindow}; use sctk::reexports::client::{Display, Proxy}; use sctk::reexports::client::protocol::{wl_seat, wl_surface, wl_output}; use sctk::reexports::client::protocol::wl_compositor::RequestsTrait as CompositorRequests; use sctk::reexports::client::protocol::wl_surface::RequestsTrait as SurfaceRequests; use sctk::output::OutputMgr; use super::{make_wid, EventsLoop, MonitorId, WindowId}; use platform::platform::wayland::event_loop::{get_available_monitors, get_primary_monitor}; pub struct Window { surface: Proxy, frame: Arc>>, monitors: Arc>, // Monitors this window is currently on outputs: OutputMgr, // Access to info for all monitors size: Arc>, kill_switch: (Arc>, Arc>), display: Arc, need_frame_refresh: Arc>, } impl Window { pub fn new(evlp: &EventsLoop, attributes: WindowAttributes) -> Result { // TODO: Update for new DPI API //let (width, height) = attributes.dimensions.unwrap_or((800, 600)); let (width, height) = (64, 64); // Create the window let size = Arc::new(Mutex::new((width, height))); // monitor tracking let monitor_list = Arc::new(Mutex::new(MonitorList::new())); let surface = evlp.env.compositor.create_surface().unwrap().implement({ let list = monitor_list.clone(); let omgr = evlp.env.outputs.clone(); let window_store = evlp.store.clone(); move |event, surface: Proxy| match event { wl_surface::Event::Enter { output } => { let dpi_change = list.lock().unwrap().add_output(MonitorId { proxy: output, mgr: omgr.clone(), }); if let Some(dpi) = dpi_change { if surface.version() >= 3 { // without version 3 we can't be dpi aware window_store.lock().unwrap().dpi_change(&surface, dpi); surface.set_buffer_scale(dpi); } } }, wl_surface::Event::Leave { output } => { let dpi_change = list.lock().unwrap().del_output(&output); if let Some(dpi) = dpi_change { if surface.version() >= 3 { // without version 3 we can't be dpi aware window_store.lock().unwrap().dpi_change(&surface, dpi); surface.set_buffer_scale(dpi); } } } } }); let window_store = evlp.store.clone(); let my_surface = surface.clone(); let mut frame = SWindow::::init( surface.clone(), (width, height), &evlp.env.compositor, &evlp.env.subcompositor, &evlp.env.shm, &evlp.env.shell, move |event, ()| match event { WEvent::Configure { new_size, .. } => { let mut store = window_store.lock().unwrap(); for window in &mut store.windows { if window.surface.equals(&my_surface) { window.newsize = new_size; window.need_refresh = true; *(window.need_frame_refresh.lock().unwrap()) = true; return; } } } WEvent::Refresh => { let store = window_store.lock().unwrap(); for window in &store.windows { if window.surface.equals(&my_surface) { *(window.need_frame_refresh.lock().unwrap()) = true; return; } } } WEvent::Close => { let mut store = window_store.lock().unwrap(); for window in &mut store.windows { if window.surface.equals(&my_surface) { window.closed = true; return; } } } }, ).unwrap(); for &(_, ref seat) in evlp.seats.lock().unwrap().iter() { frame.new_seat(seat); } // Check for fullscreen requirements if let Some(RootMonitorId { inner: PlatformMonitorId::Wayland(ref monitor_id), }) = attributes.fullscreen { frame.set_fullscreen(Some(&monitor_id.proxy)); } else if attributes.maximized { frame.set_maximized(); } frame.set_resizable(attributes.resizable); // set decorations frame.set_decorate(attributes.decorations); // min-max dimensions // TODO: Update for new DPI API //frame.set_min_size(attributes.min_dimensions); //frame.set_max_size(attributes.max_dimensions); let kill_switch = Arc::new(Mutex::new(false)); let need_frame_refresh = Arc::new(Mutex::new(true)); let frame = Arc::new(Mutex::new(frame)); evlp.store.lock().unwrap().windows.push(InternalWindow { closed: false, newsize: None, size: size.clone(), need_refresh: false, need_frame_refresh: need_frame_refresh.clone(), surface: surface.clone(), kill_switch: kill_switch.clone(), frame: Arc::downgrade(&frame), current_dpi: 1, new_dpi: None, }); evlp.evq.borrow_mut().sync_roundtrip().unwrap(); Ok(Window { display: evlp.display.clone(), surface: surface, frame: frame, monitors: monitor_list, outputs: evlp.env.outputs.clone(), size: size, kill_switch: (kill_switch, evlp.cleanup_needed.clone()), need_frame_refresh: need_frame_refresh, }) } #[inline] pub fn id(&self) -> WindowId { make_wid(&self.surface) } pub fn set_title(&self, title: &str) { self.frame.lock().unwrap().set_title(title.into()); } #[inline] pub fn show(&self) { // TODO } #[inline] pub fn hide(&self) { // TODO } #[inline] pub fn get_position(&self) -> Option { // Not possible with wayland None } #[inline] pub fn get_inner_position(&self) -> Option { // Not possible with wayland None } #[inline] pub fn set_position(&self, _pos: LogicalPosition) { // Not possible with wayland } pub fn get_inner_size(&self) -> Option { Some(self.size.lock().unwrap().clone().into()) } #[inline] pub fn get_outer_size(&self) -> Option { let (w, h) = self.size.lock().unwrap().clone(); // let (w, h) = super::wayland_window::add_borders(w as i32, h as i32); Some((w, h).into()) } #[inline] // NOTE: This will only resize the borders, the contents must be updated by the user pub fn set_inner_size(&self, size: LogicalSize) { let (w, h) = size.into(); self.frame.lock().unwrap().resize(w, h); *(self.size.lock().unwrap()) = (w, h); } #[inline] pub fn set_min_dimensions(&self, dimensions: Option) { self.frame.lock().unwrap().set_min_size(dimensions.map(Into::into)); } #[inline] pub fn set_max_dimensions(&self, dimensions: Option) { self.frame.lock().unwrap().set_max_size(dimensions.map(Into::into)); } #[inline] pub fn set_resizable(&self, resizable: bool) { self.frame.lock().unwrap().set_resizable(resizable); } #[inline] pub fn set_cursor(&self, _cursor: MouseCursor) { // TODO } #[inline] pub fn set_cursor_state(&self, state: CursorState) -> Result<(), String> { use CursorState::{Grab, Hide, Normal}; // TODO : not yet possible on wayland to grab cursor match state { Grab => Err("Cursor cannot be grabbed on wayland yet.".to_string()), Hide => Err("Cursor cannot be hidden on wayland yet.".to_string()), Normal => Ok(()), } } #[inline] pub fn hidpi_factor(&self) -> i32 { self.monitors.lock().unwrap().compute_hidpi_factor() } pub fn set_decorations(&self, decorate: bool) { self.frame.lock().unwrap().set_decorate(decorate); *(self.need_frame_refresh.lock().unwrap()) = true; } pub fn set_maximized(&self, maximized: bool) { if maximized { self.frame.lock().unwrap().set_maximized(); } else { self.frame.lock().unwrap().unset_maximized(); } } pub fn set_fullscreen(&self, monitor: Option) { if let Some(RootMonitorId { inner: PlatformMonitorId::Wayland(ref monitor_id), }) = monitor { self.frame .lock() .unwrap() .set_fullscreen(Some(&monitor_id.proxy)); } else { self.frame.lock().unwrap().unset_fullscreen(); } } #[inline] pub fn set_cursor_position(&self, _pos: LogicalPosition) -> Result<(), ()> { // TODO: not yet possible on wayland Err(()) } pub fn get_display(&self) -> &Display { &*self.display } pub fn get_surface(&self) -> &Proxy { &self.surface } pub fn get_current_monitor(&self) -> MonitorId { // we don't know how much each monitor sees us so... // just return the most recent one ? let guard = self.monitors.lock().unwrap(); guard.monitors.last().unwrap().clone() } pub fn get_available_monitors(&self) -> VecDeque { get_available_monitors(&self.outputs) } pub fn get_primary_monitor(&self) -> MonitorId { get_primary_monitor(&self.outputs) } } impl Drop for Window { fn drop(&mut self) { *(self.kill_switch.0.lock().unwrap()) = true; *(self.kill_switch.1.lock().unwrap()) = true; } } /* * Internal store for windows */ struct InternalWindow { surface: Proxy, newsize: Option<(u32, u32)>, size: Arc>, need_refresh: bool, need_frame_refresh: Arc>, closed: bool, kill_switch: Arc>, frame: Weak>>, current_dpi: i32, new_dpi: Option } pub struct WindowStore { windows: Vec, } impl WindowStore { pub fn new() -> WindowStore { WindowStore { windows: Vec::new(), } } pub fn find_wid(&self, surface: &Proxy) -> Option { for window in &self.windows { if surface.equals(&window.surface) { return Some(make_wid(surface)); } } None } pub fn cleanup(&mut self) -> Vec { let mut pruned = Vec::new(); self.windows.retain(|w| { if *w.kill_switch.lock().unwrap() { // window is dead, cleanup pruned.push(make_wid(&w.surface)); w.surface.destroy(); false } else { true } }); pruned } pub fn new_seat(&self, seat: &Proxy) { for window in &self.windows { if let Some(w) = window.frame.upgrade() { w.lock().unwrap().new_seat(seat); } } } fn dpi_change(&mut self, surface: &Proxy, new: i32) { for window in &mut self.windows { if surface.equals(&window.surface) { window.new_dpi = Some(new); } } } pub fn for_each(&mut self, mut f: F) where F: FnMut(Option<(u32, u32)>, &mut (u32, u32), Option, bool, bool, bool, WindowId, Option<&mut SWindow>), { for window in &mut self.windows { let opt_arc = window.frame.upgrade(); let mut opt_mutex_lock = opt_arc.as_ref().map(|m| m.lock().unwrap()); f( window.newsize.take(), &mut *(window.size.lock().unwrap()), window.new_dpi, window.need_refresh, ::std::mem::replace(&mut *window.need_frame_refresh.lock().unwrap(), false), window.closed, make_wid(&window.surface), opt_mutex_lock.as_mut().map(|m| &mut **m), ); if let Some(dpi) = window.new_dpi.take() { window.current_dpi = dpi; } window.need_refresh = false; // avoid re-spamming the event window.closed = false; } } } /* * Monitor list with some covenience method to compute DPI */ struct MonitorList { monitors: Vec } impl MonitorList { fn new() -> MonitorList { MonitorList { monitors: Vec::new() } } fn compute_hidpi_factor(&self) -> i32 { let mut factor = 1; for monitor_id in &self.monitors { let monitor_dpi = monitor_id.get_hidpi_factor(); if monitor_dpi > factor { factor = monitor_dpi; } } factor } fn add_output(&mut self, monitor: MonitorId) -> Option { let old_dpi = self.compute_hidpi_factor(); let monitor_dpi = monitor.get_hidpi_factor(); self.monitors.push(monitor); if monitor_dpi > old_dpi { Some(monitor_dpi) } else { None } } fn del_output(&mut self, output: &Proxy) -> Option { let old_dpi = self.compute_hidpi_factor(); self.monitors.retain(|m| !m.proxy.equals(output)); let new_dpi = self.compute_hidpi_factor(); if new_dpi != old_dpi { Some(new_dpi) } else { None } } }