2017-06-02 21:19:45 +10:00
|
|
|
use {WindowEvent as Event, ElementState, MouseButton, MouseScrollDelta, TouchPhase, ModifiersState,
|
|
|
|
KeyboardInput, EventsLoopClosed, ControlFlow};
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2017-09-01 03:43:24 +10:00
|
|
|
use std::collections::VecDeque;
|
2017-05-25 23:19:13 +10:00
|
|
|
use std::sync::{Arc, Mutex, Weak};
|
|
|
|
use std::sync::atomic::{self, AtomicBool};
|
2017-03-05 00:04:01 +11:00
|
|
|
|
2017-04-23 06:52:35 +10:00
|
|
|
use super::{DecoratedHandler, WindowId, DeviceId, WaylandContext};
|
2017-03-05 00:04:01 +11:00
|
|
|
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2017-06-24 04:25:06 +10:00
|
|
|
use wayland_client::{EventQueue, EventQueueHandle, Init, Proxy, Liveness};
|
2017-03-11 09:22:59 +11:00
|
|
|
use wayland_client::protocol::{wl_seat, wl_surface, wl_pointer, wl_keyboard};
|
|
|
|
|
|
|
|
use super::make_wid;
|
2017-03-05 00:04:01 +11:00
|
|
|
use super::wayland_window::DecoratedSurface;
|
2017-03-11 09:22:59 +11:00
|
|
|
use super::wayland_kbd::MappedKeyboard;
|
|
|
|
use super::keyboard::KbdHandler;
|
2017-03-05 00:04:01 +11:00
|
|
|
|
2017-03-11 09:56:31 +11:00
|
|
|
/// This struct is used as a holder for the callback
|
|
|
|
/// during the dispatching of events.
|
|
|
|
///
|
|
|
|
/// The proper ay to use it is:
|
|
|
|
/// - set a callback in it (and retrieve the noop one it contains)
|
|
|
|
/// - dispatch the EventQueue
|
|
|
|
/// - put back the noop callback in it
|
|
|
|
///
|
|
|
|
/// Failure to do so is unsafe™
|
2017-03-05 00:04:01 +11:00
|
|
|
pub struct EventsLoopSink {
|
2017-03-11 09:22:59 +11:00
|
|
|
callback: Box<FnMut(::Event)>
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe impl Send for EventsLoopSink { }
|
|
|
|
|
|
|
|
impl EventsLoopSink {
|
2017-09-01 03:43:24 +10:00
|
|
|
pub fn new() -> (EventsLoopSink, Arc<Mutex<VecDeque<::Event>>>) {
|
|
|
|
let buffer = Arc::new(Mutex::new(VecDeque::new()));
|
|
|
|
let buffer_clone = buffer.clone();
|
|
|
|
let sink = EventsLoopSink {
|
|
|
|
callback: Box::new(move |evt| { println!("TEMP: {:?}", evt); buffer.lock().unwrap().push_back(evt)}),
|
|
|
|
};
|
|
|
|
(sink, buffer_clone)
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2017-03-11 09:22:59 +11:00
|
|
|
pub fn send_event(&mut self, evt: ::WindowEvent, wid: WindowId) {
|
2017-03-05 00:04:01 +11:00
|
|
|
let evt = ::Event::WindowEvent {
|
|
|
|
event: evt,
|
|
|
|
window_id: ::WindowId(::platform::WindowId::Wayland(wid))
|
|
|
|
};
|
2017-03-11 09:22:59 +11:00
|
|
|
(self.callback)(evt)
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// This function is only safe of the set callback is unset before exclusive
|
|
|
|
// access to the wayland EventQueue is finished.
|
|
|
|
//
|
|
|
|
// The callback also cannot be used any longer as long as it has not been
|
|
|
|
// cleared from the Sink.
|
2017-03-11 09:22:59 +11:00
|
|
|
unsafe fn set_callback(&mut self, cb: Box<FnMut(::Event)>) -> Box<FnMut(::Event)> {
|
|
|
|
::std::mem::replace(&mut self.callback, cb)
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2017-06-02 21:19:45 +10:00
|
|
|
fn with_callback<F>(&mut self, f: F)
|
|
|
|
where F: FnOnce(&mut FnMut(::Event)),
|
|
|
|
{
|
2017-03-11 09:22:59 +11:00
|
|
|
f(&mut *self.callback)
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct EventsLoop {
|
2017-03-11 09:56:31 +11:00
|
|
|
// the global wayland context
|
2017-03-05 00:04:01 +11:00
|
|
|
ctxt: Arc<WaylandContext>,
|
2017-03-11 09:56:31 +11:00
|
|
|
// our EventQueue
|
2017-03-05 00:04:01 +11:00
|
|
|
evq: Arc<Mutex<EventQueue>>,
|
2017-03-11 09:56:31 +11:00
|
|
|
// ids of the DecoratedHandlers of the surfaces we know
|
2017-03-11 09:40:31 +11:00
|
|
|
decorated_ids: Mutex<Vec<(usize, Arc<wl_surface::WlSurface>)>>,
|
2017-03-11 09:56:31 +11:00
|
|
|
// our sink, receiver of callbacks, shared with some handlers
|
2017-03-05 00:04:01 +11:00
|
|
|
sink: Arc<Mutex<EventsLoopSink>>,
|
2017-09-01 03:43:24 +10:00
|
|
|
// a buffer in which events that were dispatched internally are stored
|
|
|
|
// until the user next dispatches events
|
|
|
|
buffer: Arc<Mutex<VecDeque<::Event>>>,
|
2017-03-11 09:56:31 +11:00
|
|
|
// trigger cleanup of the dead surfaces
|
2017-03-11 09:40:31 +11:00
|
|
|
cleanup_needed: Arc<AtomicBool>,
|
2017-05-25 23:19:13 +10:00
|
|
|
// Whether or not there is a pending `Awakened` event to be emitted.
|
|
|
|
pending_wakeup: Arc<AtomicBool>,
|
|
|
|
hid: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
pub struct EventsLoopProxy {
|
|
|
|
ctxt: Weak<WaylandContext>,
|
|
|
|
pending_wakeup: Weak<AtomicBool>,
|
|
|
|
}
|
|
|
|
|
|
|
|
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 ctxt = self.ctxt.upgrade();
|
|
|
|
let wakeup = self.pending_wakeup.upgrade();
|
|
|
|
match (ctxt, wakeup) {
|
|
|
|
(Some(ctxt), Some(wakeup)) => {
|
|
|
|
// Update the `EventsLoop`'s `pending_wakeup` flag.
|
|
|
|
wakeup.store(true, atomic::Ordering::Relaxed);
|
|
|
|
// Cause the `EventsLoop` to break from `dispatch` if it is currently blocked.
|
|
|
|
ctxt.display.sync();
|
2017-05-27 22:51:59 +10:00
|
|
|
ctxt.display.flush().ok();
|
2017-05-25 23:19:13 +10:00
|
|
|
Ok(())
|
|
|
|
},
|
|
|
|
_ => Err(EventsLoopClosed),
|
|
|
|
}
|
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl EventsLoop {
|
|
|
|
pub fn new(ctxt: Arc<WaylandContext>) -> EventsLoop {
|
2017-03-11 09:22:59 +11:00
|
|
|
let mut evq = ctxt.display.create_event_queue();
|
2017-09-01 03:43:24 +10:00
|
|
|
let (sink, buffer) = EventsLoopSink::new();
|
|
|
|
let sink = Arc::new(Mutex::new(sink));
|
2017-03-11 09:22:59 +11:00
|
|
|
let hid = evq.add_handler_with_init(InputHandler::new(&ctxt, sink.clone()));
|
2017-03-05 00:04:01 +11:00
|
|
|
EventsLoop {
|
|
|
|
ctxt: ctxt,
|
|
|
|
evq: Arc::new(Mutex::new(evq)),
|
|
|
|
decorated_ids: Mutex::new(Vec::new()),
|
2017-03-11 09:22:59 +11:00
|
|
|
sink: sink,
|
2017-09-01 03:43:24 +10:00
|
|
|
buffer: buffer,
|
2017-05-25 23:19:13 +10:00
|
|
|
pending_wakeup: Arc::new(AtomicBool::new(false)),
|
2017-03-11 09:40:31 +11:00
|
|
|
cleanup_needed: Arc::new(AtomicBool::new(false)),
|
2017-03-11 09:22:59 +11:00
|
|
|
hid: hid
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-01 19:04:57 +10:00
|
|
|
#[inline]
|
|
|
|
pub fn context(&self) -> &Arc<WaylandContext> {
|
|
|
|
&self.ctxt
|
|
|
|
}
|
|
|
|
|
2017-05-25 23:19:13 +10:00
|
|
|
pub fn create_proxy(&self) -> EventsLoopProxy {
|
|
|
|
EventsLoopProxy {
|
|
|
|
ctxt: Arc::downgrade(&self.ctxt),
|
|
|
|
pending_wakeup: Arc::downgrade(&self.pending_wakeup),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-11 09:56:31 +11:00
|
|
|
// some internals that Window needs access to
|
2017-03-11 09:40:31 +11:00
|
|
|
pub fn get_window_init(&self) -> (Arc<Mutex<EventQueue>>, Arc<AtomicBool>) {
|
|
|
|
(self.evq.clone(), self.cleanup_needed.clone())
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
|
|
|
pub fn register_window(&self, decorated_id: usize, surface: Arc<wl_surface::WlSurface>) {
|
2017-09-01 03:43:24 +10:00
|
|
|
self.buffer.lock().unwrap().push_back(::Event::WindowEvent {
|
|
|
|
window_id: ::WindowId(::platform::WindowId::Wayland(make_wid(&surface))),
|
|
|
|
event: ::WindowEvent::Refresh
|
|
|
|
});
|
2017-03-11 09:40:31 +11:00
|
|
|
self.decorated_ids.lock().unwrap().push((decorated_id, surface.clone()));
|
2017-03-11 09:22:59 +11:00
|
|
|
let mut guard = self.evq.lock().unwrap();
|
|
|
|
let mut state = guard.state();
|
|
|
|
state.get_mut_handler::<InputHandler>(self.hid).windows.push(surface);
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2017-03-11 09:40:31 +11:00
|
|
|
fn process_resize(evq: &mut EventQueue, ids: &[(usize, Arc<wl_surface::WlSurface>)], callback: &mut FnMut(::Event))
|
2017-03-05 00:04:01 +11:00
|
|
|
{
|
|
|
|
let mut state = evq.state();
|
2017-03-11 09:40:31 +11:00
|
|
|
for &(decorated_id, ref window) in ids {
|
2017-03-05 00:04:01 +11:00
|
|
|
let decorated = state.get_mut_handler::<DecoratedSurface<DecoratedHandler>>(decorated_id);
|
|
|
|
if let Some((w, h)) = decorated.handler().as_mut().and_then(|h| h.take_newsize()) {
|
|
|
|
decorated.resize(w as i32, h as i32);
|
|
|
|
callback(
|
|
|
|
::Event::WindowEvent {
|
2017-03-11 09:40:31 +11:00
|
|
|
window_id: ::WindowId(::platform::WindowId::Wayland(make_wid(&window))),
|
2017-03-05 00:04:01 +11:00
|
|
|
event: ::WindowEvent::Resized(w,h)
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2017-07-06 00:23:10 +10:00
|
|
|
if decorated.handler().as_ref().map(|h| h.is_closed()).unwrap_or(false) {
|
|
|
|
callback(
|
|
|
|
::Event::WindowEvent {
|
|
|
|
window_id: ::WindowId(::platform::WindowId::Wayland(make_wid(&window))),
|
|
|
|
event: ::WindowEvent::Closed
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-11 09:40:31 +11:00
|
|
|
fn prune_dead_windows(&self) {
|
2017-06-24 04:25:06 +10:00
|
|
|
self.decorated_ids.lock().unwrap().retain(|&(_, ref w)| w.status() == Liveness::Alive);
|
2017-03-11 09:40:31 +11:00
|
|
|
let mut evq_guard = self.evq.lock().unwrap();
|
|
|
|
let mut state = evq_guard.state();
|
|
|
|
let handler = state.get_mut_handler::<InputHandler>(self.hid);
|
2017-06-24 04:25:06 +10:00
|
|
|
handler.windows.retain(|w| w.status() == Liveness::Alive);
|
2017-03-11 09:40:31 +11:00
|
|
|
if let Some(w) = handler.mouse_focus.take() {
|
2017-06-24 04:25:06 +10:00
|
|
|
if w.status() == Liveness::Alive {
|
2017-03-11 09:40:31 +11:00
|
|
|
handler.mouse_focus = Some(w)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-01 03:43:24 +10:00
|
|
|
fn empty_buffer<F>(&self, callback: &mut F) where F: FnMut(::Event) {
|
|
|
|
let mut guard = self.buffer.lock().unwrap();
|
|
|
|
for evt in guard.drain(..) {
|
|
|
|
callback(evt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn poll_events<F>(&mut self, mut callback: F)
|
2017-03-05 00:04:01 +11:00
|
|
|
where F: FnMut(::Event)
|
|
|
|
{
|
|
|
|
// send pending requests to the server...
|
|
|
|
self.ctxt.flush();
|
|
|
|
|
|
|
|
// first of all, get exclusive access to this event queue
|
|
|
|
let mut evq_guard = self.evq.lock().unwrap();
|
|
|
|
|
2017-09-01 03:43:24 +10:00
|
|
|
// dispatch pre-buffered events
|
|
|
|
self.empty_buffer(&mut callback);
|
|
|
|
|
2017-03-05 00:04:01 +11:00
|
|
|
// read some events from the socket if some are waiting & queue is empty
|
|
|
|
if let Some(guard) = evq_guard.prepare_read() {
|
|
|
|
guard.read_events().expect("Wayland connection unexpectedly lost");
|
|
|
|
}
|
|
|
|
|
|
|
|
// set the callback into the sink
|
2017-03-11 09:22:59 +11:00
|
|
|
// we extend the lifetime of the closure to 'static to be able to put it in
|
|
|
|
// the sink, but we'll explicitly drop it at the end of this function, so it's fine
|
|
|
|
let static_cb = unsafe { ::std::mem::transmute(Box::new(callback) as Box<FnMut(_)>) };
|
|
|
|
let old_cb = unsafe { self.sink.lock().unwrap().set_callback(static_cb) };
|
2017-03-05 00:04:01 +11:00
|
|
|
|
|
|
|
// then do the actual dispatching
|
|
|
|
self.ctxt.dispatch_pending();
|
|
|
|
evq_guard.dispatch_pending().expect("Wayland connection unexpectedly lost");
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2017-05-25 23:19:13 +10:00
|
|
|
self.emit_pending_wakeup();
|
|
|
|
|
2017-05-14 23:28:27 +10:00
|
|
|
{
|
|
|
|
let mut sink_guard = self.sink.lock().unwrap();
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2017-05-14 23:28:27 +10:00
|
|
|
// events where probably dispatched, process resize
|
|
|
|
let ids_guard = self.decorated_ids.lock().unwrap();
|
|
|
|
sink_guard.with_callback(
|
|
|
|
|cb| Self::process_resize(&mut evq_guard, &ids_guard, cb)
|
|
|
|
);
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2017-05-14 23:28:27 +10:00
|
|
|
// replace the old noop callback
|
|
|
|
unsafe { sink_guard.set_callback(old_cb) };
|
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2017-05-25 23:19:13 +10:00
|
|
|
if self.cleanup_needed.swap(false, atomic::Ordering::Relaxed) {
|
2017-03-11 09:40:31 +11:00
|
|
|
self.prune_dead_windows()
|
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2017-06-02 21:19:45 +10:00
|
|
|
pub fn run_forever<F>(&mut self, mut callback: F)
|
|
|
|
where F: FnMut(::Event) -> ControlFlow,
|
2017-03-05 00:04:01 +11:00
|
|
|
{
|
|
|
|
// send pending requests to the server...
|
|
|
|
self.ctxt.flush();
|
|
|
|
|
|
|
|
// first of all, get exclusive access to this event queue
|
|
|
|
let mut evq_guard = self.evq.lock().unwrap();
|
|
|
|
|
2017-06-02 21:19:45 +10:00
|
|
|
// Check for control flow by wrapping the callback.
|
|
|
|
let control_flow = ::std::cell::Cell::new(ControlFlow::Continue);
|
2017-09-01 03:43:24 +10:00
|
|
|
let mut callback = |event| if let ControlFlow::Break = callback(event) {
|
2017-06-20 21:25:53 +10:00
|
|
|
control_flow.set(ControlFlow::Break);
|
2017-06-02 21:19:45 +10:00
|
|
|
};
|
|
|
|
|
2017-09-01 03:43:24 +10:00
|
|
|
// dispatch pre-buffered events
|
|
|
|
self.empty_buffer(&mut callback);
|
|
|
|
|
2017-03-05 00:04:01 +11:00
|
|
|
// set the callback into the sink
|
2017-03-11 09:22:59 +11:00
|
|
|
// we extend the lifetime of the closure to 'static to be able to put it in
|
|
|
|
// the sink, but we'll explicitly drop it at the end of this function, so it's fine
|
|
|
|
let static_cb = unsafe { ::std::mem::transmute(Box::new(callback) as Box<FnMut(_)>) };
|
|
|
|
let old_cb = unsafe { self.sink.lock().unwrap().set_callback(static_cb) };
|
2017-03-05 00:04:01 +11:00
|
|
|
|
2017-06-02 21:19:45 +10:00
|
|
|
loop {
|
2017-03-05 00:04:01 +11:00
|
|
|
self.ctxt.dispatch();
|
|
|
|
evq_guard.dispatch_pending().expect("Wayland connection unexpectedly lost");
|
2017-05-25 23:19:13 +10:00
|
|
|
|
|
|
|
self.emit_pending_wakeup();
|
|
|
|
|
2017-03-05 00:04:01 +11:00
|
|
|
let ids_guard = self.decorated_ids.lock().unwrap();
|
2017-06-02 21:19:45 +10:00
|
|
|
self.sink.lock().unwrap()
|
|
|
|
.with_callback(|cb| Self::process_resize(&mut evq_guard, &ids_guard, cb));
|
2017-03-05 00:04:01 +11:00
|
|
|
self.ctxt.flush();
|
2017-03-11 09:56:31 +11:00
|
|
|
|
2017-05-25 23:19:13 +10:00
|
|
|
if self.cleanup_needed.swap(false, atomic::Ordering::Relaxed) {
|
2017-03-11 09:56:31 +11:00
|
|
|
self.prune_dead_windows()
|
|
|
|
}
|
2017-06-02 21:19:45 +10:00
|
|
|
|
2017-06-20 21:25:53 +10:00
|
|
|
if let ControlFlow::Break = control_flow.get() {
|
2017-06-02 21:19:45 +10:00
|
|
|
break;
|
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2017-03-11 09:22:59 +11:00
|
|
|
// replace the old noop callback
|
|
|
|
unsafe { self.sink.lock().unwrap().set_callback(old_cb) };
|
|
|
|
}
|
2017-05-25 23:19:13 +10:00
|
|
|
|
|
|
|
// If an `EventsLoopProxy` has signalled a wakeup, emit an event and reset the flag.
|
|
|
|
fn emit_pending_wakeup(&self) {
|
|
|
|
if self.pending_wakeup.load(atomic::Ordering::Relaxed) {
|
|
|
|
self.sink.lock().unwrap().with_callback(|cb| cb(::Event::Awakened));
|
|
|
|
self.pending_wakeup.store(false, atomic::Ordering::Relaxed);
|
|
|
|
}
|
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
enum KbdType {
|
|
|
|
Mapped(MappedKeyboard<KbdHandler>),
|
|
|
|
Plain(Option<WindowId>)
|
|
|
|
}
|
|
|
|
|
|
|
|
struct InputHandler {
|
|
|
|
my_id: usize,
|
|
|
|
windows: Vec<Arc<wl_surface::WlSurface>>,
|
|
|
|
seat: Option<wl_seat::WlSeat>,
|
|
|
|
mouse: Option<wl_pointer::WlPointer>,
|
|
|
|
mouse_focus: Option<Arc<wl_surface::WlSurface>>,
|
2017-04-23 06:52:35 +10:00
|
|
|
mouse_location: (f64, f64),
|
2017-03-11 09:22:59 +11:00
|
|
|
axis_buffer: Option<(f32, f32)>,
|
|
|
|
axis_discrete_buffer: Option<(i32, i32)>,
|
|
|
|
axis_state: TouchPhase,
|
|
|
|
kbd: Option<wl_keyboard::WlKeyboard>,
|
|
|
|
kbd_handler: KbdType,
|
|
|
|
callback: Arc<Mutex<EventsLoopSink>>
|
|
|
|
}
|
|
|
|
|
|
|
|
impl InputHandler {
|
|
|
|
fn new(ctxt: &WaylandContext, sink: Arc<Mutex<EventsLoopSink>>) -> InputHandler {
|
|
|
|
let kbd_handler = match MappedKeyboard::new(KbdHandler::new(sink.clone())) {
|
|
|
|
Ok(h) => KbdType::Mapped(h),
|
|
|
|
Err(_) => KbdType::Plain(None)
|
|
|
|
};
|
|
|
|
InputHandler {
|
|
|
|
my_id: 0,
|
|
|
|
windows: Vec::new(),
|
|
|
|
seat: ctxt.get_seat(),
|
|
|
|
mouse: None,
|
|
|
|
mouse_focus: None,
|
2017-04-23 06:52:35 +10:00
|
|
|
mouse_location: (0.0,0.0),
|
2017-03-11 09:22:59 +11:00
|
|
|
axis_buffer: None,
|
|
|
|
axis_discrete_buffer: None,
|
|
|
|
axis_state: TouchPhase::Started,
|
|
|
|
kbd: None,
|
|
|
|
kbd_handler: kbd_handler,
|
|
|
|
callback: sink
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Init for InputHandler {
|
|
|
|
fn init(&mut self, evqh: &mut EventQueueHandle, index: usize) {
|
|
|
|
if let Some(ref seat) = self.seat {
|
|
|
|
evqh.register::<_, InputHandler>(seat, index);
|
|
|
|
}
|
|
|
|
self.my_id = index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl wl_seat::Handler for InputHandler {
|
|
|
|
fn capabilities(&mut self,
|
|
|
|
evqh: &mut EventQueueHandle,
|
|
|
|
seat: &wl_seat::WlSeat,
|
|
|
|
capabilities: wl_seat::Capability)
|
|
|
|
{
|
|
|
|
// create pointer if applicable
|
|
|
|
if capabilities.contains(wl_seat::Pointer) && self.mouse.is_none() {
|
|
|
|
let pointer = seat.get_pointer().expect("Seat is not dead");
|
|
|
|
evqh.register::<_, InputHandler>(&pointer, self.my_id);
|
|
|
|
self.mouse = Some(pointer);
|
|
|
|
}
|
|
|
|
// destroy pointer if applicable
|
|
|
|
if !capabilities.contains(wl_seat::Pointer) {
|
|
|
|
if let Some(pointer) = self.mouse.take() {
|
|
|
|
pointer.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// create keyboard if applicable
|
|
|
|
if capabilities.contains(wl_seat::Keyboard) && self.kbd.is_none() {
|
|
|
|
let kbd = seat.get_keyboard().expect("Seat is not dead");
|
|
|
|
evqh.register::<_, InputHandler>(&kbd, self.my_id);
|
|
|
|
self.kbd = Some(kbd);
|
|
|
|
}
|
|
|
|
// destroy keyboard if applicable
|
|
|
|
if !capabilities.contains(wl_seat::Keyboard) {
|
|
|
|
if let Some(kbd) = self.kbd.take() {
|
|
|
|
kbd.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
declare_handler!(InputHandler, wl_seat::Handler, wl_seat::WlSeat);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Pointer Handling
|
|
|
|
*/
|
|
|
|
|
|
|
|
impl wl_pointer::Handler for InputHandler {
|
|
|
|
fn enter(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
_serial: u32,
|
|
|
|
surface: &wl_surface::WlSurface,
|
|
|
|
surface_x: f64,
|
|
|
|
surface_y: f64)
|
|
|
|
{
|
2017-04-23 06:52:35 +10:00
|
|
|
self.mouse_location = (surface_x, surface_y);
|
2017-03-11 09:22:59 +11:00
|
|
|
for window in &self.windows {
|
|
|
|
if window.equals(surface) {
|
|
|
|
self.mouse_focus = Some(window.clone());
|
|
|
|
let (w, h) = self.mouse_location;
|
|
|
|
let mut guard = self.callback.lock().unwrap();
|
2017-04-23 06:52:35 +10:00
|
|
|
guard.send_event(Event::MouseEntered { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)) },
|
|
|
|
make_wid(window));
|
|
|
|
guard.send_event(Event::MouseMoved { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)),
|
|
|
|
position: (w, h) },
|
|
|
|
make_wid(window));
|
2017-03-11 09:22:59 +11:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn leave(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
_serial: u32,
|
|
|
|
surface: &wl_surface::WlSurface)
|
|
|
|
{
|
|
|
|
self.mouse_focus = None;
|
|
|
|
for window in &self.windows {
|
|
|
|
if window.equals(surface) {
|
2017-04-23 06:52:35 +10:00
|
|
|
self.callback.lock().unwrap().send_event(Event::MouseLeft { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)) },
|
|
|
|
make_wid(window));
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn motion(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
_time: u32,
|
|
|
|
surface_x: f64,
|
|
|
|
surface_y: f64)
|
|
|
|
{
|
2017-04-23 06:52:35 +10:00
|
|
|
self.mouse_location = (surface_x, surface_y);
|
2017-03-11 09:22:59 +11:00
|
|
|
if let Some(ref window) = self.mouse_focus {
|
|
|
|
let (w,h) = self.mouse_location;
|
2017-04-23 06:52:35 +10:00
|
|
|
self.callback.lock().unwrap().send_event(Event::MouseMoved { device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)),
|
|
|
|
position: (w, h) }, make_wid(window));
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn button(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
_serial: u32,
|
|
|
|
_time: u32,
|
|
|
|
button: u32,
|
|
|
|
state: wl_pointer::ButtonState)
|
|
|
|
{
|
|
|
|
if let Some(ref window) = self.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
|
|
|
|
};
|
2017-04-23 06:52:35 +10:00
|
|
|
self.callback.lock().unwrap().send_event(
|
|
|
|
Event::MouseInput {
|
|
|
|
device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)),
|
|
|
|
state: state,
|
|
|
|
button: button,
|
|
|
|
},
|
|
|
|
make_wid(window)
|
|
|
|
);
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn axis(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
_time: u32,
|
|
|
|
axis: wl_pointer::Axis,
|
|
|
|
value: f64)
|
|
|
|
{
|
|
|
|
let (mut x, mut y) = self.axis_buffer.unwrap_or((0.0, 0.0));
|
|
|
|
match axis {
|
2017-07-18 21:41:15 +10:00
|
|
|
// wayland vertical sign convention is the inverse of winit
|
|
|
|
wl_pointer::Axis::VerticalScroll => y -= value as f32,
|
2017-03-11 09:22:59 +11:00
|
|
|
wl_pointer::Axis::HorizontalScroll => x += value as f32
|
|
|
|
}
|
|
|
|
self.axis_buffer = Some((x,y));
|
|
|
|
self.axis_state = match self.axis_state {
|
|
|
|
TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved,
|
|
|
|
_ => TouchPhase::Started
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn frame(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer)
|
|
|
|
{
|
|
|
|
let axis_buffer = self.axis_buffer.take();
|
|
|
|
let axis_discrete_buffer = self.axis_discrete_buffer.take();
|
|
|
|
if let Some(ref window) = self.mouse_focus {
|
|
|
|
if let Some((x, y)) = axis_discrete_buffer {
|
|
|
|
self.callback.lock().unwrap().send_event(
|
2017-04-23 06:52:35 +10:00
|
|
|
Event::MouseWheel {
|
|
|
|
device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)),
|
|
|
|
delta: MouseScrollDelta::LineDelta(x as f32, y as f32),
|
|
|
|
phase: self.axis_state,
|
|
|
|
},
|
2017-03-11 09:22:59 +11:00
|
|
|
make_wid(window)
|
|
|
|
);
|
|
|
|
} else if let Some((x, y)) = axis_buffer {
|
|
|
|
self.callback.lock().unwrap().send_event(
|
2017-04-23 06:52:35 +10:00
|
|
|
Event::MouseWheel {
|
|
|
|
device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)),
|
|
|
|
delta: MouseScrollDelta::PixelDelta(x as f32, y as f32),
|
|
|
|
phase: self.axis_state,
|
|
|
|
},
|
2017-03-11 09:22:59 +11:00
|
|
|
make_wid(window)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn axis_source(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
_axis_source: wl_pointer::AxisSource)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
fn axis_stop(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
_time: u32,
|
|
|
|
_axis: wl_pointer::Axis)
|
|
|
|
{
|
|
|
|
self.axis_state = TouchPhase::Ended;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn axis_discrete(&mut self,
|
|
|
|
_evqh: &mut EventQueueHandle,
|
|
|
|
_proxy: &wl_pointer::WlPointer,
|
|
|
|
axis: wl_pointer::Axis,
|
|
|
|
discrete: i32)
|
|
|
|
{
|
|
|
|
let (mut x, mut y) = self.axis_discrete_buffer.unwrap_or((0,0));
|
|
|
|
match axis {
|
2017-07-18 21:41:15 +10:00
|
|
|
// wayland vertical sign convention is the inverse of winit
|
|
|
|
wl_pointer::Axis::VerticalScroll => y -= discrete,
|
2017-03-11 09:22:59 +11:00
|
|
|
wl_pointer::Axis::HorizontalScroll => x += discrete
|
|
|
|
}
|
|
|
|
self.axis_discrete_buffer = Some((x,y));
|
|
|
|
self.axis_state = match self.axis_state {
|
|
|
|
TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved,
|
|
|
|
_ => TouchPhase::Started
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
declare_handler!(InputHandler, wl_pointer::Handler, wl_pointer::WlPointer);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Keyboard Handling
|
|
|
|
*/
|
|
|
|
|
|
|
|
impl wl_keyboard::Handler for InputHandler {
|
|
|
|
// mostly pass-through
|
|
|
|
fn keymap(&mut self,
|
|
|
|
evqh: &mut EventQueueHandle,
|
|
|
|
proxy: &wl_keyboard::WlKeyboard,
|
|
|
|
format: wl_keyboard::KeymapFormat,
|
|
|
|
fd: ::std::os::unix::io::RawFd,
|
|
|
|
size: u32)
|
|
|
|
{
|
|
|
|
match self.kbd_handler {
|
|
|
|
KbdType::Mapped(ref mut h) => h.keymap(evqh, proxy, format, fd, size),
|
|
|
|
_ => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn enter(&mut self,
|
|
|
|
evqh: &mut EventQueueHandle,
|
|
|
|
proxy: &wl_keyboard::WlKeyboard,
|
|
|
|
serial: u32,
|
|
|
|
surface: &wl_surface::WlSurface,
|
|
|
|
keys: Vec<u8>)
|
|
|
|
{
|
|
|
|
for window in &self.windows {
|
|
|
|
if window.equals(surface) {
|
|
|
|
self.callback.lock().unwrap().send_event(Event::Focused(true), make_wid(window));
|
|
|
|
match self.kbd_handler {
|
|
|
|
KbdType::Mapped(ref mut h) => {
|
|
|
|
h.handler().target = Some(make_wid(window));
|
|
|
|
h.enter(evqh, proxy, serial, surface, keys);
|
|
|
|
},
|
|
|
|
KbdType::Plain(ref mut target) => {
|
|
|
|
*target = Some(make_wid(window))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn leave(&mut self,
|
|
|
|
evqh: &mut EventQueueHandle,
|
|
|
|
proxy: &wl_keyboard::WlKeyboard,
|
|
|
|
serial: u32,
|
|
|
|
surface: &wl_surface::WlSurface)
|
|
|
|
{
|
|
|
|
for window in &self.windows {
|
|
|
|
if window.equals(surface) {
|
|
|
|
self.callback.lock().unwrap().send_event(Event::Focused(false), make_wid(window));
|
|
|
|
match self.kbd_handler {
|
|
|
|
KbdType::Mapped(ref mut h) => {
|
|
|
|
h.handler().target = None;
|
|
|
|
h.leave(evqh, proxy, serial, surface);
|
|
|
|
},
|
|
|
|
KbdType::Plain(ref mut target) => {
|
|
|
|
*target = None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn key(&mut self,
|
|
|
|
evqh: &mut EventQueueHandle,
|
|
|
|
proxy: &wl_keyboard::WlKeyboard,
|
|
|
|
serial: u32,
|
|
|
|
time: u32,
|
|
|
|
key: u32,
|
|
|
|
state: wl_keyboard::KeyState)
|
|
|
|
{
|
|
|
|
match self.kbd_handler {
|
|
|
|
KbdType::Mapped(ref mut h) => h.key(evqh, proxy, serial, time, key, state),
|
|
|
|
KbdType::Plain(Some(wid)) => {
|
|
|
|
let state = match state {
|
|
|
|
wl_keyboard::KeyState::Pressed => ElementState::Pressed,
|
|
|
|
wl_keyboard::KeyState::Released => ElementState::Released,
|
|
|
|
};
|
|
|
|
// This is fallback impl if libxkbcommon was not available
|
|
|
|
// This case should probably never happen, as most wayland
|
|
|
|
// compositors _need_ libxkbcommon anyway...
|
|
|
|
//
|
|
|
|
// In this case, we don't have the modifiers state information
|
|
|
|
// anyway, as we need libxkbcommon to interpret it (it is
|
|
|
|
// supposed to be serialized by the compositor using libxkbcommon)
|
|
|
|
self.callback.lock().unwrap().send_event(
|
2017-04-23 06:52:35 +10:00
|
|
|
Event::KeyboardInput {
|
|
|
|
device_id: ::DeviceId(::platform::DeviceId::Wayland(DeviceId)),
|
|
|
|
input: KeyboardInput {
|
|
|
|
state: state,
|
|
|
|
scancode: key,
|
|
|
|
virtual_keycode: None,
|
|
|
|
modifiers: ModifiersState::default(),
|
|
|
|
},
|
|
|
|
},
|
2017-03-11 09:22:59 +11:00
|
|
|
wid
|
|
|
|
);
|
|
|
|
},
|
|
|
|
KbdType::Plain(None) => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn modifiers(&mut self,
|
|
|
|
evqh: &mut EventQueueHandle,
|
|
|
|
proxy: &wl_keyboard::WlKeyboard,
|
|
|
|
serial: u32,
|
|
|
|
mods_depressed: u32,
|
|
|
|
mods_latched: u32,
|
|
|
|
mods_locked: u32,
|
|
|
|
group: u32)
|
|
|
|
{
|
|
|
|
match self.kbd_handler {
|
|
|
|
KbdType::Mapped(ref mut h) => h.modifiers(evqh, proxy, serial, mods_depressed,
|
|
|
|
mods_latched, mods_locked, group),
|
|
|
|
_ => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn repeat_info(&mut self,
|
|
|
|
evqh: &mut EventQueueHandle,
|
|
|
|
proxy: &wl_keyboard::WlKeyboard,
|
|
|
|
rate: i32,
|
|
|
|
delay: i32)
|
|
|
|
{
|
|
|
|
match self.kbd_handler {
|
|
|
|
KbdType::Mapped(ref mut h) => h.repeat_info(evqh, proxy, rate, delay),
|
|
|
|
_ => ()
|
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
|
|
|
declare_handler!(InputHandler, wl_keyboard::Handler, wl_keyboard::WlKeyboard);
|