use std::{collections::HashMap, fmt::Debug, os::fd::OwnedFd};
use enumflags2::{bitflags, BitFlags};
use futures_util::TryFutureExt;
use serde::Deserialize;
use serde_repr::{Deserialize_repr, Serialize_repr};
use zbus::zvariant::{self, DeserializeDict, SerializeDict, Type, Value};
use super::{HandleToken, Request, Session};
use crate::{proxy::Proxy, Error, WindowIdentifier};
#[bitflags]
#[derive(Serialize_repr, Deserialize_repr, PartialEq, Eq, Copy, Clone, Debug, Type)]
#[repr(u32)]
#[doc(alias = "XdpOutputType")]
pub enum SourceType {
#[doc(alias = "XDP_OUTPUT_MONITOR")]
Monitor,
#[doc(alias = "XDP_OUTPUT_WINDOW")]
Window,
#[doc(alias = "XDP_OUTPUT_VIRTUAL")]
Virtual,
}
#[bitflags]
#[derive(Serialize_repr, Deserialize_repr, PartialEq, Eq, Debug, Copy, Clone, Type)]
#[repr(u32)]
#[doc(alias = "XdpCursorMode")]
pub enum CursorMode {
#[doc(alias = "XDP_CURSOR_MODE_HIDDEN")]
Hidden,
#[doc(alias = "XDP_CURSOR_MODE_EMBEDDED")]
Embedded,
#[doc(alias = "XDP_CURSOR_MODE_METADATA")]
Metadata,
}
#[cfg_attr(feature = "glib", derive(glib::Enum))]
#[cfg_attr(feature = "glib", enum_type(name = "AshpdPersistMode"))]
#[derive(Default, Serialize_repr, PartialEq, Eq, Debug, Copy, Clone, Type)]
#[doc(alias = "XdpPersistMode")]
#[repr(u32)]
pub enum PersistMode {
#[doc(alias = "XDP_PERSIST_MODE_NONE")]
#[default]
DoNot = 0,
#[doc(alias = "XDP_PERSIST_MODE_TRANSIENT")]
Application = 1,
#[doc(alias = "XDP_PERSIST_MODE_PERSISTENT")]
ExplicitlyRevoked = 2,
}
#[derive(SerializeDict, Type, Debug, Default)]
#[zvariant(signature = "dict")]
struct CreateSessionOptions {
handle_token: HandleToken,
session_handle_token: HandleToken,
}
#[derive(SerializeDict, Type, Debug, Default)]
#[zvariant(signature = "dict")]
struct SelectSourcesOptions {
handle_token: HandleToken,
types: Option<BitFlags<SourceType>>,
multiple: Option<bool>,
cursor_mode: Option<CursorMode>,
restore_token: Option<String>,
persist_mode: Option<PersistMode>,
}
impl SelectSourcesOptions {
#[must_use]
pub fn multiple(mut self, multiple: impl Into<Option<bool>>) -> Self {
self.multiple = multiple.into();
self
}
#[must_use]
pub fn cursor_mode(mut self, cursor_mode: impl Into<Option<CursorMode>>) -> Self {
self.cursor_mode = cursor_mode.into();
self
}
#[must_use]
pub fn types(mut self, types: impl Into<Option<BitFlags<SourceType>>>) -> Self {
self.types = types.into();
self
}
#[must_use]
pub fn persist_mode(mut self, persist_mode: impl Into<Option<PersistMode>>) -> Self {
self.persist_mode = persist_mode.into();
self
}
#[must_use]
pub fn restore_token<'a>(mut self, token: impl Into<Option<&'a str>>) -> Self {
self.restore_token = token.into().map(ToOwned::to_owned);
self
}
}
#[derive(SerializeDict, Type, Debug, Default)]
#[zvariant(signature = "dict")]
struct StartCastOptions {
handle_token: HandleToken,
}
#[derive(DeserializeDict, Type, Debug)]
#[zvariant(signature = "dict")]
struct CreateSession {
session_handle: String,
}
#[derive(DeserializeDict, Type)]
#[zvariant(signature = "dict")]
pub struct Streams {
streams: Vec<Stream>,
restore_token: Option<String>,
}
impl Streams {
pub fn restore_token(&self) -> Option<&str> {
self.restore_token.as_deref()
}
pub fn streams(&self) -> &[Stream] {
&self.streams
}
}
impl Debug for Streams {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_tuple("Streams")
.field(&self.restore_token)
.field(&self.streams)
.finish()
}
}
#[derive(Clone, Deserialize, Type)]
pub struct Stream(u32, StreamProperties);
impl Stream {
pub fn pipe_wire_node_id(&self) -> u32 {
self.0
}
pub fn position(&self) -> Option<(i32, i32)> {
self.1.position
}
pub fn size(&self) -> Option<(i32, i32)> {
self.1.size
}
pub fn source_type(&self) -> Option<SourceType> {
self.1.source_type
}
pub fn id(&self) -> Option<&str> {
self.1.id.as_deref()
}
pub fn mapping_id(&self) -> Option<&str> {
self.1.mapping_id.as_deref()
}
}
impl Debug for Stream {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("Stream")
.field("pipewire_node_id", &self.pipe_wire_node_id())
.field("position", &self.position())
.field("size", &self.size())
.field("source_type", &self.source_type())
.field("id", &self.id())
.finish()
}
}
#[derive(Clone, DeserializeDict, Type, Debug)]
#[zvariant(signature = "dict")]
struct StreamProperties {
id: Option<String>,
position: Option<(i32, i32)>,
size: Option<(i32, i32)>,
source_type: Option<SourceType>,
mapping_id: Option<String>,
}
#[derive(Debug)]
#[doc(alias = "org.freedesktop.portal.ScreenCast")]
pub struct Screencast<'a>(Proxy<'a>);
impl<'a> Screencast<'a> {
pub async fn new() -> Result<Screencast<'a>, Error> {
let proxy = Proxy::new_desktop("org.freedesktop.portal.ScreenCast").await?;
Ok(Self(proxy))
}
#[doc(alias = "CreateSession")]
#[doc(alias = "xdp_portal_create_screencast_session")]
pub async fn create_session(&self) -> Result<Session<'a>, Error> {
let options = CreateSessionOptions::default();
let (request, proxy) = futures_util::try_join!(
self.0
.request::<CreateSession>(&options.handle_token, "CreateSession", &options)
.into_future(),
Session::from_unique_name(&options.session_handle_token).into_future(),
)?;
assert_eq!(proxy.path().as_str(), &request.response()?.session_handle);
Ok(proxy)
}
#[doc(alias = "OpenPipeWireRemote")]
pub async fn open_pipe_wire_remote(&self, session: &Session<'_>) -> Result<OwnedFd, Error> {
let options: HashMap<&str, Value<'_>> = HashMap::new();
let fd = self
.0
.call::<zvariant::OwnedFd>("OpenPipeWireRemote", &(session, options))
.await?;
Ok(fd.into())
}
#[doc(alias = "SelectSources")]
pub async fn select_sources(
&self,
session: &Session<'_>,
cursor_mode: CursorMode,
types: BitFlags<SourceType>,
multiple: bool,
restore_token: Option<&str>,
persist_mode: PersistMode,
) -> Result<Request<()>, Error> {
let options = SelectSourcesOptions::default()
.cursor_mode(cursor_mode)
.multiple(multiple)
.types(types)
.persist_mode(persist_mode)
.restore_token(restore_token);
self.0
.empty_request(&options.handle_token, "SelectSources", &(session, &options))
.await
}
#[doc(alias = "Start")]
pub async fn start(
&self,
session: &Session<'_>,
identifier: &WindowIdentifier,
) -> Result<Request<Streams>, Error> {
let options = StartCastOptions::default();
self.0
.request(
&options.handle_token,
"Start",
&(session, &identifier, &options),
)
.await
}
#[doc(alias = "AvailableCursorModes")]
pub async fn available_cursor_modes(&self) -> Result<BitFlags<CursorMode>, Error> {
self.0.property_versioned("AvailableCursorModes", 2).await
}
#[doc(alias = "AvailableSourceTypes")]
pub async fn available_source_types(&self) -> Result<BitFlags<SourceType>, Error> {
self.0.property("AvailableSourceTypes").await
}
}
impl<'a> std::ops::Deref for Screencast<'a> {
type Target = zbus::Proxy<'a>;
fn deref(&self) -> &Self::Target {
&self.0
}
}