cosmic-applets/cosmic-app-list/src/wayland_handler.rs

283 lines
8.6 KiB
Rust
Raw Normal View History

2023-11-06 16:46:44 +01:00
use crate::wayland_subscription::{ToplevelRequest, ToplevelUpdate, WaylandRequest, WaylandUpdate};
use std::os::{
fd::{FromRawFd, RawFd},
unix::net::UnixStream,
};
use cctk::{
2022-12-15 14:35:31 -05:00
sctk::{
self,
2023-11-06 16:46:44 +01:00
activation::{RequestData, RequestDataExt},
reexports::{calloop, calloop_wayland_source::WaylandSource},
2022-12-15 14:35:31 -05:00
seat::{SeatHandler, SeatState},
},
toplevel_info::{ToplevelInfoHandler, ToplevelInfoState},
2022-12-13 19:58:00 -05:00
toplevel_management::{ToplevelManagerHandler, ToplevelManagerState},
2023-11-06 16:46:44 +01:00
wayland_client::{
self,
protocol::{wl_seat::WlSeat, wl_surface::WlSurface},
WEnum,
},
2022-12-16 09:52:18 -05:00
};
use cosmic_protocols::{
toplevel_info::v1::client::zcosmic_toplevel_handle_v1,
toplevel_management::v1::client::zcosmic_toplevel_manager_v1,
};
use futures::channel::mpsc::UnboundedSender;
2023-11-06 16:46:44 +01:00
use sctk::{
activation::{ActivationHandler, ActivationState},
registry::{ProvidesRegistryState, RegistryState},
};
use wayland_client::{globals::registry_queue_init, Connection, QueueHandle};
struct AppData {
exit: bool,
2023-11-06 16:46:44 +01:00
tx: UnboundedSender<WaylandUpdate>,
queue_handle: QueueHandle<Self>,
registry_state: RegistryState,
2023-11-06 16:46:44 +01:00
activation_state: Option<ActivationState>,
toplevel_info_state: ToplevelInfoState,
2022-12-13 19:58:00 -05:00
toplevel_manager_state: ToplevelManagerState,
seat_state: SeatState,
}
impl ProvidesRegistryState for AppData {
fn registry(&mut self) -> &mut RegistryState {
&mut self.registry_state
}
sctk::registry_handlers!();
}
2023-11-06 16:46:44 +01:00
struct ExecRequestData {
data: RequestData,
exec: String,
2024-01-31 12:44:56 +00:00
gpu_idx: Option<usize>,
2023-11-06 16:46:44 +01:00
}
impl RequestDataExt for ExecRequestData {
fn app_id(&self) -> Option<&str> {
self.data.app_id()
}
fn seat_and_serial(&self) -> Option<(&WlSeat, u32)> {
self.data.seat_and_serial()
}
fn surface(&self) -> Option<&WlSurface> {
self.data.surface()
}
}
impl ActivationHandler for AppData {
type RequestData = ExecRequestData;
fn new_token(&mut self, token: String, data: &ExecRequestData) {
let _ = self.tx.unbounded_send(WaylandUpdate::ActivationToken {
token: Some(token),
exec: data.exec.clone(),
2024-01-31 12:44:56 +00:00
gpu_idx: data.gpu_idx,
2023-11-06 16:46:44 +01:00
});
}
}
2022-12-13 19:58:00 -05:00
impl SeatHandler for AppData {
fn seat_state(&mut self) -> &mut sctk::seat::SeatState {
&mut self.seat_state
}
2022-12-15 14:35:31 -05:00
fn new_seat(&mut self, _: &Connection, _: &QueueHandle<Self>, _: WlSeat) {}
2022-12-13 19:58:00 -05:00
fn new_capability(
&mut self,
2022-12-15 14:35:31 -05:00
_: &Connection,
_: &QueueHandle<Self>,
_: WlSeat,
_: sctk::seat::Capability,
) {
}
2022-12-13 19:58:00 -05:00
fn remove_capability(
&mut self,
2022-12-15 14:35:31 -05:00
_: &Connection,
_: &QueueHandle<Self>,
_: WlSeat,
_: sctk::seat::Capability,
) {
}
2022-12-13 19:58:00 -05:00
2022-12-15 14:35:31 -05:00
fn remove_seat(&mut self, _: &Connection, _: &QueueHandle<Self>, _: WlSeat) {}
2022-12-13 19:58:00 -05:00
}
impl ToplevelManagerHandler for AppData {
fn toplevel_manager_state(&mut self) -> &mut cctk::toplevel_management::ToplevelManagerState {
&mut self.toplevel_manager_state
}
2022-12-16 09:52:18 -05:00
fn capabilities(
&mut self,
_: &Connection,
_: &QueueHandle<Self>,
_: Vec<WEnum<zcosmic_toplevel_manager_v1::ZcosmicToplelevelManagementCapabilitiesV1>>,
) {
2022-12-13 19:58:00 -05:00
// TODO capabilities could affect the options in the applet
}
}
impl ToplevelInfoHandler for AppData {
fn toplevel_info_state(&mut self) -> &mut ToplevelInfoState {
&mut self.toplevel_info_state
}
fn new_toplevel(
&mut self,
_conn: &Connection,
_qh: &QueueHandle<Self>,
toplevel: &zcosmic_toplevel_handle_v1::ZcosmicToplevelHandleV1,
) {
if let Some(info) = self.toplevel_info_state.info(toplevel) {
let _ = self
.tx
2023-11-16 18:20:57 +00:00
.unbounded_send(WaylandUpdate::Toplevel(ToplevelUpdate::Add(
2023-11-06 16:46:44 +01:00
toplevel.clone(),
info.clone(),
)));
}
}
fn update_toplevel(
&mut self,
_conn: &Connection,
_qh: &QueueHandle<Self>,
toplevel: &zcosmic_toplevel_handle_v1::ZcosmicToplevelHandleV1,
) {
if let Some(info) = self.toplevel_info_state.info(toplevel) {
2023-11-16 18:20:57 +00:00
let _ = self
.tx
.unbounded_send(WaylandUpdate::Toplevel(ToplevelUpdate::Update(
toplevel.clone(),
info.clone(),
)));
}
}
fn toplevel_closed(
&mut self,
_conn: &Connection,
_qh: &QueueHandle<Self>,
toplevel: &zcosmic_toplevel_handle_v1::ZcosmicToplevelHandleV1,
) {
let _ = self
.tx
2023-11-16 18:20:57 +00:00
.unbounded_send(WaylandUpdate::Toplevel(ToplevelUpdate::Remove(
2023-11-06 16:46:44 +01:00
toplevel.clone(),
)));
}
}
2023-11-06 16:46:44 +01:00
pub(crate) fn wayland_handler(
tx: UnboundedSender<WaylandUpdate>,
rx: calloop::channel::Channel<WaylandRequest>,
) {
let socket = std::env::var("X_PRIVILEGED_WAYLAND_SOCKET")
.ok()
.and_then(|fd| {
fd.parse::<RawFd>()
.ok()
.map(|fd| unsafe { UnixStream::from_raw_fd(fd) })
});
let conn = if let Some(socket) = socket {
Connection::from_socket(socket).unwrap()
} else {
Connection::connect_to_env().unwrap()
};
let (globals, event_queue) = registry_queue_init(&conn).unwrap();
2023-11-06 16:46:44 +01:00
let mut event_loop = calloop::EventLoop::<AppData>::try_new().unwrap();
let qh = event_queue.handle();
let wayland_source = WaylandSource::new(conn, event_queue);
let handle = event_loop.handle();
2023-07-25 14:13:05 -04:00
wayland_source
.insert(handle.clone())
.expect("Failed to insert wayland source.");
if handle
.insert_source(rx, |event, _, state| match event {
calloop::channel::Event::Msg(req) => match req {
2023-11-06 16:46:44 +01:00
WaylandRequest::Toplevel(req) => match req {
ToplevelRequest::Activate(handle) => {
if let Some(seat) = state.seat_state.seats().next() {
let manager = &state.toplevel_manager_state.manager;
manager.activate(&handle, &seat);
}
}
ToplevelRequest::Quit(handle) => {
let manager = &state.toplevel_manager_state.manager;
2023-11-06 16:46:44 +01:00
manager.close(&handle);
}
},
2024-01-31 12:44:56 +00:00
WaylandRequest::TokenRequest {
app_id,
exec,
gpu_idx,
} => {
2023-11-06 16:46:44 +01:00
if let Some(activation_state) = state.activation_state.as_ref() {
activation_state.request_token_with_data(
&state.queue_handle,
ExecRequestData {
data: RequestData {
app_id: Some(app_id),
seat_and_serial: state
.seat_state
.seats()
.next()
.map(|seat| (seat, 0)),
surface: None,
},
exec,
2024-01-31 12:44:56 +00:00
gpu_idx,
2023-11-06 16:46:44 +01:00
},
);
} else {
2024-01-31 12:44:56 +00:00
let _ = state.tx.unbounded_send(WaylandUpdate::ActivationToken {
token: None,
exec,
gpu_idx,
});
}
}
},
calloop::channel::Event::Closed => {
state.exit = true;
}
})
.is_err()
{
return;
}
let registry_state = RegistryState::new(&globals);
let mut app_data = AppData {
exit: false,
tx,
2023-11-06 16:46:44 +01:00
queue_handle: qh.clone(),
activation_state: ActivationState::bind::<AppData>(&globals, &qh).ok(),
2022-12-13 19:58:00 -05:00
seat_state: SeatState::new(&globals, &qh),
toplevel_info_state: ToplevelInfoState::new(&registry_state, &qh),
2022-12-13 19:58:00 -05:00
toplevel_manager_state: ToplevelManagerState::new(&registry_state, &qh),
registry_state,
};
loop {
if app_data.exit {
break;
}
event_loop.dispatch(None, &mut app_data).unwrap();
}
}
2023-11-06 16:46:44 +01:00
sctk::delegate_activation!(AppData, ExecRequestData);
2022-12-13 19:58:00 -05:00
sctk::delegate_seat!(AppData);
sctk::delegate_registry!(AppData);
cctk::delegate_toplevel_info!(AppData);
2022-12-13 19:58:00 -05:00
cctk::delegate_toplevel_manager!(AppData);