2016-03-02 13:06:13 +11:00
|
|
|
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "openbsd"))]
|
2015-09-26 02:04:55 +10:00
|
|
|
|
2017-12-17 20:17:26 +11:00
|
|
|
use std::os::raw;
|
2016-10-20 03:11:02 +11:00
|
|
|
use std::sync::Arc;
|
2017-01-07 07:59:35 +11:00
|
|
|
use std::ptr;
|
2017-09-01 19:04:57 +10:00
|
|
|
use EventsLoop;
|
2017-08-30 16:49:18 +10:00
|
|
|
use MonitorId;
|
2015-09-26 02:04:55 +10:00
|
|
|
use Window;
|
2017-09-01 19:04:57 +10:00
|
|
|
use platform::EventsLoop as LinuxEventsLoop;
|
2017-09-07 01:32:24 +10:00
|
|
|
use platform::Window as LinuxWindow;
|
2016-01-08 02:01:18 +11:00
|
|
|
use WindowBuilder;
|
2017-03-04 06:56:27 +11:00
|
|
|
use platform::x11::XConnection;
|
|
|
|
use platform::x11::ffi::XVisualInfo;
|
2015-09-26 02:04:55 +10:00
|
|
|
|
2017-09-23 17:36:30 +10:00
|
|
|
// TODO: stupid hack so that glutin can do its work
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub use platform::x11;
|
|
|
|
|
2017-09-01 19:04:57 +10:00
|
|
|
pub use platform::XNotSupported;
|
2017-03-15 20:11:43 +11:00
|
|
|
|
2017-09-01 19:04:57 +10:00
|
|
|
/// Additional methods on `EventsLoop` that are specific to Linux.
|
|
|
|
pub trait EventsLoopExt {
|
|
|
|
/// Builds a new `EventsLoop` that is forced to use X11.
|
|
|
|
fn new_x11() -> Result<Self, XNotSupported>
|
|
|
|
where Self: Sized;
|
|
|
|
|
|
|
|
/// Builds a new `EventsLoop` that is forced to use Wayland.
|
|
|
|
fn new_wayland() -> Self
|
|
|
|
where Self: Sized;
|
2017-09-23 17:36:30 +10:00
|
|
|
|
|
|
|
/// True if the `EventsLoop` uses Wayland.
|
|
|
|
fn is_wayland(&self) -> bool;
|
|
|
|
|
|
|
|
/// True if the `EventsLoop` uses X11.
|
|
|
|
fn is_x11(&self) -> bool;
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
fn get_xlib_xconnection(&self) -> Option<Arc<XConnection>>;
|
2017-09-01 19:04:57 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
impl EventsLoopExt for EventsLoop {
|
|
|
|
#[inline]
|
|
|
|
fn new_x11() -> Result<Self, XNotSupported> {
|
2017-10-19 05:40:21 +11:00
|
|
|
LinuxEventsLoop::new_x11().map(|ev|
|
|
|
|
EventsLoop {
|
|
|
|
events_loop: ev,
|
|
|
|
_marker: ::std::marker::PhantomData,
|
|
|
|
}
|
|
|
|
)
|
2017-09-01 19:04:57 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn new_wayland() -> Self {
|
|
|
|
EventsLoop {
|
|
|
|
events_loop: match LinuxEventsLoop::new_wayland() {
|
|
|
|
Ok(e) => e,
|
|
|
|
Err(_) => panic!() // TODO: propagate
|
2017-10-19 05:40:21 +11:00
|
|
|
},
|
|
|
|
_marker: ::std::marker::PhantomData,
|
2017-09-01 19:04:57 +10:00
|
|
|
}
|
2017-01-08 00:34:38 +11:00
|
|
|
}
|
2017-09-23 17:36:30 +10:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn is_wayland(&self) -> bool {
|
|
|
|
self.events_loop.is_wayland()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn is_x11(&self) -> bool {
|
|
|
|
!self.events_loop.is_wayland()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn get_xlib_xconnection(&self) -> Option<Arc<XConnection>> {
|
|
|
|
self.events_loop.x_connection().cloned()
|
|
|
|
}
|
2017-01-08 00:34:38 +11:00
|
|
|
}
|
|
|
|
|
2016-01-08 02:01:18 +11:00
|
|
|
/// Additional methods on `Window` that are specific to Unix.
|
2015-09-26 02:04:55 +10:00
|
|
|
pub trait WindowExt {
|
2017-12-17 20:17:26 +11:00
|
|
|
/// Returns the ID of the `Window` xlib object that is used by this window.
|
2015-09-26 02:04:55 +10:00
|
|
|
///
|
|
|
|
/// Returns `None` if the window doesn't use xlib (if it uses wayland for example).
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xlib_window(&self) -> Option<raw::c_ulong>;
|
2015-09-26 02:04:55 +10:00
|
|
|
|
|
|
|
/// Returns a pointer to the `Display` object of xlib that is used by this window.
|
|
|
|
///
|
|
|
|
/// Returns `None` if the window doesn't use xlib (if it uses wayland for example).
|
|
|
|
///
|
|
|
|
/// The pointer will become invalid when the glutin `Window` is destroyed.
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xlib_display(&self) -> Option<*mut raw::c_void>;
|
2016-10-20 03:11:02 +11:00
|
|
|
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xlib_screen_id(&self) -> Option<raw::c_int>;
|
2016-10-20 03:11:02 +11:00
|
|
|
|
|
|
|
fn get_xlib_xconnection(&self) -> Option<Arc<XConnection>>;
|
2017-07-12 14:00:51 +10:00
|
|
|
|
|
|
|
fn send_xim_spot(&self, x: i16, y: i16);
|
2017-12-17 20:17:26 +11:00
|
|
|
|
2016-07-31 07:58:28 +10:00
|
|
|
/// This function returns the underlying `xcb_connection_t` of an xlib `Display`.
|
|
|
|
///
|
|
|
|
/// Returns `None` if the window doesn't use xlib (if it uses wayland for example).
|
|
|
|
///
|
|
|
|
/// The pointer will become invalid when the glutin `Window` is destroyed.
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xcb_connection(&self) -> Option<*mut raw::c_void>;
|
2016-05-08 17:28:54 +10:00
|
|
|
|
2016-10-10 17:01:58 +11:00
|
|
|
/// Returns a pointer to the `wl_surface` object of wayland that is used by this window.
|
|
|
|
///
|
|
|
|
/// Returns `None` if the window doesn't use wayland (if it uses xlib for example).
|
|
|
|
///
|
|
|
|
/// The pointer will become invalid when the glutin `Window` is destroyed.
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_wayland_surface(&self) -> Option<*mut raw::c_void>;
|
2016-10-10 17:01:58 +11:00
|
|
|
|
|
|
|
/// Returns a pointer to the `wl_display` object of wayland that is used by this window.
|
|
|
|
///
|
|
|
|
/// Returns `None` if the window doesn't use wayland (if it uses xlib for example).
|
|
|
|
///
|
|
|
|
/// The pointer will become invalid when the glutin `Window` is destroyed.
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_wayland_display(&self) -> Option<*mut raw::c_void>;
|
2017-09-28 00:31:46 +10:00
|
|
|
|
|
|
|
/// Check if the window is ready for drawing
|
|
|
|
///
|
2017-11-04 03:35:29 +11:00
|
|
|
/// It is a remnant of a previous implementation detail for the
|
|
|
|
/// wayland backend, and is no longer relevant.
|
2017-09-28 00:31:46 +10:00
|
|
|
///
|
2017-11-04 03:35:29 +11:00
|
|
|
/// Always return true.
|
|
|
|
#[deprecated]
|
2017-09-28 00:31:46 +10:00
|
|
|
fn is_ready(&self) -> bool;
|
2015-09-26 02:04:55 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
impl WindowExt for Window {
|
|
|
|
#[inline]
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xlib_window(&self) -> Option<raw::c_ulong> {
|
2017-03-04 07:41:51 +11:00
|
|
|
match self.window {
|
2015-09-26 02:04:55 +10:00
|
|
|
LinuxWindow::X(ref w) => Some(w.get_xlib_window()),
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xlib_display(&self) -> Option<*mut raw::c_void> {
|
2017-03-04 07:41:51 +11:00
|
|
|
match self.window {
|
2015-10-23 07:59:52 +11:00
|
|
|
LinuxWindow::X(ref w) => Some(w.get_xlib_display()),
|
2015-09-26 02:04:55 +10:00
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
2016-05-08 17:28:54 +10:00
|
|
|
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xlib_screen_id(&self) -> Option<raw::c_int> {
|
2017-03-04 07:41:51 +11:00
|
|
|
match self.window {
|
2016-10-20 03:11:02 +11:00
|
|
|
LinuxWindow::X(ref w) => Some(w.get_xlib_screen_id()),
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_xlib_xconnection(&self) -> Option<Arc<XConnection>> {
|
2017-03-04 07:41:51 +11:00
|
|
|
match self.window {
|
2016-10-20 03:11:02 +11:00
|
|
|
LinuxWindow::X(ref w) => Some(w.get_xlib_xconnection()),
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_xcb_connection(&self) -> Option<*mut raw::c_void> {
|
2017-03-04 07:41:51 +11:00
|
|
|
match self.window {
|
2016-07-31 07:58:28 +10:00
|
|
|
LinuxWindow::X(ref w) => Some(w.get_xcb_connection()),
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-12 14:00:51 +10:00
|
|
|
fn send_xim_spot(&self, x: i16, y: i16) {
|
|
|
|
if let LinuxWindow::X(ref w) = self.window {
|
|
|
|
w.send_xim_spot(x, y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-08 17:28:54 +10:00
|
|
|
#[inline]
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_wayland_surface(&self) -> Option<*mut raw::c_void> {
|
2016-10-10 17:01:58 +11:00
|
|
|
use wayland_client::Proxy;
|
2017-03-04 07:41:51 +11:00
|
|
|
match self.window {
|
2017-06-24 04:20:49 +10:00
|
|
|
LinuxWindow::Wayland(ref w) => Some(w.get_surface().ptr() as *mut _),
|
2016-05-08 17:28:54 +10:00
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2017-12-17 20:17:26 +11:00
|
|
|
fn get_wayland_display(&self) -> Option<*mut raw::c_void> {
|
2017-06-24 04:20:49 +10:00
|
|
|
use wayland_client::Proxy;
|
2017-03-04 07:41:51 +11:00
|
|
|
match self.window {
|
2017-06-24 04:20:49 +10:00
|
|
|
LinuxWindow::Wayland(ref w) => Some(w.get_display().ptr() as *mut _),
|
2016-05-08 17:28:54 +10:00
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
2017-09-28 00:31:46 +10:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn is_ready(&self) -> bool {
|
2017-11-04 03:35:29 +11:00
|
|
|
true
|
2017-09-28 00:31:46 +10:00
|
|
|
}
|
2015-09-26 02:04:55 +10:00
|
|
|
}
|
2016-01-08 02:01:18 +11:00
|
|
|
|
|
|
|
/// Additional methods on `WindowBuilder` that are specific to Unix.
|
|
|
|
pub trait WindowBuilderExt {
|
2017-01-08 00:34:38 +11:00
|
|
|
fn with_x11_visual<T>(self, visual_infos: *const T) -> WindowBuilder;
|
|
|
|
fn with_x11_screen(self, screen_id: i32) -> WindowBuilder;
|
2016-01-08 02:01:18 +11:00
|
|
|
}
|
|
|
|
|
2016-02-27 23:59:11 +11:00
|
|
|
impl WindowBuilderExt for WindowBuilder {
|
2017-01-07 07:59:35 +11:00
|
|
|
#[inline]
|
2017-01-08 00:34:38 +11:00
|
|
|
fn with_x11_visual<T>(mut self, visual_infos: *const T) -> WindowBuilder {
|
2017-01-07 07:59:35 +11:00
|
|
|
self.platform_specific.visual_infos = Some(
|
2017-01-08 00:34:38 +11:00
|
|
|
unsafe { ptr::read(visual_infos as *const XVisualInfo) }
|
|
|
|
);
|
2017-01-07 07:59:35 +11:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2017-01-08 00:34:38 +11:00
|
|
|
fn with_x11_screen(mut self, screen_id: i32) -> WindowBuilder {
|
2017-01-07 07:59:35 +11:00
|
|
|
self.platform_specific.screen_id = Some(screen_id);
|
|
|
|
self
|
|
|
|
}
|
2016-01-08 02:01:18 +11:00
|
|
|
}
|
2017-08-30 16:49:18 +10:00
|
|
|
|
|
|
|
/// Additional methods on `MonitorId` that are specific to Linux.
|
|
|
|
pub trait MonitorIdExt {
|
|
|
|
/// Returns the inner identifier of the monitor.
|
|
|
|
fn native_id(&self) -> u32;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl MonitorIdExt for MonitorId {
|
|
|
|
#[inline]
|
|
|
|
fn native_id(&self) -> u32 {
|
|
|
|
self.inner.get_native_identifier()
|
|
|
|
}
|
|
|
|
}
|