use std::sync::{Arc, Mutex, Weak}; use wayland_client::protocol::{wl_display,wl_surface}; use wayland_client::{Proxy, StateToken}; use {CreationError, MouseCursor, CursorState, WindowAttributes}; use platform::MonitorId as PlatformMonitorId; use window::MonitorId as RootMonitorId; use super::{EventsLoop, WindowId, make_wid, MonitorId}; use super::wayland_window::{DecoratedSurface, DecoratedSurfaceImplementation}; use super::event_loop::StateContext; pub struct Window { display: Arc, surface: wl_surface::WlSurface, decorated: Arc>, monitors: Arc>, ready: Arc>, size: Arc>, kill_switch: (Arc>, Arc>), } impl Window { pub fn new(evlp: &EventsLoop, attributes: &WindowAttributes) -> Result { let (width, height) = attributes.dimensions.unwrap_or((800,600)); // Create the decorated surface let ready = Arc::new(Mutex::new(false)); let size = Arc::new(Mutex::new((width, height))); let store_token = evlp.store.clone(); let (surface, mut decorated, xdg) = evlp.create_window( width, height, attributes.decorations, decorated_impl(), |surface| DecoratedIData { ready: ready.clone(), surface: surface.clone().unwrap(), store_token: store_token.clone() } ); // If we are using xdg, we are not ready yet { *ready.lock().unwrap() = !xdg; } // Check for fullscreen requirements if let Some(RootMonitorId { inner: PlatformMonitorId::Wayland(ref monitor_id) }) = attributes.fullscreen { let info = monitor_id.info.lock().unwrap(); decorated.set_fullscreen(Some(&info.output)); } // setup the monitor tracking let monitor_list = Arc::new(Mutex::new(MonitorList::default())); { let mut evq = evlp.evq.borrow_mut(); let idata = (evlp.ctxt_token.clone(), monitor_list.clone()); evq.register(&surface, surface_impl(), idata); } // a surface commit with no buffer so that the compositor don't // forget to configure us surface.commit(); let kill_switch = Arc::new(Mutex::new(false)); let decorated = Arc::new(Mutex::new(decorated)); { let mut evq = evlp.evq.borrow_mut(); evq.state().get_mut(&store_token).windows.push(InternalWindow { closed: false, newsize: None, need_refresh: false, surface: surface.clone().unwrap(), kill_switch: kill_switch.clone(), decorated: Arc::downgrade(&decorated) }); evq.sync_roundtrip().unwrap(); } Ok(Window { display: evlp.display.clone(), surface: surface, decorated: decorated, monitors: monitor_list, ready: ready, size: size, kill_switch: (kill_switch, evlp.cleanup_needed.clone()) }) } #[inline] pub fn id(&self) -> WindowId { make_wid(&self.surface) } pub fn set_title(&self, title: &str) { self.decorated.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<(i32, i32)> { // Not possible with wayland None } #[inline] pub fn set_position(&self, _x: i32, _y: i32) { // Not possible with wayland } pub fn get_inner_size(&self) -> Option<(u32, u32)> { Some(self.size.lock().unwrap().clone()) } #[inline] pub fn get_outer_size(&self) -> Option<(u32, u32)> { let (w, h) = self.size.lock().unwrap().clone(); let (w, h) = super::wayland_window::add_borders(w as i32, h as i32); Some((w as u32, h as u32)) } #[inline] // NOTE: This will only resize the borders, the contents must be updated by the user pub fn set_inner_size(&self, x: u32, y: u32) { self.decorated.lock().unwrap().resize(x as i32, y as i32); *(self.size.lock().unwrap()) = (x, y); } #[inline] pub fn set_cursor(&self, _cursor: MouseCursor) { // TODO } #[inline] pub fn set_cursor_state(&self, state: CursorState) -> Result<(), String> { use CursorState::{Grab, Normal, Hide}; // 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) -> f32 { let mut factor = 1.0; let guard = self.monitors.lock().unwrap(); for monitor_id in &guard.monitors { let info = monitor_id.info.lock().unwrap(); if info.scale > factor { factor = info.scale; } } factor } #[inline] pub fn set_cursor_position(&self, _x: i32, _y: i32) -> Result<(), ()> { // TODO: not yet possible on wayland Err(()) } pub fn get_display(&self) -> &wl_display::WlDisplay { &*self.display } pub fn get_surface(&self) -> &wl_surface::WlSurface { &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 is_ready(&self) -> bool { *self.ready.lock().unwrap() } } 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: wl_surface::WlSurface, newsize: Option<(i32, i32)>, need_refresh: bool, closed: bool, kill_switch: Arc>, decorated: Weak> } pub struct WindowStore { windows: Vec } impl WindowStore { pub fn new() -> WindowStore { WindowStore { windows: Vec::new() } } pub fn find_wid(&self, surface: &wl_surface::WlSurface) -> Option { for window in &self.windows { if surface.equals(&window.surface) { return Some(make_wid(surface)); } } None } pub fn cleanup(&mut self) { self.windows.retain(|w| { if *w.kill_switch.lock().unwrap() { // window is dead, cleanup w.surface.destroy(); false } else { true } }); } pub fn for_each(&mut self, mut f: F) where F: FnMut(Option<(i32, i32)>, bool, bool, WindowId, Option<&mut DecoratedSurface>) { for window in &mut self.windows { let opt_arc = window.decorated.upgrade(); let mut opt_mutex_lock = opt_arc.as_ref().map(|m| m.lock().unwrap()); f( window.newsize.take(), window.need_refresh, window.closed, make_wid(&window.surface), opt_mutex_lock.as_mut().map(|m| &mut **m) ); window.need_refresh = false; // avoid re-spamming the event window.closed = false; } } } /* * Protocol implementation */ struct DecoratedIData { ready: Arc>, store_token: StateToken, surface: wl_surface::WlSurface } fn decorated_impl() -> DecoratedSurfaceImplementation { DecoratedSurfaceImplementation { configure: |evqh, idata, _, newsize| { *idata.ready.lock().unwrap() = true; let store = evqh.state().get_mut(&idata.store_token); for window in &mut store.windows { if window.surface.equals(&idata.surface) { window.newsize = newsize; window.need_refresh = true; return; } } }, close: |evqh, idata| { let store = evqh.state().get_mut(&idata.store_token); for window in &mut store.windows { if window.surface.equals(&idata.surface) { window.closed = true; return; } } } } } #[derive(Default)] struct MonitorList { monitors: Vec } fn surface_impl() -> wl_surface::Implementation<(StateToken, Arc>)> { wl_surface::Implementation { enter: |evqh, &mut (ref token, ref list), _, output| { let mut guard = list.lock().unwrap(); let ctxt = evqh.state().get(token); let monitor = ctxt.monitor_id_for(output); guard.monitors.push(monitor); }, leave: |evqh, &mut (ref token, ref list), _, output| { let mut guard = list.lock().unwrap(); let ctxt = evqh.state().get(token); let monitor = ctxt.monitor_id_for(output); guard.monitors.retain(|m| !Arc::ptr_eq(&m.info, &monitor.info)); } } }