2016-03-02 13:06:13 +11:00
|
|
|
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "openbsd"))]
|
2015-04-24 17:51:23 +10:00
|
|
|
|
2015-09-24 17:11:59 +10:00
|
|
|
pub use self::monitor::{MonitorId, get_available_monitors, get_primary_monitor};
|
2016-02-27 23:59:11 +11:00
|
|
|
pub use self::window::{Window, XWindow, PollEventsIterator, WaitEventsIterator, WindowProxy};
|
2015-12-24 20:57:08 +11:00
|
|
|
pub use self::xdisplay::{XConnection, XNotSupported, XError};
|
2015-04-24 22:22:57 +10:00
|
|
|
|
2015-04-25 02:52:07 +10:00
|
|
|
pub mod ffi;
|
2015-04-24 22:22:57 +10:00
|
|
|
|
2017-03-04 07:41:51 +11:00
|
|
|
use platform::PlatformSpecificWindowBuilderAttributes;
|
|
|
|
use CreationError;
|
|
|
|
|
|
|
|
use std::sync::Arc;
|
|
|
|
|
2015-05-17 19:19:06 +10:00
|
|
|
mod events;
|
2015-06-23 07:49:48 +10:00
|
|
|
mod input;
|
2015-05-17 19:19:06 +10:00
|
|
|
mod monitor;
|
|
|
|
mod window;
|
|
|
|
mod xdisplay;
|
2017-03-04 07:41:51 +11:00
|
|
|
|
|
|
|
// API TRANSITION
|
|
|
|
//
|
|
|
|
// We don't use the gen_api_transistion!() macro but rather do the expansion manually:
|
|
|
|
//
|
|
|
|
// As this module is nested into platform/linux, its code is not _exactly_ the same as
|
|
|
|
// the one generated by the macro.
|
|
|
|
|
|
|
|
pub struct EventsLoop {
|
|
|
|
windows: ::std::sync::Mutex<Vec<::std::sync::Arc<Window>>>,
|
|
|
|
interrupted: ::std::sync::atomic::AtomicBool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EventsLoop {
|
|
|
|
pub fn new() -> EventsLoop {
|
|
|
|
EventsLoop {
|
|
|
|
windows: ::std::sync::Mutex::new(vec![]),
|
|
|
|
interrupted: ::std::sync::atomic::AtomicBool::new(false),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn interrupt(&self) {
|
|
|
|
self.interrupted.store(true, ::std::sync::atomic::Ordering::Relaxed);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn poll_events<F>(&self, mut callback: F)
|
|
|
|
where F: FnMut(::Event)
|
|
|
|
{
|
2017-03-04 21:14:08 +11:00
|
|
|
let windows = self.windows.lock().unwrap();
|
2017-03-04 07:41:51 +11:00
|
|
|
for window in windows.iter() {
|
|
|
|
for event in window.poll_events() {
|
|
|
|
callback(::Event::WindowEvent {
|
|
|
|
window_id: ::WindowId(::platform::WindowId::X(WindowId(&**window as *const Window as usize))),
|
|
|
|
event: event,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn run_forever<F>(&self, mut callback: F)
|
|
|
|
where F: FnMut(::Event)
|
|
|
|
{
|
|
|
|
self.interrupted.store(false, ::std::sync::atomic::Ordering::Relaxed);
|
|
|
|
|
|
|
|
// Yeah that's a very bad implementation.
|
|
|
|
loop {
|
|
|
|
self.poll_events(|e| callback(e));
|
2017-03-04 21:14:08 +11:00
|
|
|
::std::thread::sleep(::std::time::Duration::from_millis(5));
|
2017-03-04 07:41:51 +11:00
|
|
|
if self.interrupted.load(::std::sync::atomic::Ordering::Relaxed) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
|
|
|
pub struct WindowId(usize);
|
|
|
|
|
|
|
|
pub struct Window2 {
|
|
|
|
pub window: ::std::sync::Arc<Window>,
|
|
|
|
events_loop: ::std::sync::Weak<::platform::EventsLoop>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ::std::ops::Deref for Window2 {
|
|
|
|
type Target = Window;
|
|
|
|
#[inline]
|
|
|
|
fn deref(&self) -> &Window {
|
|
|
|
&*self.window
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Window2 {
|
|
|
|
pub fn new(events_loop: ::std::sync::Arc<::platform::EventsLoop>, display: &Arc<XConnection>,
|
|
|
|
window: &::WindowAttributes, pl_attribs: &PlatformSpecificWindowBuilderAttributes)
|
|
|
|
-> Result<Window2, CreationError>
|
|
|
|
{
|
|
|
|
let win = ::std::sync::Arc::new(try!(Window::new(display, window, pl_attribs)));
|
|
|
|
if let ::platform::EventsLoop::X(ref ev) = *events_loop {
|
|
|
|
ev.windows.lock().unwrap().push(win.clone());
|
|
|
|
} else {
|
|
|
|
// It should not be possible to create an eventloop not matching the backend
|
|
|
|
// in use
|
|
|
|
unreachable!()
|
|
|
|
}
|
|
|
|
Ok(Window2 {
|
|
|
|
window: win,
|
|
|
|
events_loop: ::std::sync::Arc::downgrade(&events_loop),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn id(&self) -> WindowId {
|
|
|
|
WindowId(&*self.window as *const Window as usize)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Window2 {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
if let Some(ev) = self.events_loop.upgrade() {
|
|
|
|
if let ::platform::EventsLoop::X(ref ev) = *ev {
|
|
|
|
let mut windows = ev.windows.lock().unwrap();
|
|
|
|
windows.retain(|w| &**w as *const Window != &*self.window as *const _);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|