linux: hook up X11rb for Window creation
This commit is contained in:
parent
e95bf24a1f
commit
cefc98258f
5 changed files with 181 additions and 25 deletions
28
Cargo.lock
generated
28
Cargo.lock
generated
|
@ -3088,6 +3088,16 @@ dependencies = [
|
||||||
"version_check",
|
"version_check",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "gethostname"
|
||||||
|
version = "0.4.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "0176e0459c2e4a1fe232f984bca6890e681076abb9934f6cea7c326f3fc47818"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
"windows-targets 0.48.5",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "getrandom"
|
name = "getrandom"
|
||||||
version = "0.1.16"
|
version = "0.1.16"
|
||||||
|
@ -3277,6 +3287,7 @@ dependencies = [
|
||||||
"util",
|
"util",
|
||||||
"uuid 1.4.1",
|
"uuid 1.4.1",
|
||||||
"waker-fn",
|
"waker-fn",
|
||||||
|
"x11rb",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -10273,6 +10284,23 @@ dependencies = [
|
||||||
"tap",
|
"tap",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "x11rb"
|
||||||
|
version = "0.13.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f8f25ead8c7e4cba123243a6367da5d3990e0d3affa708ea19dce96356bd9f1a"
|
||||||
|
dependencies = [
|
||||||
|
"gethostname",
|
||||||
|
"rustix 0.38.30",
|
||||||
|
"x11rb-protocol",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "x11rb-protocol"
|
||||||
|
version = "0.13.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "e63e71c4b8bd9ffec2c963173a4dc4cbde9ee96961d4fcb4429db9929b606c34"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "xattr"
|
name = "xattr"
|
||||||
version = "0.2.3"
|
version = "0.2.3"
|
||||||
|
|
|
@ -96,3 +96,4 @@ objc = "0.2"
|
||||||
|
|
||||||
[target.'cfg(target_os = "linux")'.dependencies]
|
[target.'cfg(target_os = "linux")'.dependencies]
|
||||||
flume = "0.11"
|
flume = "0.11"
|
||||||
|
x11rb = "0.13"
|
||||||
|
|
|
@ -1,23 +1,42 @@
|
||||||
use crate::{point, size, Bounds, DisplayId, GlobalPixels, PlatformDisplay};
|
use crate::{point, size, Bounds, DisplayId, GlobalPixels, PlatformDisplay, Size};
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
use x11rb::{connection::Connection as _, rust_connection::RustConnection};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct LinuxDisplay;
|
pub(crate) struct LinuxDisplay {
|
||||||
|
x11_screen_index: usize,
|
||||||
|
bounds: Bounds<GlobalPixels>,
|
||||||
|
uuid: Uuid,
|
||||||
|
}
|
||||||
|
|
||||||
impl PlatformDisplay for LinuxDisplay {
|
impl LinuxDisplay {
|
||||||
fn id(&self) -> DisplayId {
|
pub(crate) fn new(xc: &RustConnection, x11_screen_index: usize) -> Self {
|
||||||
DisplayId(0)
|
let screen = &xc.setup().roots[x11_screen_index];
|
||||||
}
|
Self {
|
||||||
|
x11_screen_index,
|
||||||
fn uuid(&self) -> Result<Uuid> {
|
bounds: Bounds {
|
||||||
Ok(Uuid::from_bytes([0; 16]))
|
origin: Default::default(),
|
||||||
}
|
size: Size {
|
||||||
|
width: GlobalPixels(screen.width_in_pixels as f32),
|
||||||
fn bounds(&self) -> Bounds<GlobalPixels> {
|
height: GlobalPixels(screen.height_in_pixels as f32),
|
||||||
Bounds {
|
},
|
||||||
origin: point(GlobalPixels(0.0), GlobalPixels(0.0)),
|
},
|
||||||
size: size(GlobalPixels(100.0), GlobalPixels(100.0)),
|
uuid: Uuid::from_bytes([0; 16]),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl PlatformDisplay for LinuxDisplay {
|
||||||
|
fn id(&self) -> DisplayId {
|
||||||
|
DisplayId(self.x11_screen_index as u32)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn uuid(&self) -> Result<Uuid> {
|
||||||
|
Ok(self.uuid)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn bounds(&self) -> Bounds<GlobalPixels> {
|
||||||
|
self.bounds
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -17,10 +17,13 @@ use std::{
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
use time::UtcOffset;
|
use time::UtcOffset;
|
||||||
|
use x11rb::{connection::Connection as _, rust_connection::RustConnection};
|
||||||
|
|
||||||
pub(crate) struct LinuxPlatform(Mutex<LinuxPlatformState>);
|
pub(crate) struct LinuxPlatform(Mutex<LinuxPlatformState>);
|
||||||
|
|
||||||
pub(crate) struct LinuxPlatformState {
|
pub(crate) struct LinuxPlatformState {
|
||||||
|
x11_connection: RustConnection,
|
||||||
|
x11_root_index: usize,
|
||||||
gpu: Arc<blade::Context>,
|
gpu: Arc<blade::Context>,
|
||||||
background_executor: BackgroundExecutor,
|
background_executor: BackgroundExecutor,
|
||||||
foreground_executor: ForegroundExecutor,
|
foreground_executor: ForegroundExecutor,
|
||||||
|
@ -35,17 +38,22 @@ impl Default for LinuxPlatform {
|
||||||
|
|
||||||
impl LinuxPlatform {
|
impl LinuxPlatform {
|
||||||
pub(crate) fn new() -> Self {
|
pub(crate) fn new() -> Self {
|
||||||
|
let (x11_connection, x11_root_index) = x11rb::connect(None).unwrap();
|
||||||
|
|
||||||
let dispatcher = Arc::new(LinuxDispatcher::new());
|
let dispatcher = Arc::new(LinuxDispatcher::new());
|
||||||
let gpu = Arc::new(
|
let gpu = Arc::new(
|
||||||
unsafe {
|
unsafe {
|
||||||
blade::Context::init(blade::ContextDesc {
|
blade::Context::init(blade::ContextDesc {
|
||||||
validation: true, //FIXME
|
validation: cfg!(debug_assertions),
|
||||||
capture: false,
|
capture: false,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
);
|
);
|
||||||
|
|
||||||
Self(Mutex::new(LinuxPlatformState {
|
Self(Mutex::new(LinuxPlatformState {
|
||||||
|
x11_connection,
|
||||||
|
x11_root_index,
|
||||||
gpu,
|
gpu,
|
||||||
background_executor: BackgroundExecutor::new(dispatcher.clone()),
|
background_executor: BackgroundExecutor::new(dispatcher.clone()),
|
||||||
foreground_executor: ForegroundExecutor::new(dispatcher),
|
foreground_executor: ForegroundExecutor::new(dispatcher),
|
||||||
|
@ -84,11 +92,21 @@ impl Platform for LinuxPlatform {
|
||||||
fn unhide_other_apps(&self) {}
|
fn unhide_other_apps(&self) {}
|
||||||
|
|
||||||
fn displays(&self) -> Vec<Rc<dyn PlatformDisplay>> {
|
fn displays(&self) -> Vec<Rc<dyn PlatformDisplay>> {
|
||||||
Vec::new()
|
let lock = self.0.lock();
|
||||||
|
let setup = lock.x11_connection.setup();
|
||||||
|
(0..setup.roots.len())
|
||||||
|
.map(|id| {
|
||||||
|
Rc::new(LinuxDisplay::new(&lock.x11_connection, id)) as Rc<dyn PlatformDisplay>
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn display(&self, id: DisplayId) -> Option<Rc<dyn PlatformDisplay>> {
|
fn display(&self, id: DisplayId) -> Option<Rc<dyn PlatformDisplay>> {
|
||||||
None
|
let lock = self.0.lock();
|
||||||
|
Some(Rc::new(LinuxDisplay::new(
|
||||||
|
&lock.x11_connection,
|
||||||
|
id.0 as usize,
|
||||||
|
)))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn active_window(&self) -> Option<AnyWindowHandle> {
|
fn active_window(&self) -> Option<AnyWindowHandle> {
|
||||||
|
@ -104,7 +122,8 @@ impl Platform for LinuxPlatform {
|
||||||
Box::new(LinuxWindow::new(
|
Box::new(LinuxWindow::new(
|
||||||
options,
|
options,
|
||||||
handle,
|
handle,
|
||||||
Rc::new(LinuxDisplay),
|
&lock.x11_connection,
|
||||||
|
lock.x11_root_index,
|
||||||
&lock.gpu,
|
&lock.gpu,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
px, AnyWindowHandle, AtlasKey, AtlasTextureId, AtlasTile, BladeAtlas, Bounds, KeyDownEvent,
|
px, AnyWindowHandle, AtlasKey, AtlasTextureId, AtlasTile, BladeAtlas, Bounds, KeyDownEvent,
|
||||||
Keystroke, Pixels, PlatformAtlas, PlatformDisplay, PlatformInput, PlatformInputHandler,
|
Keystroke, LinuxDisplay, Pixels, PlatformAtlas, PlatformDisplay, PlatformInput,
|
||||||
PlatformWindow, Point, Size, TileId, WindowAppearance, WindowBounds, WindowOptions,
|
PlatformInputHandler, PlatformWindow, Point, Size, TileId, WindowAppearance, WindowBounds,
|
||||||
|
WindowOptions,
|
||||||
};
|
};
|
||||||
use collections::HashMap;
|
use collections::HashMap;
|
||||||
use parking_lot::Mutex;
|
use parking_lot::Mutex;
|
||||||
|
@ -9,9 +10,18 @@ use std::{
|
||||||
rc::{Rc, Weak},
|
rc::{Rc, Weak},
|
||||||
sync::{self, Arc},
|
sync::{self, Arc},
|
||||||
};
|
};
|
||||||
|
use x11rb::{
|
||||||
|
connection::Connection as _,
|
||||||
|
protocol::xproto::{
|
||||||
|
AtomEnum, ConnectionExt as _, CreateWindowAux, EventMask, PropMode, WindowClass,
|
||||||
|
},
|
||||||
|
rust_connection::RustConnection,
|
||||||
|
wrapper::ConnectionExt as _,
|
||||||
|
};
|
||||||
|
|
||||||
pub(crate) struct LinuxWindowState {
|
pub(crate) struct LinuxWindowState {
|
||||||
display: Rc<dyn crate::PlatformDisplay>,
|
display: Rc<dyn PlatformDisplay>,
|
||||||
|
win_id: u32,
|
||||||
sprite_atlas: Arc<BladeAtlas>,
|
sprite_atlas: Arc<BladeAtlas>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,11 +32,90 @@ impl LinuxWindow {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
options: WindowOptions,
|
options: WindowOptions,
|
||||||
handle: AnyWindowHandle,
|
handle: AnyWindowHandle,
|
||||||
display: Rc<dyn PlatformDisplay>,
|
x11_connection: &RustConnection,
|
||||||
|
x11_main_screen_index: usize,
|
||||||
gpu: &Arc<blade::Context>,
|
gpu: &Arc<blade::Context>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
|
let x11_screen_index = options
|
||||||
|
.display_id
|
||||||
|
.map_or(x11_main_screen_index, |did| did.0 as usize);
|
||||||
|
let screen = &x11_connection.setup().roots[x11_screen_index];
|
||||||
|
|
||||||
|
let win_id = x11_connection.generate_id().unwrap();
|
||||||
|
let win_aux = CreateWindowAux::new()
|
||||||
|
.event_mask(
|
||||||
|
EventMask::EXPOSURE | EventMask::STRUCTURE_NOTIFY | EventMask::POINTER_MOTION,
|
||||||
|
)
|
||||||
|
.background_pixel(screen.white_pixel);
|
||||||
|
|
||||||
|
let wm_protocols = x11_connection
|
||||||
|
.intern_atom(false, b"WM_PROTOCOLS")
|
||||||
|
.unwrap()
|
||||||
|
.reply()
|
||||||
|
.unwrap()
|
||||||
|
.atom;
|
||||||
|
let wm_delete_window = x11_connection
|
||||||
|
.intern_atom(false, b"WM_DELETE_WINDOW")
|
||||||
|
.unwrap()
|
||||||
|
.reply()
|
||||||
|
.unwrap()
|
||||||
|
.atom;
|
||||||
|
let (bound_x, bound_y, bound_width, bound_height) = match options.bounds {
|
||||||
|
WindowBounds::Fullscreen | WindowBounds::Maximized => {
|
||||||
|
(0, 0, screen.width_in_pixels, screen.height_in_pixels)
|
||||||
|
}
|
||||||
|
WindowBounds::Fixed(bounds) => (
|
||||||
|
bounds.origin.x.0 as i16,
|
||||||
|
bounds.origin.y.0 as i16,
|
||||||
|
bounds.size.width.0 as u16,
|
||||||
|
bounds.size.height.0 as u16,
|
||||||
|
),
|
||||||
|
};
|
||||||
|
|
||||||
|
x11_connection
|
||||||
|
.create_window(
|
||||||
|
x11rb::COPY_DEPTH_FROM_PARENT,
|
||||||
|
win_id,
|
||||||
|
screen.root,
|
||||||
|
bound_x,
|
||||||
|
bound_y,
|
||||||
|
bound_width,
|
||||||
|
bound_height,
|
||||||
|
0,
|
||||||
|
WindowClass::INPUT_OUTPUT,
|
||||||
|
0,
|
||||||
|
&win_aux,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
if let Some(titlebar) = options.titlebar {
|
||||||
|
if let Some(title) = titlebar.title {
|
||||||
|
x11_connection
|
||||||
|
.change_property8(
|
||||||
|
PropMode::REPLACE,
|
||||||
|
win_id,
|
||||||
|
AtomEnum::WM_NAME,
|
||||||
|
AtomEnum::STRING,
|
||||||
|
title.as_bytes(),
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
x11_connection
|
||||||
|
.change_property32(
|
||||||
|
PropMode::REPLACE,
|
||||||
|
win_id,
|
||||||
|
wm_protocols,
|
||||||
|
AtomEnum::ATOM,
|
||||||
|
&[wm_delete_window],
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
x11_connection.map_window(win_id).unwrap();
|
||||||
|
|
||||||
Self(Arc::new(Mutex::new(LinuxWindowState {
|
Self(Arc::new(Mutex::new(LinuxWindowState {
|
||||||
display,
|
display: Rc::new(LinuxDisplay::new(x11_connection, x11_screen_index)),
|
||||||
|
win_id,
|
||||||
sprite_atlas: Arc::new(BladeAtlas::new(gpu)),
|
sprite_atlas: Arc::new(BladeAtlas::new(gpu)),
|
||||||
})))
|
})))
|
||||||
}
|
}
|
||||||
|
@ -53,7 +142,7 @@ impl PlatformWindow for LinuxWindow {
|
||||||
unimplemented!()
|
unimplemented!()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn display(&self) -> Rc<dyn crate::PlatformDisplay> {
|
fn display(&self) -> Rc<dyn PlatformDisplay> {
|
||||||
Rc::clone(&self.0.lock().display)
|
Rc::clone(&self.0.lock().display)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue