use { crate::{ client::{Client, ClientError, ClientId}, ifs::{ ipc::{ break_device_loops, destroy_data_device, zwp_primary_selection_offer_v1::ZwpPrimarySelectionOfferV1, zwp_primary_selection_source_v1::ZwpPrimarySelectionSourceV1, DeviceData, IpcVtable, OfferData, Role, XIpcVtable, }, wl_seat::{WlSeatError, WlSeatGlobal}, }, leaks::Tracker, object::Object, utils::buffd::{MsgParser, MsgParserError}, wire::{ zwp_primary_selection_device_v1::*, ZwpPrimarySelectionDeviceV1Id, ZwpPrimarySelectionOfferV1Id, ZwpPrimarySelectionSourceV1Id, }, xwayland::XWaylandEvent, }, std::rc::Rc, thiserror::Error, }; pub struct ZwpPrimarySelectionDeviceV1 { pub id: ZwpPrimarySelectionDeviceV1Id, pub client: Rc, pub version: u32, pub seat: Rc, data: DeviceData, pub tracker: Tracker, } impl ZwpPrimarySelectionDeviceV1 { pub fn new( id: ZwpPrimarySelectionDeviceV1Id, client: &Rc, version: u32, seat: &Rc, is_xwm: bool, ) -> Self { Self { id, client: client.clone(), version, seat: seat.clone(), data: DeviceData { selection: Default::default(), dnd: Default::default(), is_xwm, }, tracker: Default::default(), } } pub fn send_data_offer(&self, offer: &Rc) { if self.data.is_xwm { self.client .state .xwayland .queue .push(XWaylandEvent::PrimarySelectionSetOffer(offer.clone())); } else { self.client.event(DataOffer { self_id: self.id, offer: offer.id, }) } } pub fn send_selection(&self, offer: Option<&Rc>) { if self.data.is_xwm { self.client .state .xwayland .queue .push(XWaylandEvent::PrimarySelectionSetSelection( self.seat.id(), offer.cloned(), )); } else { let id = offer .map(|o| o.id) .unwrap_or(ZwpPrimarySelectionOfferV1Id::NONE); self.client.event(Selection { self_id: self.id, id, }) } } fn set_selection( &self, parser: MsgParser<'_, '_>, ) -> Result<(), ZwpPrimarySelectionDeviceV1Error> { let req: SetSelection = self.client.parse(self, parser)?; if !self.client.valid_serial(req.serial) { log::warn!("Client tried to set_selection with an invalid serial"); return Ok(()); } if !self .seat .may_modify_primary_selection(&self.client, Some(req.serial)) { log::warn!("Ignoring disallowed set_selection request"); return Ok(()); } let src = if req.source.is_none() { None } else { Some(self.client.lookup(req.source)?) }; self.seat.set_zwp_primary_selection(src, Some(req.serial))?; Ok(()) } fn destroy(&self, parser: MsgParser<'_, '_>) -> Result<(), ZwpPrimarySelectionDeviceV1Error> { let _req: Destroy = self.client.parse(self, parser)?; destroy_data_device::(self); self.seat.remove_primary_selection_device(self); self.client.remove_obj(self)?; Ok(()) } } pub struct PrimarySelectionIpc; impl XIpcVtable for PrimarySelectionIpc { fn create_xwm_source(client: &Rc) -> Self::Source { ZwpPrimarySelectionSourceV1::new(ZwpPrimarySelectionSourceV1Id::NONE, client, true) } fn remove_from_seat(device: &Self::Device) { device.seat.remove_primary_selection_device(device); } } impl IpcVtable for PrimarySelectionIpc { type Device = ZwpPrimarySelectionDeviceV1; type Source = ZwpPrimarySelectionSourceV1; type Offer = ZwpPrimarySelectionOfferV1; fn get_device_data(dd: &Self::Device) -> &DeviceData { &dd.data } fn get_device_seat(dd: &Self::Device) -> Rc { dd.seat.clone() } fn set_seat_selection( seat: &Rc, source: &Rc, serial: Option, ) -> Result<(), WlSeatError> { seat.set_zwp_primary_selection(Some(source.clone()), serial) } fn for_each_device(seat: &WlSeatGlobal, client: ClientId, f: C) where C: FnMut(&Rc), { seat.for_each_primary_selection_device(0, client, f) } fn create_offer( client: &Rc, device: &Rc, offer_data: OfferData, ) -> Result, ClientError> { let id = if device.data.is_xwm { ZwpPrimarySelectionOfferV1Id::NONE } else { client.new_id()? }; let rc = Rc::new(ZwpPrimarySelectionOfferV1 { id, offer_id: client.state.data_offer_ids.next(), seat: device.seat.clone(), client: client.clone(), data: offer_data, tracker: Default::default(), }); track!(client, rc); Ok(rc) } fn send_selection(dd: &Self::Device, offer: Option<&Rc>) { dd.send_selection(offer); } fn send_offer(dd: &Self::Device, offer: &Rc) { dd.send_data_offer(offer); } fn unset(seat: &Rc, _role: Role) { seat.unset_primary_selection(); } } object_base! { self = ZwpPrimarySelectionDeviceV1; SET_SELECTION => set_selection, DESTROY => destroy, } impl Object for ZwpPrimarySelectionDeviceV1 { fn break_loops(&self) { break_device_loops::(self); self.seat.remove_primary_selection_device(self); } } simple_add_obj!(ZwpPrimarySelectionDeviceV1); #[derive(Debug, Error)] pub enum ZwpPrimarySelectionDeviceV1Error { #[error(transparent)] ClientError(Box), #[error("Parsing failed")] MsgParserError(#[source] Box), #[error(transparent)] WlSeatError(Box), } efrom!(ZwpPrimarySelectionDeviceV1Error, ClientError); efrom!(ZwpPrimarySelectionDeviceV1Error, MsgParserError); efrom!(ZwpPrimarySelectionDeviceV1Error, WlSeatError);