use std::cell::RefCell; use std::collections::VecDeque; use std::fs::File; use std::io::Write; use std::os::unix::io::AsRawFd; use std::sync::{Arc, Mutex, Weak}; use std::sync::atomic::{AtomicBool, Ordering}; use {WindowEvent as Event, ElementState, MouseButton, MouseScrollDelta, TouchPhase, EventsLoopClosed, ControlFlow}; use super::{WindowId, DeviceId}; use super::window::WindowStore; use super::keyboard::init_keyboard; use wayland_client::{EnvHandler, EnvNotify, default_connect, EventQueue, EventQueueHandle, Proxy, StateToken}; use wayland_client::protocol::{wl_compositor, wl_seat, wl_shell, wl_shm, wl_subcompositor, wl_display, wl_registry, wl_output, wl_surface, wl_buffer, wl_pointer, wl_keyboard}; use super::wayland_window::{DecoratedSurface, Shell, init_decorated_surface, DecoratedSurfaceImplementation}; use super::wayland_protocols::unstable::xdg_shell::v6::client::zxdg_shell_v6; use super::tempfile; pub struct EventsLoopSink { buffer: VecDeque<::Event> } unsafe impl Send for EventsLoopSink { } impl EventsLoopSink { pub fn new() -> EventsLoopSink{ EventsLoopSink { buffer: VecDeque::new() } } pub fn send_event(&mut self, evt: ::WindowEvent, wid: WindowId) { let evt = ::Event::WindowEvent { event: evt, window_id: ::WindowId(::platform::WindowId::Wayland(wid)) }; self.buffer.push_back(evt); } pub fn send_raw_event(&mut self, evt: ::Event) { self.buffer.push_back(evt); } fn empty_with(&mut self, callback: &mut F) where F: FnMut(::Event) { for evt in self.buffer.drain(..) { callback(evt) } } } pub struct EventsLoop { // The Event Queue pub evq: RefCell, // our sink, shared with some handlers, buffering the events sink: Arc>, // Whether or not there is a pending `Awakened` event to be emitted. pending_wakeup: Arc, // The window store pub store: StateToken, // the env env_token: StateToken>, // the ctxt pub ctxt_token: StateToken, // a cleanup switch to prune dead windows pub cleanup_needed: Arc>, // The wayland display pub display: Arc, } // A handle that can be sent across threads and used to wake up the `EventsLoop`. // // We should only try and wake up the `EventsLoop` if it still exists, so we hold Weak ptrs. #[derive(Clone)] pub struct EventsLoopProxy { display: Weak, pending_wakeup: Weak, } impl EventsLoopProxy { // Causes the `EventsLoop` to stop blocking on `run_forever` and emit an `Awakened` event. // // Returns `Err` if the associated `EventsLoop` no longer exists. pub fn wakeup(&self) -> Result<(), EventsLoopClosed> { let display = self.display.upgrade(); let wakeup = self.pending_wakeup.upgrade(); match (display, wakeup) { (Some(display), Some(wakeup)) => { // Update the `EventsLoop`'s `pending_wakeup` flag. wakeup.store(true, Ordering::Relaxed); // Cause the `EventsLoop` to break from `dispatch` if it is currently blocked. display.sync(); display.flush().map_err(|_| EventsLoopClosed)?; Ok(()) }, _ => Err(EventsLoopClosed), } } } impl EventsLoop { pub fn new() -> Option { let (display, mut event_queue) = match default_connect() { Ok(ret) => ret, Err(_) => return None }; let registry = display.get_registry(); let ctxt_token = event_queue.state().insert( StateContext::new(registry.clone().unwrap()) ); let env_token = EnvHandler::init_with_notify( &mut event_queue, ®istry, env_notify(), ctxt_token.clone() ); // two round trips to fully initialize event_queue.sync_roundtrip().expect("Wayland connection unexpectedly lost"); event_queue.sync_roundtrip().expect("Wayland connection unexpectedly lost"); event_queue.state().with_value(&ctxt_token, |proxy, ctxt| { ctxt.ensure_shell(proxy.get_mut(&env_token)) }); let sink = Arc::new(Mutex::new(EventsLoopSink::new())); let store = event_queue.state().insert(WindowStore::new()); let seat_idata = SeatIData { sink: sink.clone(), keyboard: None, pointer: None, windows_token: store.clone() }; let mut me = EventsLoop { display: Arc::new(display), evq: RefCell::new(event_queue), sink: sink, pending_wakeup: Arc::new(AtomicBool::new(false)), store: store, ctxt_token: ctxt_token, env_token: env_token, cleanup_needed: Arc::new(Mutex::new(false)) }; me.init_seat(|evqh, seat| { evqh.register(seat, seat_implementation(), seat_idata); }); Some(me) } pub fn create_proxy(&self) -> EventsLoopProxy { EventsLoopProxy { display: Arc::downgrade(&self.display), pending_wakeup: Arc::downgrade(&self.pending_wakeup), } } pub fn poll_events(&mut self, mut callback: F) where F: FnMut(::Event) { // send pending events to the server self.display.flush().expect("Wayland connection lost."); // dispatch any pre-buffered events self.sink.lock().unwrap().empty_with(&mut callback); // try to read pending events if let Some(h) = self.evq.get_mut().prepare_read() { h.read_events().expect("Wayland connection lost."); } // dispatch wayland events self.evq.get_mut().dispatch_pending().expect("Wayland connection lost."); self.post_dispatch_triggers(); // dispatch buffered events to client self.sink.lock().unwrap().empty_with(&mut callback); } pub fn run_forever(&mut self, mut callback: F) where F: FnMut(::Event) -> ControlFlow, { // send pending events to the server self.display.flush().expect("Wayland connection lost."); // Check for control flow by wrapping the callback. let control_flow = ::std::cell::Cell::new(ControlFlow::Continue); let mut callback = |event| if let ControlFlow::Break = callback(event) { control_flow.set(ControlFlow::Break); }; // dispatch any pre-buffered events self.post_dispatch_triggers(); self.sink.lock().unwrap().empty_with(&mut callback); loop { // dispatch events blocking if needed self.evq.get_mut().dispatch().expect("Wayland connection lost."); self.post_dispatch_triggers(); // empty buffer of events self.sink.lock().unwrap().empty_with(&mut callback); if let ControlFlow::Break = control_flow.get() { break; } } } pub fn get_primary_monitor(&self) -> MonitorId { let mut guard = self.evq.borrow_mut(); let state = guard.state(); let state_ctxt = state.get(&self.ctxt_token); if let Some(info) = state_ctxt.monitors.iter().next() { MonitorId { info: info.clone() } } else { panic!("No monitor is available.") } } pub fn get_available_monitors(&self) -> VecDeque { let mut guard = self.evq.borrow_mut(); let state = guard.state(); let state_ctxt = state.get(&self.ctxt_token); state_ctxt.monitors.iter() .map(|m| MonitorId { info: m.clone() }) .collect() } } /* * Private EventsLoop Internals */ wayland_env!(InnerEnv, compositor: wl_compositor::WlCompositor, shm: wl_shm::WlShm, subcompositor: wl_subcompositor::WlSubcompositor ); pub struct StateContext { registry: wl_registry::WlRegistry, seat: Option, shell: Option, monitors: Vec>> } impl StateContext { fn new(registry: wl_registry::WlRegistry) -> StateContext { StateContext { registry: registry, seat: None, shell: None, monitors: Vec::new() } } /// Ensures a shell is available /// /// If a shell is already bound, do nothing. Otherwise, /// try to bind wl_shell as a fallback. If this fails, /// panic, as this is a bug from the compositor. fn ensure_shell(&mut self, env: &mut EnvHandler) { if self.shell.is_some() { return; } // xdg_shell is not available, so initialize wl_shell for &(name, ref interface, _) in env.globals() { if interface == "wl_shell" { self.shell = Some(Shell::Wl(self.registry.bind::(1, name))); return; } } // This is a compositor bug, it _must_ at least support wl_shell panic!("Compositor didi not advertize xdg_shell not wl_shell."); } pub fn monitor_id_for(&self, output: &wl_output::WlOutput) -> MonitorId { for info in &self.monitors { let guard = info.lock().unwrap(); if guard.output.equals(output) { return MonitorId { info: info.clone() }; } } panic!("Received an inexistent wl_output?!"); } } impl EventsLoop { pub fn init_seat(&mut self, f: F) where F: FnOnce(&mut EventQueueHandle, &wl_seat::WlSeat) { let mut guard = self.evq.borrow_mut(); if guard.state().get(&self.ctxt_token).seat.is_some() { // seat has already been init return; } // clone the token to make borrow checker happy let ctxt_token = self.ctxt_token.clone(); let seat = guard.state().with_value(&self.env_token, |proxy, env| { let ctxt = proxy.get(&ctxt_token); for &(name, ref interface, _) in env.globals() { if interface == wl_seat::WlSeat::interface_name() { return Some(ctxt.registry.bind::(5, name)); } } None }); if let Some(seat) = seat { f(&mut *guard, &seat); guard.state().get_mut(&self.ctxt_token).seat = Some(seat) } } fn post_dispatch_triggers(&mut self) { let mut sink = self.sink.lock().unwrap(); let evq = self.evq.get_mut(); // process a possible pending wakeup call if self.pending_wakeup.load(Ordering::Relaxed) { sink.send_raw_event(::Event::Awakened); self.pending_wakeup.store(false, Ordering::Relaxed); } // prune possible dead windows { let mut cleanup_needed = self.cleanup_needed.lock().unwrap(); if *cleanup_needed { evq.state().get_mut(&self.store).cleanup(); *cleanup_needed = false; } } // process pending resize/refresh evq.state().get_mut(&self.store).for_each( |newsize, refresh, closed, wid, decorated| { if let (Some((w, h)), Some(decorated)) = (newsize, decorated) { decorated.resize(w as i32, h as i32); sink.send_event(::WindowEvent::Resized(w as u32, h as u32), wid); } if refresh { sink.send_event(::WindowEvent::Refresh, wid); } if closed { sink.send_event(::WindowEvent::Closed, wid); } } ) } /// Creates a buffer of given size and assign it to the surface /// /// This buffer only contains white pixels, and is needed when using wl_shell /// to make sure the window actually exists and can receive events before the /// use starts its event loop fn blank_surface(&self, surface: &wl_surface::WlSurface, width: i32, height: i32) { let mut tmp = tempfile::tempfile().expect("Failed to create a tmpfile buffer."); for _ in 0..(width*height) { tmp.write_all(&[0xff,0xff,0xff,0xff]).unwrap(); } tmp.flush().unwrap(); let mut evq = self.evq.borrow_mut(); let pool = evq.state() .get(&self.env_token) .shm .create_pool(tmp.as_raw_fd(), width*height*4); let buffer = pool.create_buffer(0, width, height, width, wl_shm::Format::Argb8888) .expect("Pool cannot be already dead"); surface.attach(Some(&buffer), 0, 0); surface.commit(); // the buffer will keep the contents alive as needed pool.destroy(); // register the buffer for freeing evq.register(&buffer, free_buffer(), Some(tmp)); } /// Create a new window with given dimensions /// /// Grabs a lock on the event queue in the process pub fn create_window(&self, width: u32, height: u32, decorated: bool, implem: DecoratedSurfaceImplementation, idata: F) -> (wl_surface::WlSurface, DecoratedSurface, bool) where F: FnOnce(&wl_surface::WlSurface) -> ID { let (surface, decorated, xdg) = { let mut guard = self.evq.borrow_mut(); let env = guard.state().get(&self.env_token).clone_inner().unwrap(); let (shell, xdg) = match guard.state().get(&self.ctxt_token).shell { Some(Shell::Wl(ref wl_shell)) => (Shell::Wl(wl_shell.clone().unwrap()), false), Some(Shell::Xdg(ref xdg_shell)) => (Shell::Xdg(xdg_shell.clone().unwrap()), true), None => unreachable!() }; let seat = guard.state().get(&self.ctxt_token).seat.as_ref().and_then(|s| s.clone()); let surface = env.compositor.create_surface(); let decorated = init_decorated_surface( &mut guard, implem, idata(&surface), &surface, width as i32, height as i32, &env.compositor, &env.subcompositor, &env.shm, &shell, seat, decorated ).expect("Failed to create a tmpfile buffer."); (surface, decorated, xdg) }; if !xdg { // if using wl_shell, we need to draw something in order to kickstart // the event loop // if using xdg_shell, it is an error to do it now, and the events loop will not // be stuck. We cannot draw anything before having received an appropriate event // from the compositor self.blank_surface(&surface, width as i32, height as i32); } (surface, decorated, xdg) } } /* * Wayland protocol implementations */ fn env_notify() -> EnvNotify> { EnvNotify { new_global: |evqh, token, registry, id, interface, version| { use std::cmp::min; if interface == wl_output::WlOutput::interface_name() { // a new output is available let output = registry.bind::(min(version, 3), id); evqh.register(&output, output_impl(), token.clone()); evqh.state().get_mut(&token).monitors.push( Arc::new(Mutex::new(OutputInfo::new(output, id))) ); } else if interface == zxdg_shell_v6::ZxdgShellV6::interface_name() { // We have an xdg_shell, bind it let xdg_shell = registry.bind::(1, id); evqh.register(&xdg_shell, xdg_ping_implementation(), ()); evqh.state().get_mut(&token).shell = Some(Shell::Xdg(xdg_shell)); } }, del_global: |evqh, token, _, id| { // maybe this was a monitor, cleanup evqh.state().get_mut(&token).monitors.retain( |m| m.lock().unwrap().id != id ); }, ready: |_, _, _| {} } } fn xdg_ping_implementation() -> zxdg_shell_v6::Implementation<()> { zxdg_shell_v6::Implementation { ping: |_, _, shell, serial| { shell.pong(serial); } } } fn free_buffer() -> wl_buffer::Implementation> { wl_buffer::Implementation { release: |_, data, buffer| { buffer.destroy(); *data = None; } } } struct SeatIData { sink: Arc>, pointer: Option, keyboard: Option, windows_token: StateToken } fn seat_implementation() -> wl_seat::Implementation { wl_seat::Implementation { name: |_, _, _, _| {}, capabilities: |evqh, idata, seat, capabilities| { // create pointer if applicable if capabilities.contains(wl_seat::Capability::Pointer) && idata.pointer.is_none() { let pointer = seat.get_pointer().expect("Seat is not dead"); let p_idata = PointerIData::new(&idata.sink, idata.windows_token.clone()); evqh.register(&pointer, pointer_implementation(), p_idata); idata.pointer = Some(pointer); } // destroy pointer if applicable if !capabilities.contains(wl_seat::Capability::Pointer) { if let Some(pointer) = idata.pointer.take() { pointer.release(); } } // create keyboard if applicable if capabilities.contains(wl_seat::Capability::Keyboard) && idata.keyboard.is_none() { let kbd = seat.get_keyboard().expect("Seat is not dead"); init_keyboard(evqh, &kbd, &idata.sink); idata.keyboard = Some(kbd); } // destroy keyboard if applicable if !capabilities.contains(wl_seat::Capability::Keyboard) { if let Some(kbd) = idata.keyboard.take() { kbd.release(); } } // TODO: Handle touch } } } struct PointerIData { sink: Arc>, windows_token: StateToken, mouse_focus: Option, axis_buffer: Option<(f32, f32)>, axis_discrete_buffer: Option<(i32, i32)>, axis_state: TouchPhase, } impl PointerIData { fn new(sink: &Arc>, token: StateToken) -> PointerIData { PointerIData { sink: sink.clone(), windows_token: token, mouse_focus: None, axis_buffer: None, axis_discrete_buffer: None, axis_state: TouchPhase::Cancelled } } } fn pointer_implementation() -> wl_pointer::Implementation { wl_pointer::Implementation { enter: |evqh, idata, _, _, surface, x, y| { let wid = evqh.state().get(&idata.windows_token).find_wid(surface); if let Some(wid) = wid { idata.mouse_focus = Some(wid); let mut guard = idata.sink.lock().unwrap(); guard.send_event( Event::MouseEntered { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), }, wid, ); guard.send_event( Event::MouseMoved { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), position: (x, y), }, wid, ); } }, leave: |evqh, idata, _, _, surface| { idata.mouse_focus = None; let wid = evqh.state().get(&idata.windows_token).find_wid(surface); if let Some(wid) = wid { let mut guard = idata.sink.lock().unwrap(); guard.send_event( Event::MouseLeft { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), }, wid, ); } }, motion: |_, idata, _, _, x, y| { if let Some(wid) = idata.mouse_focus { idata.sink.lock().unwrap().send_event( Event::MouseMoved { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), position: (x, y) }, wid ); } }, button: |_, idata, _, _, _, button, state| { if let Some(wid) = idata.mouse_focus { let state = match state { wl_pointer::ButtonState::Pressed => ElementState::Pressed, wl_pointer::ButtonState::Released => ElementState::Released }; let button = match button { 0x110 => MouseButton::Left, 0x111 => MouseButton::Right, 0x112 => MouseButton::Middle, // TODO figure out the translation ? _ => return }; idata.sink.lock().unwrap().send_event( Event::MouseInput { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), state: state, button: button, }, wid ); } }, axis: |_, idata, pointer, _, axis, value| { if let Some(wid) = idata.mouse_focus { if pointer.version() < 5 { let (mut x, mut y) = (0.0, 0.0); // old seat compatibility match axis { // wayland vertical sign convention is the inverse of winit wl_pointer::Axis::VerticalScroll => y -= value as f32, wl_pointer::Axis::HorizontalScroll => x += value as f32 } idata.sink.lock().unwrap().send_event( Event::MouseWheel { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), delta: MouseScrollDelta::PixelDelta(x as f32, y as f32), phase: TouchPhase::Moved, }, wid ); } else { let (mut x, mut y) = idata.axis_buffer.unwrap_or((0.0, 0.0)); match axis { // wayland vertical sign convention is the inverse of winit wl_pointer::Axis::VerticalScroll => y -= value as f32, wl_pointer::Axis::HorizontalScroll => x += value as f32 } idata.axis_buffer = Some((x,y)); idata.axis_state = match idata.axis_state { TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved, _ => TouchPhase::Started } } } }, frame: |_, idata, _| { let axis_buffer = idata.axis_buffer.take(); let axis_discrete_buffer = idata.axis_discrete_buffer.take(); if let Some(wid) = idata.mouse_focus { if let Some((x, y)) = axis_discrete_buffer { idata.sink.lock().unwrap().send_event( Event::MouseWheel { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), delta: MouseScrollDelta::LineDelta(x as f32, y as f32), phase: idata.axis_state, }, wid ); } else if let Some((x, y)) = axis_buffer { idata.sink.lock().unwrap().send_event( Event::MouseWheel { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)), delta: MouseScrollDelta::PixelDelta(x as f32, y as f32), phase: idata.axis_state, }, wid ); } } }, axis_source: |_, _, _, _| {}, axis_stop: |_, idata, _, _, _| { idata.axis_state = TouchPhase::Ended; }, axis_discrete: |_, idata, _, axis, discrete| { let (mut x, mut y) = idata.axis_discrete_buffer.unwrap_or((0,0)); match axis { // wayland vertical sign convention is the inverse of winit wl_pointer::Axis::VerticalScroll => y -= discrete, wl_pointer::Axis::HorizontalScroll => x += discrete } idata.axis_discrete_buffer = Some((x,y)); idata.axis_state = match idata.axis_state { TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved, _ => TouchPhase::Started } }, } } /* * Monitor stuff */ fn output_impl() -> wl_output::Implementation> { wl_output::Implementation { geometry: |evqh, token, output, x, y, _, _, _, make, model, _| { let ctxt = evqh.state().get_mut(token); for info in &ctxt.monitors { let mut guard = info.lock().unwrap(); if guard.output.equals(output) { guard.pix_pos = (x, y); guard.name = format!("{} - {}", make, model); return; } } }, mode: |evqh, token, output, flags, w, h, _refresh| { if flags.contains(wl_output::Mode::Current) { let ctxt = evqh.state().get_mut(token); for info in &ctxt.monitors { let mut guard = info.lock().unwrap(); if guard.output.equals(output) { guard.pix_size = (w as u32, h as u32); return; } } } }, done: |_, _, _| {}, scale: |evqh, token, output, scale| { let ctxt = evqh.state().get_mut(token); for info in &ctxt.monitors { let mut guard = info.lock().unwrap(); if guard.output.equals(output) { guard.scale = scale as f32; return; } } } } } pub struct OutputInfo { pub output: wl_output::WlOutput, pub id: u32, pub scale: f32, pub pix_size: (u32, u32), pub pix_pos: (i32, i32), pub name: String } impl OutputInfo { fn new(output: wl_output::WlOutput, id: u32) -> OutputInfo { OutputInfo { output: output, id: id, scale: 1.0, pix_size: (0, 0), pix_pos: (0, 0), name: "".into() } } } #[derive(Clone)] pub struct MonitorId { pub info: Arc> } impl MonitorId { pub fn get_name(&self) -> Option { Some(self.info.lock().unwrap().name.clone()) } #[inline] pub fn get_native_identifier(&self) -> u32 { self.info.lock().unwrap().id } pub fn get_dimensions(&self) -> (u32, u32) { self.info.lock().unwrap().pix_size } pub fn get_position(&self) -> (i32, i32) { self.info.lock().unwrap().pix_pos } #[inline] pub fn get_hidpi_factor(&self) -> f32 { self.info.lock().unwrap().scale } }