2019-06-22 01:33:15 +10:00
|
|
|
use std::{
|
|
|
|
cell::RefCell,
|
|
|
|
collections::VecDeque,
|
|
|
|
fmt,
|
|
|
|
rc::Rc,
|
|
|
|
sync::{Arc, Mutex},
|
|
|
|
time::Instant,
|
|
|
|
};
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
use smithay_client_toolkit::reexports::protocols::unstable::relative_pointer::v1::client::{
|
|
|
|
zwp_relative_pointer_manager_v1::ZwpRelativePointerManagerV1,
|
|
|
|
zwp_relative_pointer_v1::ZwpRelativePointerV1,
|
|
|
|
};
|
|
|
|
|
2019-06-22 01:33:15 +10:00
|
|
|
use crate::{
|
|
|
|
dpi::{PhysicalPosition, PhysicalSize},
|
|
|
|
event::ModifiersState,
|
|
|
|
event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootELW},
|
2019-07-30 04:16:14 +10:00
|
|
|
monitor::{MonitorHandle as RootMonitorHandle, VideoMode as RootVideoMode},
|
|
|
|
platform_impl::platform::{
|
|
|
|
sticky_exit_callback, MonitorHandle as PlatformMonitorHandle,
|
|
|
|
VideoMode as PlatformVideoMode,
|
|
|
|
},
|
2019-06-22 01:33:15 +10:00
|
|
|
};
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
use super::{window::WindowStore, DeviceId, WindowId};
|
2019-06-22 01:33:15 +10:00
|
|
|
|
|
|
|
use smithay_client_toolkit::{
|
|
|
|
output::OutputMgr,
|
|
|
|
reexports::client::{
|
|
|
|
protocol::{wl_keyboard, wl_output, wl_pointer, wl_registry, wl_seat, wl_touch},
|
|
|
|
ConnectError, Display, EventQueue, GlobalEvent,
|
|
|
|
},
|
|
|
|
Environment,
|
2018-08-20 07:17:40 +10:00
|
|
|
};
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
pub struct WindowEventsSink<T> {
|
|
|
|
buffer: VecDeque<crate::event::Event<T>>,
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
impl<T> WindowEventsSink<T> {
|
|
|
|
pub fn new() -> WindowEventsSink<T> {
|
2019-02-21 20:51:43 +11:00
|
|
|
WindowEventsSink {
|
2018-05-06 03:36:34 +10:00
|
|
|
buffer: VecDeque::new(),
|
2017-09-28 00:31:46 +10:00
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
pub fn send_window_event(&mut self, evt: crate::event::WindowEvent, wid: WindowId) {
|
|
|
|
self.buffer.push_back(crate::event::Event::WindowEvent {
|
|
|
|
event: evt,
|
|
|
|
window_id: crate::window::WindowId(crate::platform_impl::WindowId::Wayland(wid)),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn send_device_event(&mut self, evt: crate::event::DeviceEvent, dev_id: DeviceId) {
|
|
|
|
self.buffer.push_back(crate::event::Event::DeviceEvent {
|
|
|
|
event: evt,
|
|
|
|
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(dev_id)),
|
|
|
|
});
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
fn empty_with<F>(&mut self, mut callback: F)
|
2018-05-06 03:36:34 +10:00
|
|
|
where
|
2019-06-18 04:27:00 +10:00
|
|
|
F: FnMut(crate::event::Event<T>),
|
2018-05-06 03:36:34 +10:00
|
|
|
{
|
2019-06-25 19:00:41 +10:00
|
|
|
for evt in self.buffer.drain(..) {
|
|
|
|
callback(evt)
|
2017-09-28 00:31:46 +10:00
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
pub struct EventLoop<T: 'static> {
|
|
|
|
// The loop
|
|
|
|
inner_loop: ::calloop::EventLoop<()>,
|
|
|
|
// The wayland display
|
|
|
|
pub display: Arc<Display>,
|
|
|
|
// the output manager
|
|
|
|
pub outputs: OutputMgr,
|
2017-09-28 00:31:46 +10:00
|
|
|
// our sink, shared with some handlers, buffering the events
|
2019-06-25 19:00:41 +10:00
|
|
|
sink: Arc<Mutex<WindowEventsSink<T>>>,
|
2019-02-21 20:51:43 +11:00
|
|
|
pending_user_events: Rc<RefCell<VecDeque<T>>>,
|
|
|
|
_user_source: ::calloop::Source<::calloop::channel::Channel<T>>,
|
|
|
|
user_sender: ::calloop::channel::Sender<T>,
|
2019-06-22 01:33:15 +10:00
|
|
|
_kbd_source: ::calloop::Source<
|
|
|
|
::calloop::channel::Channel<(crate::event::WindowEvent, super::WindowId)>,
|
|
|
|
>,
|
|
|
|
window_target: RootELW<T>,
|
2019-02-21 20:51:43 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// A handle that can be sent across threads and used to wake up the `EventLoop`.
|
|
|
|
//
|
|
|
|
// We should only try and wake up the `EventLoop` if it still exists, so we hold Weak ptrs.
|
|
|
|
pub struct EventLoopProxy<T: 'static> {
|
2019-06-22 01:33:15 +10:00
|
|
|
user_sender: ::calloop::channel::Sender<T>,
|
2019-02-21 20:51:43 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct EventLoopWindowTarget<T> {
|
|
|
|
// the event queue
|
|
|
|
pub evq: RefCell<::calloop::Source<EventQueue>>,
|
2017-10-20 18:46:42 +11:00
|
|
|
// The window store
|
2018-05-06 03:36:34 +10:00
|
|
|
pub store: Arc<Mutex<WindowStore>>,
|
2017-10-20 18:46:42 +11:00
|
|
|
// the env
|
2018-05-06 03:36:34 +10:00
|
|
|
pub env: Environment,
|
2017-10-20 18:46:42 +11:00
|
|
|
// a cleanup switch to prune dead windows
|
2017-10-26 03:28:24 +11:00
|
|
|
pub cleanup_needed: Arc<Mutex<bool>>,
|
|
|
|
// The wayland display
|
2018-05-06 03:36:34 +10:00
|
|
|
pub display: Arc<Display>,
|
|
|
|
// The list of seats
|
2019-02-21 20:51:43 +11:00
|
|
|
pub seats: Arc<Mutex<Vec<(u32, wl_seat::WlSeat)>>>,
|
2019-06-22 01:33:15 +10:00
|
|
|
_marker: ::std::marker::PhantomData<T>,
|
2017-05-25 23:19:13 +10:00
|
|
|
}
|
|
|
|
|
2019-08-06 06:51:42 +10:00
|
|
|
impl<T: 'static> Clone for EventLoopProxy<T> {
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
EventLoopProxy {
|
|
|
|
user_sender: self.user_sender.clone(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
impl<T: 'static> EventLoopProxy<T> {
|
|
|
|
pub fn send_event(&self, event: T) -> Result<(), EventLoopClosed> {
|
|
|
|
self.user_sender.send(event).map_err(|_| EventLoopClosed)
|
2017-05-25 23:19:13 +10:00
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
impl<T: 'static> EventLoop<T> {
|
|
|
|
pub fn new() -> Result<EventLoop<T>, ConnectError> {
|
2018-05-12 21:58:11 +10:00
|
|
|
let (display, mut event_queue) = Display::connect_to_env()?;
|
2017-10-20 18:46:42 +11:00
|
|
|
|
2018-08-20 07:17:40 +10:00
|
|
|
let display = Arc::new(display);
|
2019-02-21 20:51:43 +11:00
|
|
|
let sink = Arc::new(Mutex::new(WindowEventsSink::new()));
|
2018-05-06 03:36:34 +10:00
|
|
|
let store = Arc::new(Mutex::new(WindowStore::new()));
|
|
|
|
let seats = Arc::new(Mutex::new(Vec::new()));
|
2017-09-28 00:31:46 +10:00
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
let inner_loop = ::calloop::EventLoop::new().unwrap();
|
|
|
|
|
|
|
|
let (kbd_sender, kbd_channel) = ::calloop::channel::channel();
|
|
|
|
let kbd_sink = sink.clone();
|
2019-06-22 01:33:15 +10:00
|
|
|
let kbd_source = inner_loop
|
|
|
|
.handle()
|
|
|
|
.insert_source(kbd_channel, move |evt, &mut ()| {
|
|
|
|
if let ::calloop::channel::Event::Msg((evt, wid)) = evt {
|
2019-06-25 19:00:41 +10:00
|
|
|
kbd_sink.lock().unwrap().send_window_event(evt, wid);
|
2019-06-22 01:33:15 +10:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.unwrap();
|
2019-02-21 20:51:43 +11:00
|
|
|
|
2018-10-15 10:15:43 +11:00
|
|
|
let mut seat_manager = SeatManager {
|
|
|
|
sink: sink.clone(),
|
2019-06-25 19:00:41 +10:00
|
|
|
relative_pointer_manager_proxy: None,
|
2018-10-15 10:15:43 +11:00
|
|
|
store: store.clone(),
|
|
|
|
seats: seats.clone(),
|
2019-02-21 20:51:43 +11:00
|
|
|
kbd_sender,
|
2018-10-15 10:15:43 +11:00
|
|
|
};
|
|
|
|
|
|
|
|
let env = Environment::from_display_with_cb(
|
|
|
|
&display,
|
2018-05-06 03:36:34 +10:00
|
|
|
&mut event_queue,
|
2019-06-25 02:14:55 +10:00
|
|
|
move |event, registry| match event {
|
|
|
|
GlobalEvent::New {
|
|
|
|
id,
|
|
|
|
ref interface,
|
|
|
|
version,
|
|
|
|
} => {
|
2019-06-25 19:00:41 +10:00
|
|
|
if interface == "zwp_relative_pointer_manager_v1" {
|
|
|
|
seat_manager.relative_pointer_manager_proxy = Some(
|
|
|
|
registry
|
|
|
|
.bind(version, id, move |pointer_manager| {
|
|
|
|
pointer_manager.implement_closure(|_, _| (), ())
|
|
|
|
})
|
|
|
|
.unwrap(),
|
|
|
|
)
|
|
|
|
}
|
2019-06-25 02:14:55 +10:00
|
|
|
if interface == "wl_seat" {
|
|
|
|
seat_manager.add_seat(id, version, registry)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
GlobalEvent::Removed { id, ref interface } => {
|
|
|
|
if interface == "wl_seat" {
|
|
|
|
seat_manager.remove_seat(id)
|
|
|
|
}
|
2018-11-21 07:21:58 +11:00
|
|
|
}
|
2018-05-06 03:36:34 +10:00
|
|
|
},
|
2019-06-22 01:33:15 +10:00
|
|
|
)
|
|
|
|
.unwrap();
|
2017-10-20 18:46:42 +11:00
|
|
|
|
2019-06-22 01:33:15 +10:00
|
|
|
let source = inner_loop
|
|
|
|
.handle()
|
|
|
|
.insert_source(event_queue, |(), &mut ()| {})
|
|
|
|
.unwrap();
|
2019-02-21 20:51:43 +11:00
|
|
|
|
|
|
|
let pending_user_events = Rc::new(RefCell::new(VecDeque::new()));
|
|
|
|
let pending_user_events2 = pending_user_events.clone();
|
|
|
|
|
|
|
|
let (user_sender, user_channel) = ::calloop::channel::channel();
|
|
|
|
|
2019-06-22 01:33:15 +10:00
|
|
|
let user_source = inner_loop
|
|
|
|
.handle()
|
|
|
|
.insert_source(user_channel, move |evt, &mut ()| {
|
|
|
|
if let ::calloop::channel::Event::Msg(msg) = evt {
|
|
|
|
pending_user_events2.borrow_mut().push_back(msg);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.unwrap();
|
2019-02-21 20:51:43 +11:00
|
|
|
|
2019-02-06 02:30:33 +11:00
|
|
|
Ok(EventLoop {
|
2019-02-21 20:51:43 +11:00
|
|
|
inner_loop,
|
2018-08-20 07:17:40 +10:00
|
|
|
sink,
|
2019-02-21 20:51:43 +11:00
|
|
|
pending_user_events,
|
|
|
|
display: display.clone(),
|
|
|
|
outputs: env.outputs.clone(),
|
|
|
|
_user_source: user_source,
|
|
|
|
user_sender,
|
|
|
|
_kbd_source: kbd_source,
|
|
|
|
window_target: RootELW {
|
2019-06-18 04:27:00 +10:00
|
|
|
p: crate::platform_impl::EventLoopWindowTarget::Wayland(EventLoopWindowTarget {
|
2019-02-21 20:51:43 +11:00
|
|
|
evq: RefCell::new(source),
|
|
|
|
store,
|
|
|
|
env,
|
|
|
|
cleanup_needed: Arc::new(Mutex::new(false)),
|
|
|
|
seats,
|
|
|
|
display,
|
2019-06-22 01:33:15 +10:00
|
|
|
_marker: ::std::marker::PhantomData,
|
2019-02-21 20:51:43 +11:00
|
|
|
}),
|
2019-06-22 01:33:15 +10:00
|
|
|
_marker: ::std::marker::PhantomData,
|
|
|
|
},
|
2018-05-06 03:36:34 +10:00
|
|
|
})
|
2017-09-01 19:04:57 +10:00
|
|
|
}
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
pub fn create_proxy(&self) -> EventLoopProxy<T> {
|
2019-02-06 02:30:33 +11:00
|
|
|
EventLoopProxy {
|
2019-06-22 01:33:15 +10:00
|
|
|
user_sender: self.user_sender.clone(),
|
2017-05-25 23:19:13 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
pub fn run<F>(mut self, callback: F) -> !
|
2019-06-22 01:33:15 +10:00
|
|
|
where
|
|
|
|
F: 'static + FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow),
|
2017-03-05 00:04:01 +11:00
|
|
|
{
|
2019-02-21 20:51:43 +11:00
|
|
|
self.run_return(callback);
|
|
|
|
::std::process::exit(0);
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
pub fn run_return<F>(&mut self, mut callback: F)
|
2019-06-22 01:33:15 +10:00
|
|
|
where
|
|
|
|
F: FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow),
|
2017-03-05 00:04:01 +11:00
|
|
|
{
|
2017-10-20 18:46:42 +11:00
|
|
|
// send pending events to the server
|
|
|
|
self.display.flush().expect("Wayland connection lost.");
|
2017-03-05 00:04:01 +11:00
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
let mut control_flow = ControlFlow::default();
|
|
|
|
|
|
|
|
let sink = self.sink.clone();
|
|
|
|
let user_events = self.pending_user_events.clone();
|
2017-06-02 21:19:45 +10:00
|
|
|
|
2019-06-22 01:33:15 +10:00
|
|
|
callback(
|
|
|
|
crate::event::Event::NewEvents(crate::event::StartCause::Init),
|
|
|
|
&self.window_target,
|
|
|
|
&mut control_flow,
|
|
|
|
);
|
2017-03-05 00:04:01 +11:00
|
|
|
|
2017-06-02 21:19:45 +10:00
|
|
|
loop {
|
2017-09-28 00:31:46 +10:00
|
|
|
self.post_dispatch_triggers();
|
2017-05-25 23:19:13 +10:00
|
|
|
|
2017-09-28 00:31:46 +10:00
|
|
|
// empty buffer of events
|
2019-02-21 20:51:43 +11:00
|
|
|
{
|
|
|
|
let mut guard = sink.lock().unwrap();
|
2019-04-28 02:06:51 +10:00
|
|
|
guard.empty_with(|evt| {
|
2019-06-22 01:33:15 +10:00
|
|
|
sticky_exit_callback(
|
|
|
|
evt,
|
|
|
|
&self.window_target,
|
|
|
|
&mut control_flow,
|
|
|
|
&mut callback,
|
|
|
|
);
|
2019-04-28 02:06:51 +10:00
|
|
|
});
|
2019-02-21 20:51:43 +11:00
|
|
|
}
|
|
|
|
// empty user events
|
|
|
|
{
|
|
|
|
let mut guard = user_events.borrow_mut();
|
|
|
|
for evt in guard.drain(..) {
|
2019-04-28 02:06:51 +10:00
|
|
|
sticky_exit_callback(
|
2019-06-18 04:27:00 +10:00
|
|
|
crate::event::Event::UserEvent(evt),
|
2019-04-28 02:06:51 +10:00
|
|
|
&self.window_target,
|
|
|
|
&mut control_flow,
|
2019-06-22 01:33:15 +10:00
|
|
|
&mut callback,
|
2019-04-28 02:06:51 +10:00
|
|
|
);
|
2019-02-21 20:51:43 +11:00
|
|
|
}
|
|
|
|
}
|
2019-04-28 02:06:51 +10:00
|
|
|
// do a second run of post-dispatch-triggers, to handle user-generated "request-redraw"
|
|
|
|
// in response of resize & friends
|
2019-02-21 20:51:43 +11:00
|
|
|
self.post_dispatch_triggers();
|
|
|
|
{
|
|
|
|
let mut guard = sink.lock().unwrap();
|
2019-04-28 02:06:51 +10:00
|
|
|
guard.empty_with(|evt| {
|
2019-06-22 01:33:15 +10:00
|
|
|
sticky_exit_callback(
|
|
|
|
evt,
|
|
|
|
&self.window_target,
|
|
|
|
&mut control_flow,
|
|
|
|
&mut callback,
|
|
|
|
);
|
2019-04-28 02:06:51 +10:00
|
|
|
});
|
|
|
|
}
|
|
|
|
// send Events cleared
|
|
|
|
{
|
|
|
|
sticky_exit_callback(
|
2019-06-18 04:27:00 +10:00
|
|
|
crate::event::Event::EventsCleared,
|
2019-04-28 02:06:51 +10:00
|
|
|
&self.window_target,
|
|
|
|
&mut control_flow,
|
2019-06-22 01:33:15 +10:00
|
|
|
&mut callback,
|
2019-04-28 02:06:51 +10:00
|
|
|
);
|
2019-02-21 20:51:43 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// send pending events to the server
|
|
|
|
self.display.flush().expect("Wayland connection lost.");
|
|
|
|
|
2019-09-11 16:28:21 +10:00
|
|
|
// During the run of the user callback, some other code monitoring and reading the
|
|
|
|
// wayland socket may have been run (mesa for example does this with vsync), if that
|
|
|
|
// is the case, some events may have been enqueued in our event queue.
|
|
|
|
//
|
|
|
|
// If some messages are there, the event loop needs to behave as if it was instantly
|
|
|
|
// woken up by messages arriving from the wayland socket, to avoid getting stuck.
|
|
|
|
let instant_wakeup = {
|
|
|
|
let window_target = match self.window_target.p {
|
|
|
|
crate::platform_impl::EventLoopWindowTarget::Wayland(ref wt) => wt,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
let dispatched = window_target
|
|
|
|
.evq
|
|
|
|
.borrow_mut()
|
|
|
|
.dispatch_pending()
|
|
|
|
.expect("Wayland connection lost.");
|
|
|
|
dispatched > 0
|
|
|
|
};
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
match control_flow {
|
|
|
|
ControlFlow::Exit => break,
|
|
|
|
ControlFlow::Poll => {
|
|
|
|
// non-blocking dispatch
|
2019-06-22 01:33:15 +10:00
|
|
|
self.inner_loop
|
|
|
|
.dispatch(Some(::std::time::Duration::from_millis(0)), &mut ())
|
|
|
|
.unwrap();
|
|
|
|
callback(
|
|
|
|
crate::event::Event::NewEvents(crate::event::StartCause::Poll),
|
|
|
|
&self.window_target,
|
|
|
|
&mut control_flow,
|
|
|
|
);
|
2019-06-25 02:14:55 +10:00
|
|
|
}
|
2019-02-21 20:51:43 +11:00
|
|
|
ControlFlow::Wait => {
|
2019-09-11 16:28:21 +10:00
|
|
|
let timeout = if instant_wakeup {
|
|
|
|
Some(::std::time::Duration::from_millis(0))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
self.inner_loop.dispatch(timeout, &mut ()).unwrap();
|
2019-02-21 20:51:43 +11:00
|
|
|
callback(
|
2019-06-18 04:27:00 +10:00
|
|
|
crate::event::Event::NewEvents(crate::event::StartCause::WaitCancelled {
|
2019-02-21 20:51:43 +11:00
|
|
|
start: Instant::now(),
|
2019-06-22 01:33:15 +10:00
|
|
|
requested_resume: None,
|
2019-02-21 20:51:43 +11:00
|
|
|
}),
|
|
|
|
&self.window_target,
|
2019-06-22 01:33:15 +10:00
|
|
|
&mut control_flow,
|
2019-02-21 20:51:43 +11:00
|
|
|
);
|
2019-06-25 02:14:55 +10:00
|
|
|
}
|
2019-02-21 20:51:43 +11:00
|
|
|
ControlFlow::WaitUntil(deadline) => {
|
|
|
|
let start = Instant::now();
|
|
|
|
// compute the blocking duration
|
2019-09-11 16:28:21 +10:00
|
|
|
let duration = if deadline > start && !instant_wakeup {
|
2019-05-30 14:33:52 +10:00
|
|
|
deadline - start
|
|
|
|
} else {
|
|
|
|
::std::time::Duration::from_millis(0)
|
|
|
|
};
|
2019-02-21 20:51:43 +11:00
|
|
|
self.inner_loop.dispatch(Some(duration), &mut ()).unwrap();
|
|
|
|
let now = Instant::now();
|
|
|
|
if now < deadline {
|
|
|
|
callback(
|
2019-06-22 01:33:15 +10:00
|
|
|
crate::event::Event::NewEvents(
|
|
|
|
crate::event::StartCause::WaitCancelled {
|
|
|
|
start,
|
|
|
|
requested_resume: Some(deadline),
|
|
|
|
},
|
|
|
|
),
|
2019-02-21 20:51:43 +11:00
|
|
|
&self.window_target,
|
2019-06-22 01:33:15 +10:00
|
|
|
&mut control_flow,
|
2019-02-21 20:51:43 +11:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
callback(
|
2019-06-22 01:33:15 +10:00
|
|
|
crate::event::Event::NewEvents(
|
|
|
|
crate::event::StartCause::ResumeTimeReached {
|
|
|
|
start,
|
|
|
|
requested_resume: deadline,
|
|
|
|
},
|
|
|
|
),
|
2019-02-21 20:51:43 +11:00
|
|
|
&self.window_target,
|
2019-06-22 01:33:15 +10:00
|
|
|
&mut control_flow,
|
2019-02-21 20:51:43 +11:00
|
|
|
);
|
|
|
|
}
|
2019-06-25 02:14:55 +10:00
|
|
|
}
|
2017-06-02 21:19:45 +10:00
|
|
|
}
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
2019-02-21 20:51:43 +11:00
|
|
|
|
2019-06-22 01:33:15 +10:00
|
|
|
callback(
|
|
|
|
crate::event::Event::LoopDestroyed,
|
|
|
|
&self.window_target,
|
|
|
|
&mut control_flow,
|
|
|
|
);
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
2017-05-25 23:19:13 +10:00
|
|
|
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn primary_monitor(&self) -> MonitorHandle {
|
|
|
|
primary_monitor(&self.outputs)
|
2017-05-25 23:19:13 +10:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn available_monitors(&self) -> VecDeque<MonitorHandle> {
|
|
|
|
available_monitors(&self.outputs)
|
2019-02-21 20:51:43 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn window_target(&self) -> &RootELW<T> {
|
|
|
|
&self.window_target
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
|
2019-08-09 07:50:22 +10:00
|
|
|
impl<T> EventLoopWindowTarget<T> {
|
|
|
|
pub fn display(&self) -> &Display {
|
|
|
|
&*self.display
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-20 18:46:42 +11:00
|
|
|
/*
|
2019-02-06 02:30:33 +11:00
|
|
|
* Private EventLoop Internals
|
2017-10-20 18:46:42 +11:00
|
|
|
*/
|
|
|
|
|
2019-02-21 20:51:43 +11:00
|
|
|
impl<T> EventLoop<T> {
|
2017-10-20 18:46:42 +11:00
|
|
|
fn post_dispatch_triggers(&mut self) {
|
|
|
|
let mut sink = self.sink.lock().unwrap();
|
2019-02-21 20:51:43 +11:00
|
|
|
let window_target = match self.window_target.p {
|
2019-06-18 04:27:00 +10:00
|
|
|
crate::platform_impl::EventLoopWindowTarget::Wayland(ref wt) => wt,
|
2019-06-22 01:33:15 +10:00
|
|
|
_ => unreachable!(),
|
2019-02-21 20:51:43 +11:00
|
|
|
};
|
2017-10-20 18:46:42 +11:00
|
|
|
// prune possible dead windows
|
|
|
|
{
|
2019-02-21 20:51:43 +11:00
|
|
|
let mut cleanup_needed = window_target.cleanup_needed.lock().unwrap();
|
2017-10-20 18:46:42 +11:00
|
|
|
if *cleanup_needed {
|
2019-02-21 20:51:43 +11:00
|
|
|
let pruned = window_target.store.lock().unwrap().cleanup();
|
2017-10-20 18:46:42 +11:00
|
|
|
*cleanup_needed = false;
|
2018-04-25 06:20:40 +10:00
|
|
|
for wid in pruned {
|
2019-06-25 19:00:41 +10:00
|
|
|
sink.send_window_event(crate::event::WindowEvent::Destroyed, wid);
|
2018-04-25 06:20:40 +10:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
// process pending resize/refresh
|
2019-02-21 20:51:43 +11:00
|
|
|
window_target.store.lock().unwrap().for_each(
|
2018-06-15 09:42:18 +10:00
|
|
|
|newsize, size, new_dpi, refresh, frame_refresh, closed, wid, frame| {
|
2017-11-04 03:35:29 +11:00
|
|
|
if let Some(frame) = frame {
|
|
|
|
if let Some((w, h)) = newsize {
|
2018-06-15 09:42:18 +10:00
|
|
|
frame.resize(w, h);
|
2017-11-04 03:35:29 +11:00
|
|
|
frame.refresh();
|
2019-06-18 04:27:00 +10:00
|
|
|
let logical_size = crate::dpi::LogicalSize::new(w as f64, h as f64);
|
2019-06-25 19:00:41 +10:00
|
|
|
sink.send_window_event(
|
|
|
|
crate::event::WindowEvent::Resized(logical_size),
|
|
|
|
wid,
|
|
|
|
);
|
2018-06-15 09:42:18 +10:00
|
|
|
*size = (w, h);
|
2017-11-04 03:35:29 +11:00
|
|
|
} else if frame_refresh {
|
|
|
|
frame.refresh();
|
2018-08-24 03:20:02 +10:00
|
|
|
if !refresh {
|
|
|
|
frame.surface().commit()
|
|
|
|
}
|
2017-11-04 03:35:29 +11:00
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2018-06-15 09:42:18 +10:00
|
|
|
if let Some(dpi) = new_dpi {
|
2019-06-25 19:00:41 +10:00
|
|
|
sink.send_window_event(
|
2019-06-22 01:33:15 +10:00
|
|
|
crate::event::WindowEvent::HiDpiFactorChanged(dpi as f64),
|
|
|
|
wid,
|
|
|
|
);
|
2018-06-15 09:42:18 +10:00
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
if refresh {
|
2019-06-25 19:00:41 +10:00
|
|
|
sink.send_window_event(crate::event::WindowEvent::RedrawRequested, wid);
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
|
|
|
if closed {
|
2019-06-25 19:00:41 +10:00
|
|
|
sink.send_window_event(crate::event::WindowEvent::CloseRequested, wid);
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2018-05-06 03:36:34 +10:00
|
|
|
},
|
2017-10-20 18:46:42 +11:00
|
|
|
)
|
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2017-10-20 18:46:42 +11:00
|
|
|
* Wayland protocol implementations
|
2017-03-11 09:22:59 +11:00
|
|
|
*/
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
struct SeatManager<T: 'static> {
|
|
|
|
sink: Arc<Mutex<WindowEventsSink<T>>>,
|
2018-05-06 03:36:34 +10:00
|
|
|
store: Arc<Mutex<WindowStore>>,
|
2019-02-21 20:51:43 +11:00
|
|
|
seats: Arc<Mutex<Vec<(u32, wl_seat::WlSeat)>>>,
|
2019-06-22 01:33:15 +10:00
|
|
|
kbd_sender: ::calloop::channel::Sender<(crate::event::WindowEvent, super::WindowId)>,
|
2019-06-25 19:00:41 +10:00
|
|
|
relative_pointer_manager_proxy: Option<ZwpRelativePointerManagerV1>,
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
impl<T: 'static> SeatManager<T> {
|
2019-02-21 20:51:43 +11:00
|
|
|
fn add_seat(&mut self, id: u32, version: u32, registry: wl_registry::WlRegistry) {
|
2018-11-21 07:21:58 +11:00
|
|
|
use std::cmp::min;
|
|
|
|
|
|
|
|
let mut seat_data = SeatData {
|
|
|
|
sink: self.sink.clone(),
|
|
|
|
store: self.store.clone(),
|
|
|
|
pointer: None,
|
2019-06-25 19:00:41 +10:00
|
|
|
relative_pointer: None,
|
|
|
|
relative_pointer_manager_proxy: self.relative_pointer_manager_proxy.as_ref().cloned(),
|
2018-11-21 07:21:58 +11:00
|
|
|
keyboard: None,
|
|
|
|
touch: None,
|
2019-02-21 20:51:43 +11:00
|
|
|
kbd_sender: self.kbd_sender.clone(),
|
2018-11-21 07:21:58 +11:00
|
|
|
modifiers_tracker: Arc::new(Mutex::new(ModifiersState::default())),
|
|
|
|
};
|
|
|
|
let seat = registry
|
|
|
|
.bind(min(version, 5), id, move |seat| {
|
2019-06-22 01:33:15 +10:00
|
|
|
seat.implement_closure(move |event, seat| seat_data.receive(event, seat), ())
|
2018-11-21 07:21:58 +11:00
|
|
|
})
|
|
|
|
.unwrap();
|
|
|
|
self.store.lock().unwrap().new_seat(&seat);
|
|
|
|
self.seats.lock().unwrap().push((id, seat));
|
|
|
|
}
|
|
|
|
|
|
|
|
fn remove_seat(&mut self, id: u32) {
|
|
|
|
let mut seats = self.seats.lock().unwrap();
|
|
|
|
if let Some(idx) = seats.iter().position(|&(i, _)| i == id) {
|
|
|
|
let (_, seat) = seats.swap_remove(idx);
|
2019-02-21 20:51:43 +11:00
|
|
|
if seat.as_ref().version() >= 5 {
|
2018-11-21 07:21:58 +11:00
|
|
|
seat.release();
|
2018-05-06 03:36:34 +10:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
struct SeatData<T> {
|
|
|
|
sink: Arc<Mutex<WindowEventsSink<T>>>,
|
2018-05-06 03:36:34 +10:00
|
|
|
store: Arc<Mutex<WindowStore>>,
|
2019-06-18 04:27:00 +10:00
|
|
|
kbd_sender: ::calloop::channel::Sender<(crate::event::WindowEvent, super::WindowId)>,
|
2019-02-21 20:51:43 +11:00
|
|
|
pointer: Option<wl_pointer::WlPointer>,
|
2019-06-25 19:00:41 +10:00
|
|
|
relative_pointer: Option<ZwpRelativePointerV1>,
|
|
|
|
relative_pointer_manager_proxy: Option<ZwpRelativePointerManagerV1>,
|
2019-02-21 20:51:43 +11:00
|
|
|
keyboard: Option<wl_keyboard::WlKeyboard>,
|
|
|
|
touch: Option<wl_touch::WlTouch>,
|
2018-10-18 13:34:02 +11:00
|
|
|
modifiers_tracker: Arc<Mutex<ModifiersState>>,
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
impl<T: 'static> SeatData<T> {
|
2019-02-21 20:51:43 +11:00
|
|
|
fn receive(&mut self, evt: wl_seat::Event, seat: wl_seat::WlSeat) {
|
2018-05-06 03:36:34 +10:00
|
|
|
match evt {
|
|
|
|
wl_seat::Event::Name { .. } => (),
|
|
|
|
wl_seat::Event::Capabilities { capabilities } => {
|
|
|
|
// create pointer if applicable
|
|
|
|
if capabilities.contains(wl_seat::Capability::Pointer) && self.pointer.is_none() {
|
|
|
|
self.pointer = Some(super::pointer::implement_pointer(
|
2018-10-15 10:15:43 +11:00
|
|
|
&seat,
|
2018-05-06 03:36:34 +10:00
|
|
|
self.sink.clone(),
|
|
|
|
self.store.clone(),
|
2018-10-18 13:34:02 +11:00
|
|
|
self.modifiers_tracker.clone(),
|
2019-06-25 19:00:41 +10:00
|
|
|
));
|
|
|
|
|
|
|
|
self.relative_pointer =
|
|
|
|
self.relative_pointer_manager_proxy
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|manager| {
|
|
|
|
super::pointer::implement_relative_pointer(
|
|
|
|
self.sink.clone(),
|
|
|
|
self.pointer.as_ref().unwrap(),
|
|
|
|
manager,
|
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
})
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2018-05-06 03:36:34 +10:00
|
|
|
// destroy pointer if applicable
|
|
|
|
if !capabilities.contains(wl_seat::Capability::Pointer) {
|
|
|
|
if let Some(pointer) = self.pointer.take() {
|
2019-02-21 20:51:43 +11:00
|
|
|
if pointer.as_ref().version() >= 3 {
|
2018-05-06 03:36:34 +10:00
|
|
|
pointer.release();
|
|
|
|
}
|
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2018-05-06 03:36:34 +10:00
|
|
|
// create keyboard if applicable
|
|
|
|
if capabilities.contains(wl_seat::Capability::Keyboard) && self.keyboard.is_none() {
|
|
|
|
self.keyboard = Some(super::keyboard::init_keyboard(
|
2018-10-15 10:15:43 +11:00
|
|
|
&seat,
|
2019-02-21 20:51:43 +11:00
|
|
|
self.kbd_sender.clone(),
|
2018-10-18 13:34:02 +11:00
|
|
|
self.modifiers_tracker.clone(),
|
2018-05-06 03:36:34 +10:00
|
|
|
))
|
|
|
|
}
|
|
|
|
// destroy keyboard if applicable
|
|
|
|
if !capabilities.contains(wl_seat::Capability::Keyboard) {
|
|
|
|
if let Some(kbd) = self.keyboard.take() {
|
2019-02-21 20:51:43 +11:00
|
|
|
if kbd.as_ref().version() >= 3 {
|
2018-05-06 03:36:34 +10:00
|
|
|
kbd.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// create touch if applicable
|
|
|
|
if capabilities.contains(wl_seat::Capability::Touch) && self.touch.is_none() {
|
|
|
|
self.touch = Some(super::touch::implement_touch(
|
2018-10-15 10:15:43 +11:00
|
|
|
&seat,
|
2018-05-06 03:36:34 +10:00
|
|
|
self.sink.clone(),
|
|
|
|
self.store.clone(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
// destroy touch if applicable
|
|
|
|
if !capabilities.contains(wl_seat::Capability::Touch) {
|
|
|
|
if let Some(touch) = self.touch.take() {
|
2019-02-21 20:51:43 +11:00
|
|
|
if touch.as_ref().version() >= 3 {
|
2018-05-06 03:36:34 +10:00
|
|
|
touch.release();
|
|
|
|
}
|
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2019-06-25 02:14:55 +10:00
|
|
|
}
|
2019-06-22 01:33:15 +10:00
|
|
|
_ => unreachable!(),
|
2017-11-11 20:03:42 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-25 19:00:41 +10:00
|
|
|
impl<T> Drop for SeatData<T> {
|
2018-05-06 03:36:34 +10:00
|
|
|
fn drop(&mut self) {
|
|
|
|
if let Some(pointer) = self.pointer.take() {
|
2019-02-21 20:51:43 +11:00
|
|
|
if pointer.as_ref().version() >= 3 {
|
2018-05-06 03:36:34 +10:00
|
|
|
pointer.release();
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
2018-05-06 03:36:34 +10:00
|
|
|
}
|
|
|
|
if let Some(kbd) = self.keyboard.take() {
|
2019-02-21 20:51:43 +11:00
|
|
|
if kbd.as_ref().version() >= 3 {
|
2018-05-06 03:36:34 +10:00
|
|
|
kbd.release();
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2018-05-06 03:36:34 +10:00
|
|
|
}
|
|
|
|
if let Some(touch) = self.touch.take() {
|
2019-02-21 20:51:43 +11:00
|
|
|
if touch.as_ref().version() >= 3 {
|
2018-05-06 03:36:34 +10:00
|
|
|
touch.release();
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2018-05-06 03:36:34 +10:00
|
|
|
/*
|
|
|
|
* Monitor stuff
|
|
|
|
*/
|
|
|
|
|
2019-07-30 04:16:14 +10:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
|
|
|
pub struct VideoMode {
|
|
|
|
pub(crate) size: (u32, u32),
|
|
|
|
pub(crate) bit_depth: u16,
|
|
|
|
pub(crate) refresh_rate: u16,
|
|
|
|
pub(crate) monitor: MonitorHandle,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl VideoMode {
|
|
|
|
#[inline]
|
|
|
|
pub fn size(&self) -> PhysicalSize {
|
|
|
|
self.size.into()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn bit_depth(&self) -> u16 {
|
|
|
|
self.bit_depth
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn refresh_rate(&self) -> u16 {
|
|
|
|
self.refresh_rate
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn monitor(&self) -> RootMonitorHandle {
|
|
|
|
RootMonitorHandle {
|
|
|
|
inner: PlatformMonitorHandle::Wayland(self.monitor.clone()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
2019-02-06 02:30:33 +11:00
|
|
|
pub struct MonitorHandle {
|
2019-02-21 20:51:43 +11:00
|
|
|
pub(crate) proxy: wl_output::WlOutput,
|
2018-05-06 03:36:34 +10:00
|
|
|
pub(crate) mgr: OutputMgr,
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
|
|
|
|
2019-07-30 04:16:14 +10:00
|
|
|
impl PartialEq for MonitorHandle {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.native_identifier() == other.native_identifier()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for MonitorHandle {}
|
|
|
|
|
|
|
|
impl PartialOrd for MonitorHandle {
|
|
|
|
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
|
|
|
Some(self.cmp(&other))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Ord for MonitorHandle {
|
|
|
|
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
|
|
|
|
self.native_identifier().cmp(&other.native_identifier())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl std::hash::Hash for MonitorHandle {
|
|
|
|
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
|
|
|
|
self.native_identifier().hash(state);
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2019-02-06 02:30:33 +11:00
|
|
|
impl fmt::Debug for MonitorHandle {
|
2019-06-18 04:27:00 +10:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2018-05-14 22:14:57 +10:00
|
|
|
#[derive(Debug)]
|
2019-02-06 02:30:33 +11:00
|
|
|
struct MonitorHandle {
|
2018-05-14 22:14:57 +10:00
|
|
|
name: Option<String>,
|
|
|
|
native_identifier: u32,
|
2019-06-13 16:33:44 +10:00
|
|
|
size: PhysicalSize,
|
2018-06-15 09:42:18 +10:00
|
|
|
position: PhysicalPosition,
|
|
|
|
hidpi_factor: i32,
|
2018-05-14 22:14:57 +10:00
|
|
|
}
|
|
|
|
|
2019-02-06 02:30:33 +11:00
|
|
|
let monitor_id_proxy = MonitorHandle {
|
2019-05-30 11:29:54 +10:00
|
|
|
name: self.name(),
|
|
|
|
native_identifier: self.native_identifier(),
|
2019-06-13 16:33:44 +10:00
|
|
|
size: self.size(),
|
2019-05-30 11:29:54 +10:00
|
|
|
position: self.position(),
|
|
|
|
hidpi_factor: self.hidpi_factor(),
|
2018-05-14 22:14:57 +10:00
|
|
|
};
|
|
|
|
|
|
|
|
monitor_id_proxy.fmt(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-06 02:30:33 +11:00
|
|
|
impl MonitorHandle {
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn name(&self) -> Option<String> {
|
2018-05-06 03:36:34 +10:00
|
|
|
self.mgr.with_info(&self.proxy, |_, info| {
|
|
|
|
format!("{} ({})", info.model, info.make)
|
|
|
|
})
|
2017-03-11 09:22:59 +11:00
|
|
|
}
|
|
|
|
|
2017-10-20 18:46:42 +11:00
|
|
|
#[inline]
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn native_identifier(&self) -> u32 {
|
2018-05-06 03:36:34 +10:00
|
|
|
self.mgr.with_info(&self.proxy, |id, _| id).unwrap_or(0)
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
|
|
|
|
2019-06-13 16:33:44 +10:00
|
|
|
pub fn size(&self) -> PhysicalSize {
|
2018-05-06 03:36:34 +10:00
|
|
|
match self.mgr.with_info(&self.proxy, |_, info| {
|
|
|
|
info.modes
|
|
|
|
.iter()
|
|
|
|
.find(|m| m.is_current)
|
|
|
|
.map(|m| m.dimensions)
|
|
|
|
}) {
|
|
|
|
Some(Some((w, h))) => (w as u32, h as u32),
|
|
|
|
_ => (0, 0),
|
2019-06-22 01:33:15 +10:00
|
|
|
}
|
|
|
|
.into()
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
|
|
|
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn position(&self) -> PhysicalPosition {
|
2018-05-06 03:36:34 +10:00
|
|
|
self.mgr
|
|
|
|
.with_info(&self.proxy, |_, info| info.location)
|
|
|
|
.unwrap_or((0, 0))
|
2018-06-15 09:42:18 +10:00
|
|
|
.into()
|
2017-03-05 00:04:01 +11:00
|
|
|
}
|
2017-03-11 09:22:59 +11:00
|
|
|
|
2017-10-20 18:46:42 +11:00
|
|
|
#[inline]
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn hidpi_factor(&self) -> i32 {
|
2018-05-06 03:36:34 +10:00
|
|
|
self.mgr
|
2018-06-15 09:42:18 +10:00
|
|
|
.with_info(&self.proxy, |_, info| info.scale_factor)
|
|
|
|
.unwrap_or(1)
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2019-06-13 04:07:25 +10:00
|
|
|
|
|
|
|
#[inline]
|
2019-07-30 04:16:14 +10:00
|
|
|
pub fn video_modes(&self) -> impl Iterator<Item = RootVideoMode> {
|
|
|
|
let monitor = self.clone();
|
|
|
|
|
2019-06-13 04:07:25 +10:00
|
|
|
self.mgr
|
|
|
|
.with_info(&self.proxy, |_, info| info.modes.clone())
|
|
|
|
.unwrap_or(vec![])
|
|
|
|
.into_iter()
|
2019-07-30 04:16:14 +10:00
|
|
|
.map(move |x| RootVideoMode {
|
|
|
|
video_mode: PlatformVideoMode::Wayland(VideoMode {
|
|
|
|
size: (x.dimensions.0 as u32, x.dimensions.1 as u32),
|
|
|
|
refresh_rate: (x.refresh_rate as f32 / 1000.0).round() as u16,
|
|
|
|
bit_depth: 32,
|
|
|
|
monitor: monitor.clone(),
|
|
|
|
}),
|
2019-06-13 04:07:25 +10:00
|
|
|
})
|
|
|
|
}
|
2017-10-20 18:46:42 +11:00
|
|
|
}
|
2018-06-17 00:14:12 +10:00
|
|
|
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn primary_monitor(outputs: &OutputMgr) -> MonitorHandle {
|
2018-06-17 00:14:12 +10:00
|
|
|
outputs.with_all(|list| {
|
|
|
|
if let Some(&(_, ref proxy, _)) = list.first() {
|
2019-02-06 02:30:33 +11:00
|
|
|
MonitorHandle {
|
2018-06-17 00:14:12 +10:00
|
|
|
proxy: proxy.clone(),
|
|
|
|
mgr: outputs.clone(),
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
panic!("No monitor is available.")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-05-30 11:29:54 +10:00
|
|
|
pub fn available_monitors(outputs: &OutputMgr) -> VecDeque<MonitorHandle> {
|
2018-06-17 00:14:12 +10:00
|
|
|
outputs.with_all(|list| {
|
|
|
|
list.iter()
|
2019-06-25 02:14:55 +10:00
|
|
|
.map(|&(_, ref proxy, _)| MonitorHandle {
|
|
|
|
proxy: proxy.clone(),
|
|
|
|
mgr: outputs.clone(),
|
2018-06-17 00:14:12 +10:00
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
})
|
|
|
|
}
|