use cctk::{ cosmic_protocols, sctk::{ self, event_loop::WaylandSource, reexports::{calloop, client::protocol::wl_seat::WlSeat}, seat::{SeatHandler, SeatState}, }, toplevel_info::{ToplevelInfo, ToplevelInfoHandler, ToplevelInfoState}, toplevel_management::{ToplevelManagerHandler, ToplevelManagerState}, wayland_client::{self, protocol::wl_output::WlOutput, WEnum}, }; use cosmic_protocols::{ toplevel_info::v1::client::zcosmic_toplevel_handle_v1::{self, ZcosmicToplevelHandleV1}, toplevel_management::v1::client::zcosmic_toplevel_manager_v1, workspace::v1::server::zcosmic_workspace_handle_v1::ZcosmicWorkspaceHandleV1, }; use futures::channel::mpsc::UnboundedSender; use sctk::registry::{ProvidesRegistryState, RegistryState}; use wayland_client::{globals::registry_queue_init, Connection, QueueHandle}; #[derive(Debug, Clone)] pub enum ToplevelAction { Activate(ZcosmicToplevelHandleV1), Close(ZcosmicToplevelHandleV1), } #[derive(Debug, Clone)] pub enum ToplevelEvent { Add(ZcosmicToplevelHandleV1, ToplevelInfo), Remove(ZcosmicToplevelHandleV1), Update(ZcosmicToplevelHandleV1, ToplevelInfo), } #[derive(Debug, Clone)] pub struct Toplevel { pub name: String, pub app_id: String, pub toplevel_handle: ZcosmicToplevelHandleV1, pub states: Vec, pub output: Option, pub workspace: Option, } struct AppData { exit: bool, tx: UnboundedSender, registry_state: RegistryState, toplevel_info_state: ToplevelInfoState, toplevel_manager_state: ToplevelManagerState, seat_state: SeatState, } impl ProvidesRegistryState for AppData { fn registry(&mut self) -> &mut RegistryState { &mut self.registry_state } sctk::registry_handlers!(); } impl SeatHandler for AppData { fn seat_state(&mut self) -> &mut sctk::seat::SeatState { &mut self.seat_state } fn new_seat(&mut self, _: &Connection, _: &QueueHandle, _: WlSeat) {} fn new_capability( &mut self, _: &Connection, _: &QueueHandle, _: WlSeat, _: sctk::seat::Capability, ) { } fn remove_capability( &mut self, _: &Connection, _: &QueueHandle, _: WlSeat, _: sctk::seat::Capability, ) { } fn remove_seat(&mut self, _: &Connection, _: &QueueHandle, _: WlSeat) {} } impl ToplevelManagerHandler for AppData { fn toplevel_manager_state(&mut self) -> &mut cctk::toplevel_management::ToplevelManagerState { &mut self.toplevel_manager_state } fn capabilities( &mut self, _: &Connection, _: &QueueHandle, _: Vec>, ) { } } 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, toplevel: &zcosmic_toplevel_handle_v1::ZcosmicToplevelHandleV1, ) { if let Some(info) = self.toplevel_info_state.info(toplevel) { let _ = self .tx .unbounded_send(ToplevelEvent::Add(toplevel.clone(), info.clone())); } } fn update_toplevel( &mut self, _conn: &Connection, _qh: &QueueHandle, toplevel: &zcosmic_toplevel_handle_v1::ZcosmicToplevelHandleV1, ) { if let Some(info) = self.toplevel_info_state.info(toplevel) { let _ = self .tx .unbounded_send(ToplevelEvent::Update(toplevel.clone(), info.clone())); } } fn toplevel_closed( &mut self, _conn: &Connection, _qh: &QueueHandle, toplevel: &zcosmic_toplevel_handle_v1::ZcosmicToplevelHandleV1, ) { let _ = self .tx .unbounded_send(ToplevelEvent::Remove(toplevel.clone())); } } pub(crate) fn toplevel_handler( tx: UnboundedSender, rx: calloop::channel::Channel, ) -> anyhow::Result<()> { let conn = Connection::connect_to_env()?; let (globals, event_queue) = registry_queue_init(&conn)?; let mut event_loop = calloop::EventLoop::::try_new()?; let qh = event_queue.handle(); let wayland_source = WaylandSource::new(event_queue)?; let handle = event_loop.handle(); handle.insert_source(wayland_source, |_, q, state| q.dispatch_pending(state))?; let _ = handle.insert_source(rx, |event, _, state| match event { calloop::channel::Event::Msg(req) => match req { ToplevelAction::Activate(handle) => { let manager = &state.toplevel_manager_state.manager; let state = &state.seat_state; // TODO Ashley how to choose the seat in a multi-seat setup? for s in state.seats() { manager.activate(&handle, &s); } } ToplevelAction::Close(handle) => { let manager = &state.toplevel_manager_state.manager; manager.close(&handle); } }, calloop::channel::Event::Closed => { state.exit = true; } }); let registry_state = RegistryState::new(&globals); let mut app_data = AppData { exit: false, tx, seat_state: SeatState::new(&globals, &qh), toplevel_info_state: ToplevelInfoState::new(®istry_state, &qh), toplevel_manager_state: ToplevelManagerState::new(®istry_state, &qh), registry_state, }; loop { if app_data.exit { break Ok(()); } event_loop.dispatch(None, &mut app_data)?; } } sctk::delegate_seat!(AppData); sctk::delegate_registry!(AppData); cctk::delegate_toplevel_info!(AppData); cctk::delegate_toplevel_manager!(AppData);