2020-04-21 16:56:59 +00:00
|
|
|
use std::{
|
|
|
|
cell::RefCell,
|
|
|
|
rc::Rc,
|
|
|
|
sync::{
|
|
|
|
atomic::{AtomicBool, Ordering},
|
|
|
|
Arc, Mutex,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
use smithay::{
|
|
|
|
reexports::{
|
2021-05-30 22:56:37 +00:00
|
|
|
calloop::{generic::Generic, Interest, LoopHandle, Mode},
|
2020-04-21 16:56:59 +00:00
|
|
|
wayland_server::{protocol::wl_surface::WlSurface, Display},
|
|
|
|
},
|
|
|
|
wayland::{
|
|
|
|
compositor::CompositorToken,
|
2020-05-04 13:04:30 +00:00
|
|
|
data_device::{default_action_chooser, init_data_device, set_data_device_focus, DataDeviceEvent},
|
|
|
|
seat::{CursorImageStatus, KeyboardHandle, PointerHandle, Seat, XkbConfig},
|
2020-04-21 16:56:59 +00:00
|
|
|
shm::init_shm_global,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2021-05-15 16:17:43 +00:00
|
|
|
#[cfg(feature = "egl")]
|
|
|
|
use smithay::backend::egl::display::EGLBufferReader;
|
2021-01-03 15:21:22 +00:00
|
|
|
#[cfg(feature = "xwayland")]
|
|
|
|
use smithay::xwayland::XWayland;
|
2020-05-04 13:04:30 +00:00
|
|
|
|
2021-05-15 16:17:43 +00:00
|
|
|
use crate::shell::init_shell;
|
2021-01-03 15:21:22 +00:00
|
|
|
#[cfg(feature = "xwayland")]
|
|
|
|
use crate::xwayland::XWm;
|
2020-04-21 16:56:59 +00:00
|
|
|
|
2021-05-30 22:56:37 +00:00
|
|
|
pub struct AnvilState<BackendData> {
|
|
|
|
pub backend_data: BackendData,
|
2020-04-21 16:56:59 +00:00
|
|
|
pub socket_name: String,
|
|
|
|
pub running: Arc<AtomicBool>,
|
|
|
|
pub display: Rc<RefCell<Display>>,
|
2021-05-30 22:56:37 +00:00
|
|
|
pub handle: LoopHandle<'static, AnvilState<BackendData>>,
|
2020-04-21 16:56:59 +00:00
|
|
|
pub ctoken: CompositorToken<crate::shell::Roles>,
|
|
|
|
pub window_map: Rc<RefCell<crate::window_map::WindowMap<crate::shell::Roles>>>,
|
|
|
|
pub dnd_icon: Arc<Mutex<Option<WlSurface>>>,
|
|
|
|
pub log: slog::Logger,
|
2020-05-04 13:04:30 +00:00
|
|
|
// input-related fields
|
|
|
|
pub pointer: PointerHandle,
|
|
|
|
pub keyboard: KeyboardHandle,
|
|
|
|
pub pointer_location: Rc<RefCell<(f64, f64)>>,
|
|
|
|
pub cursor_status: Arc<Mutex<CursorImageStatus>>,
|
|
|
|
pub seat_name: String,
|
2021-05-30 22:56:37 +00:00
|
|
|
#[cfg(feature = "egl")]
|
|
|
|
pub egl_reader: Rc<RefCell<Option<EGLBufferReader>>>,
|
2020-04-21 16:56:59 +00:00
|
|
|
// things we must keep alive
|
2021-01-03 15:21:22 +00:00
|
|
|
#[cfg(feature = "xwayland")]
|
2021-05-30 22:56:37 +00:00
|
|
|
_xwayland: XWayland<XWm<BackendData>>,
|
2020-04-21 16:56:59 +00:00
|
|
|
}
|
|
|
|
|
2021-05-30 22:56:37 +00:00
|
|
|
impl<BackendData: Backend + 'static> AnvilState<BackendData> {
|
2020-04-21 16:56:59 +00:00
|
|
|
pub fn init(
|
|
|
|
display: Rc<RefCell<Display>>,
|
2021-05-30 22:56:37 +00:00
|
|
|
handle: LoopHandle<'static, AnvilState<BackendData>>,
|
|
|
|
backend_data: BackendData,
|
2021-05-13 17:59:47 +00:00
|
|
|
#[cfg(feature = "egl")] egl_reader: Rc<RefCell<Option<EGLBufferReader>>>,
|
2020-04-21 16:56:59 +00:00
|
|
|
log: slog::Logger,
|
2021-05-30 22:56:37 +00:00
|
|
|
) -> AnvilState<BackendData> {
|
2020-04-21 16:56:59 +00:00
|
|
|
// init the wayland connection
|
2021-05-30 22:56:37 +00:00
|
|
|
handle
|
2020-04-21 16:56:59 +00:00
|
|
|
.insert_source(
|
2021-05-30 20:01:36 +00:00
|
|
|
Generic::from_fd(display.borrow().get_poll_fd(), Interest::READ, Mode::Level),
|
2021-05-30 22:56:37 +00:00
|
|
|
move |_, _, state: &mut AnvilState<BackendData>| {
|
|
|
|
let display = state.display.clone();
|
|
|
|
let mut display = display.borrow_mut();
|
|
|
|
match display.dispatch(std::time::Duration::from_millis(0), state) {
|
|
|
|
Ok(_) => Ok(()),
|
|
|
|
Err(e) => {
|
|
|
|
error!(state.log, "I/O error on the Wayland display: {}", e);
|
|
|
|
state.running.store(false, Ordering::SeqCst);
|
|
|
|
Err(e)
|
2020-04-21 16:56:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.expect("Failed to init the wayland event source.");
|
|
|
|
|
2020-04-21 17:42:03 +00:00
|
|
|
// Init the basic compositor globals
|
|
|
|
|
2021-05-30 22:56:37 +00:00
|
|
|
init_shm_global(&mut (*display).borrow_mut(), vec![], log.clone());
|
2020-04-21 16:56:59 +00:00
|
|
|
|
2021-05-13 17:59:47 +00:00
|
|
|
#[cfg(feature = "egl")]
|
2021-05-30 22:56:37 +00:00
|
|
|
let shell_handles =
|
|
|
|
init_shell::<BackendData>(&mut display.borrow_mut(), egl_reader.clone(), log.clone());
|
2021-05-13 17:59:47 +00:00
|
|
|
#[cfg(not(feature = "egl"))]
|
|
|
|
let shell_handles = init_shell(&mut display.borrow_mut(), log.clone());
|
2020-04-21 16:56:59 +00:00
|
|
|
|
|
|
|
let socket_name = display
|
|
|
|
.borrow_mut()
|
|
|
|
.add_socket_auto()
|
|
|
|
.unwrap()
|
|
|
|
.into_string()
|
|
|
|
.unwrap();
|
|
|
|
info!(log, "Listening on wayland socket"; "name" => socket_name.clone());
|
|
|
|
::std::env::set_var("WAYLAND_DISPLAY", &socket_name);
|
|
|
|
|
2020-04-21 17:42:03 +00:00
|
|
|
// init data device
|
|
|
|
|
2020-04-21 16:56:59 +00:00
|
|
|
let dnd_icon = Arc::new(Mutex::new(None));
|
|
|
|
|
|
|
|
let dnd_icon2 = dnd_icon.clone();
|
|
|
|
init_data_device(
|
|
|
|
&mut display.borrow_mut(),
|
|
|
|
move |event| match event {
|
|
|
|
DataDeviceEvent::DnDStarted { icon, .. } => {
|
|
|
|
*dnd_icon2.lock().unwrap() = icon;
|
|
|
|
}
|
|
|
|
DataDeviceEvent::DnDDropped => {
|
|
|
|
*dnd_icon2.lock().unwrap() = None;
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
},
|
|
|
|
default_action_chooser,
|
2020-05-02 10:42:02 +00:00
|
|
|
shell_handles.token,
|
2020-04-21 16:56:59 +00:00
|
|
|
log.clone(),
|
|
|
|
);
|
|
|
|
|
2020-05-04 13:04:30 +00:00
|
|
|
// init input
|
2021-05-30 22:56:37 +00:00
|
|
|
let seat_name = backend_data.seat_name();
|
2020-05-04 13:04:30 +00:00
|
|
|
|
|
|
|
let (mut seat, _) = Seat::new(
|
|
|
|
&mut display.borrow_mut(),
|
|
|
|
seat_name.clone(),
|
|
|
|
shell_handles.token,
|
|
|
|
log.clone(),
|
|
|
|
);
|
|
|
|
|
|
|
|
let cursor_status = Arc::new(Mutex::new(CursorImageStatus::Default));
|
|
|
|
|
|
|
|
let cursor_status2 = cursor_status.clone();
|
2020-06-07 13:29:37 +00:00
|
|
|
let pointer = seat.add_pointer(shell_handles.token, move |new_status| {
|
2020-05-04 13:04:30 +00:00
|
|
|
// TODO: hide winit system cursor when relevant
|
|
|
|
*cursor_status2.lock().unwrap() = new_status
|
|
|
|
});
|
|
|
|
|
|
|
|
let keyboard = seat
|
|
|
|
.add_keyboard(XkbConfig::default(), 200, 25, |seat, focus| {
|
|
|
|
set_data_device_focus(seat, focus.and_then(|s| s.as_ref().client()))
|
|
|
|
})
|
|
|
|
.expect("Failed to initialize the keyboard");
|
|
|
|
|
2021-01-03 15:21:22 +00:00
|
|
|
#[cfg(feature = "xwayland")]
|
|
|
|
let _xwayland = {
|
2021-01-04 08:46:59 +00:00
|
|
|
let xwm = XWm::new(
|
|
|
|
handle.clone(),
|
|
|
|
shell_handles.token,
|
|
|
|
shell_handles.window_map.clone(),
|
|
|
|
log.clone(),
|
|
|
|
);
|
2021-01-03 15:21:22 +00:00
|
|
|
XWayland::init(xwm, handle.clone(), display.clone(), &mut (), log.clone()).unwrap()
|
|
|
|
};
|
|
|
|
|
2020-04-21 16:56:59 +00:00
|
|
|
AnvilState {
|
2021-05-30 22:56:37 +00:00
|
|
|
backend_data,
|
2020-04-21 16:56:59 +00:00
|
|
|
running: Arc::new(AtomicBool::new(true)),
|
|
|
|
display,
|
|
|
|
handle,
|
2020-05-02 10:42:02 +00:00
|
|
|
ctoken: shell_handles.token,
|
|
|
|
window_map: shell_handles.window_map,
|
2020-04-21 16:56:59 +00:00
|
|
|
dnd_icon,
|
|
|
|
log,
|
|
|
|
socket_name,
|
2020-05-04 13:04:30 +00:00
|
|
|
pointer,
|
|
|
|
keyboard,
|
|
|
|
cursor_status,
|
|
|
|
pointer_location: Rc::new(RefCell::new((0.0, 0.0))),
|
|
|
|
seat_name,
|
2021-05-30 22:56:37 +00:00
|
|
|
egl_reader,
|
2021-01-03 15:21:22 +00:00
|
|
|
#[cfg(feature = "xwayland")]
|
|
|
|
_xwayland,
|
2020-04-21 16:56:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-05-30 22:56:37 +00:00
|
|
|
|
|
|
|
pub trait Backend {
|
|
|
|
fn seat_name(&self) -> String;
|
|
|
|
}
|