use std::fmt::Debug;
use futures::TryFutureExt;
use serde::{Deserialize, Serialize};
use serde_repr::{Deserialize_repr, Serialize_repr};
use zbus::zvariant::{DeserializeDict, OwnedObjectPath, SerializeDict, Type};
use super::{HandleToken, SessionProxy, DESTINATION, PATH};
use crate::{
helpers::{call_basic_response_method, call_method, receive_signal},
Error, WindowIdentifier,
};
#[derive(Serialize_repr, Deserialize_repr, PartialEq, Clone, Copy, Debug, Type)]
#[doc(alias = "XdpLocationAccuracy")]
#[repr(u8)]
pub enum Accuracy {
#[doc(alias = "XDP_LOCATION_ACCURACY_NONE")]
None = 0,
#[doc(alias = "XDP_LOCATION_ACCURACY_COUNTRY")]
Country = 1,
#[doc(alias = "XDP_LOCATION_ACCURACY_CITY")]
City = 2,
#[doc(alias = "XDP_LOCATION_ACCURACY_NEIGHBORHOOD")]
Neighborhood = 3,
#[doc(alias = "XDP_LOCATION_ACCURACY_STREET")]
Street = 4,
#[doc(alias = "XDP_LOCATION_ACCURACY_EXACT")]
Exact = 5,
}
#[derive(SerializeDict, DeserializeDict, Type, Debug, Default)]
#[zvariant(signature = "dict")]
struct CreateSessionOptions {
session_handle_token: HandleToken,
#[zvariant(rename = "distance-threshold")]
distance_threshold: Option<u32>,
#[zvariant(rename = "time-threshold")]
time_threshold: Option<u32>,
accuracy: Option<Accuracy>,
}
impl CreateSessionOptions {
pub fn distance_threshold(mut self, distance_threshold: u32) -> Self {
self.distance_threshold = Some(distance_threshold);
self
}
pub fn time_threshold(mut self, time_threshold: u32) -> Self {
self.time_threshold = Some(time_threshold);
self
}
pub fn accuracy(mut self, accuracy: Accuracy) -> Self {
self.accuracy = Some(accuracy);
self
}
}
#[derive(SerializeDict, DeserializeDict, Type, Debug, Default)]
#[zvariant(signature = "dict")]
struct SessionStartOptions {
handle_token: HandleToken,
}
#[derive(Serialize, Deserialize, Type)]
pub struct Location(OwnedObjectPath, LocationInner);
impl Location {
pub fn accuracy(&self) -> f64 {
self.1.accuracy
}
pub fn altitude(&self) -> Option<f64> {
if self.1.altitude == -f64::MAX {
None
} else {
Some(self.1.altitude)
}
}
pub fn speed(&self) -> Option<f64> {
if self.1.speed == -1f64 {
None
} else {
Some(self.1.speed)
}
}
pub fn heading(&self) -> Option<f64> {
if self.1.heading == -1f64 {
None
} else {
Some(self.1.heading)
}
}
pub fn description(&self) -> Option<&str> {
if self.1.description.is_empty() {
None
} else {
Some(&self.1.description)
}
}
pub fn latitude(&self) -> f64 {
self.1.latitude
}
pub fn longitude(&self) -> f64 {
self.1.longitude
}
pub fn timestamp(&self) -> std::time::Duration {
std::time::Duration::from_secs(self.1.timestamp.0)
}
}
impl Debug for Location {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("Location")
.field("accuracy", &self.accuracy())
.field("altitude", &self.altitude())
.field("speed", &self.speed())
.field("heading", &self.heading())
.field("description", &self.description())
.field("latitude", &self.latitude())
.field("longitude", &self.longitude())
.field("timestamp", &self.timestamp())
.finish()
}
}
#[derive(Debug, SerializeDict, DeserializeDict, Type)]
#[zvariant(signature = "dict")]
struct LocationInner {
#[zvariant(rename = "Accuracy")]
accuracy: f64,
#[zvariant(rename = "Altitude")]
altitude: f64,
#[zvariant(rename = "Speed")]
speed: f64,
#[zvariant(rename = "Heading")]
heading: f64,
#[zvariant(rename = "Description")]
description: String,
#[zvariant(rename = "Latitude")]
latitude: f64,
#[zvariant(rename = "Longitude")]
longitude: f64,
#[zvariant(rename = "Timestamp")]
timestamp: (u64, u64),
}
#[derive(Debug)]
#[doc(alias = "org.freedesktop.portal.Location")]
pub struct LocationProxy<'a>(zbus::Proxy<'a>);
impl<'a> LocationProxy<'a> {
pub async fn new(connection: &zbus::Connection) -> Result<LocationProxy<'a>, Error> {
let proxy = zbus::ProxyBuilder::new_bare(connection)
.interface("org.freedesktop.portal.Location")?
.path(PATH)?
.destination(DESTINATION)?
.build()
.await?;
Ok(Self(proxy))
}
pub fn inner(&self) -> &zbus::Proxy<'_> {
&self.0
}
#[doc(alias = "LocationUpdated")]
#[doc(alias = "XdpPortal::location-updated")]
pub async fn receive_location_updated(&self) -> Result<Location, Error> {
receive_signal(&self.0, "LocationUpdated").await
}
#[doc(alias = "CreateSession")]
pub async fn create_session(
&self,
distance_threshold: Option<u32>,
time_threshold: Option<u32>,
accuracy: Option<Accuracy>,
) -> Result<SessionProxy<'a>, Error> {
let options = CreateSessionOptions::default()
.distance_threshold(distance_threshold.unwrap_or(0))
.time_threshold(time_threshold.unwrap_or(0))
.accuracy(accuracy.unwrap_or(Accuracy::Exact));
let (path, proxy) = futures::try_join!(
call_method::<OwnedObjectPath, CreateSessionOptions>(
&self.0,
"CreateSession",
&(options)
)
.into_future(),
SessionProxy::from_unique_name(self.0.connection(), &options.session_handle_token)
.into_future(),
)?;
assert_eq!(proxy.inner().path(), &path.into_inner());
Ok(proxy)
}
#[doc(alias = "Start")]
#[doc(alias = "xdp_portal_location_monitor_start")]
pub async fn start(
&self,
session: &SessionProxy<'_>,
identifier: &WindowIdentifier,
) -> Result<(), Error> {
let options = SessionStartOptions::default();
call_basic_response_method(
&self.0,
&options.handle_token,
"Start",
&(session, &identifier, &options),
)
.await
}
}