use std::{
os::unix::io::{AsFd, AsRawFd, BorrowedFd},
os::unix::net::UnixStream,
sync::Arc,
};
use wayland_backend::{
protocol::ObjectInfo,
server::{Backend, ClientData, GlobalId, Handle, InitError, InvalidId, ObjectId},
};
use crate::{
global::{GlobalData, GlobalDispatch},
Client, Resource,
};
#[derive(Debug)]
pub struct Display<State: 'static> {
backend: Backend<State>,
}
impl<State: 'static> Display<State> {
pub fn new() -> Result<Display<State>, InitError> {
Ok(Display { backend: Backend::new()? })
}
pub fn handle(&self) -> DisplayHandle {
DisplayHandle { handle: self.backend.handle() }
}
pub fn dispatch_clients(&mut self, state: &mut State) -> std::io::Result<usize> {
self.backend.dispatch_all_clients(state)
}
pub fn flush_clients(&mut self) -> std::io::Result<()> {
self.backend.flush(None)
}
pub fn backend(&mut self) -> &mut Backend<State> {
&mut self.backend
}
}
impl<State> AsFd for Display<State> {
fn as_fd(&self) -> BorrowedFd<'_> {
self.backend.poll_fd()
}
}
#[derive(Clone)]
pub struct DisplayHandle {
pub(crate) handle: Handle,
}
impl std::fmt::Debug for DisplayHandle {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("DisplayHandle").finish_non_exhaustive()
}
}
impl DisplayHandle {
pub fn backend_handle(&self) -> Handle {
self.handle.clone()
}
pub fn insert_client(
&mut self,
stream: UnixStream,
data: Arc<dyn ClientData>,
) -> std::io::Result<Client> {
let id = self.handle.insert_client(stream, data.clone())?;
Ok(Client { id, data })
}
pub fn get_client(&self, id: ObjectId) -> Result<Client, InvalidId> {
let client_id = self.handle.get_client(id)?;
Client::from_id(self, client_id)
}
pub fn create_global<State, I: Resource + 'static, U: Send + Sync + 'static>(
&self,
version: u32,
data: U,
) -> GlobalId
where
State: GlobalDispatch<I, U> + 'static,
{
self.handle.create_global::<State>(
I::interface(),
version,
Arc::new(GlobalData { data, _types: std::marker::PhantomData }),
)
}
pub fn disable_global<State: 'static>(&self, id: GlobalId) {
self.handle.disable_global::<State>(id)
}
pub fn remove_global<State: 'static>(&self, id: GlobalId) {
self.handle.remove_global::<State>(id)
}
pub fn object_info(&self, id: ObjectId) -> Result<ObjectInfo, InvalidId> {
self.handle.object_info(id)
}
pub fn send_event<I: Resource>(
&self,
resource: &I,
event: I::Event<'_>,
) -> Result<(), InvalidId> {
let msg = resource.write_event(self, event)?;
let msg = msg.map_fd(|fd| fd.as_raw_fd());
self.handle.send_event(msg)
}
pub fn post_error<I: Resource>(&self, resource: &I, code: u32, error: String) {
self.handle.post_error(resource.id(), code, std::ffi::CString::new(error).unwrap())
}
pub fn get_object_data(
&self,
id: ObjectId,
) -> Result<Arc<dyn std::any::Any + Send + Sync + 'static>, InvalidId> {
self.handle.get_object_data_any(id)
}
pub fn flush_clients(&mut self) -> std::io::Result<()> {
self.handle.flush(None)
}
#[cfg(feature = "libwayland_1_23")]
pub fn set_default_max_buffer_size(&self, max_buffer_size: usize) {
self.handle.set_default_max_buffer_size(max_buffer_size);
}
}
impl From<Handle> for DisplayHandle {
fn from(handle: Handle) -> Self {
Self { handle }
}
}