use { crate::{ client::{CAP_SESSION_LOCK_MANAGER, Client, ClientCaps, ClientError}, globals::{Global, GlobalName}, ifs::ext_session_lock_v1::ExtSessionLockV1, leaks::Tracker, object::{Object, Version}, wire::{ExtSessionLockManagerV1Id, ext_session_lock_manager_v1::*}, }, std::{cell::Cell, rc::Rc}, thiserror::Error, }; pub struct ExtSessionLockManagerV1Global { pub name: GlobalName, } impl ExtSessionLockManagerV1Global { pub fn new(name: GlobalName) -> Self { Self { name } } fn bind_( self: Rc, id: ExtSessionLockManagerV1Id, client: &Rc, version: Version, ) -> Result<(), ExtSessionLockManagerV1Error> { let obj = Rc::new(ExtSessionLockManagerV1 { id, client: client.clone(), tracker: Default::default(), version, }); track!(client, obj); client.add_client_obj(&obj)?; Ok(()) } } pub struct ExtSessionLockManagerV1 { pub id: ExtSessionLockManagerV1Id, pub client: Rc, pub tracker: Tracker, pub version: Version, } impl ExtSessionLockManagerV1RequestHandler for ExtSessionLockManagerV1 { type Error = ExtSessionLockManagerV1Error; fn destroy(&self, _req: Destroy, _slf: &Rc) -> Result<(), Self::Error> { self.client.remove_obj(self)?; Ok(()) } fn lock(&self, req: Lock, _slf: &Rc) -> Result<(), Self::Error> { let did_lock = self.client.state.lock.locked.get() == false; let new = Rc::new(ExtSessionLockV1 { id: req.id, client: self.client.clone(), tracker: Default::default(), did_lock, awaiting_locked: Cell::new(true), finished: Cell::new(false), version: self.version, }); track!(new.client, new); self.client.add_client_obj(&new)?; if did_lock { log::info!("Client {} locks the screen", self.client.id); let state = &self.client.state; for seat in state.globals.seats.lock().values() { seat.prepare_for_lock(); } state.lock.locked.set(true); state.lock.lock.set(Some(new.clone())); state.tree_changed(); state.damage(state.root.extents.get()); new.check_locked(); } else { new.finish(); } Ok(()) } } global_base!( ExtSessionLockManagerV1Global, ExtSessionLockManagerV1, ExtSessionLockManagerV1Error ); impl Global for ExtSessionLockManagerV1Global { fn version(&self) -> u32 { 1 } fn required_caps(&self) -> ClientCaps { CAP_SESSION_LOCK_MANAGER } } simple_add_global!(ExtSessionLockManagerV1Global); object_base! { self = ExtSessionLockManagerV1; version = self.version; } impl Object for ExtSessionLockManagerV1 {} simple_add_obj!(ExtSessionLockManagerV1); #[derive(Debug, Error)] pub enum ExtSessionLockManagerV1Error { #[error(transparent)] ClientError(Box), } efrom!(ExtSessionLockManagerV1Error, ClientError);