1//! Types for extensible location message events ([MSC3488]).
2//!
3//! [MSC3488]: https://github.com/matrix-org/matrix-spec-proposals/pull/3488
45use js_int::UInt;
6use ruma_macros::{EventContent, StringEnum};
7use serde::{Deserialize, Serialize};
89mod zoomlevel_serde;
1011use ruma_common::MilliSecondsSinceUnixEpoch;
1213use super::{message::TextContentBlock, room::message::Relation};
14use crate::PrivOwnedStr;
1516/// The payload for an extensible location message.
17///
18/// This is the new primary type introduced in [MSC3488] and should only be sent in rooms with a
19/// version that supports it. See the documentation of the [`message`] module for more information.
20///
21/// [MSC3488]: https://github.com/matrix-org/matrix-spec-proposals/pull/3488
22/// [`message`]: super::message
23#[derive(Clone, Debug, Serialize, Deserialize, EventContent)]
24#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
25#[ruma_event(type = "m.location", kind = MessageLike, without_relation)]
26pub struct LocationEventContent {
27/// The text representation of the message.
28#[serde(rename = "org.matrix.msc1767.text")]
29pub text: TextContentBlock,
3031/// The location info of the message.
32#[serde(rename = "m.location")]
33pub location: LocationContent,
3435/// The asset this message refers to.
36#[serde(default, rename = "m.asset", skip_serializing_if = "ruma_common::serde::is_default")]
37pub asset: AssetContent,
3839/// The timestamp this message refers to.
40#[serde(rename = "m.ts", skip_serializing_if = "Option::is_none")]
41pub ts: Option<MilliSecondsSinceUnixEpoch>,
4243/// Whether this message is automated.
44#[cfg(feature = "unstable-msc3955")]
45 #[serde(
46 default,
47 skip_serializing_if = "ruma_common::serde::is_default",
48 rename = "org.matrix.msc1767.automated"
49)]
50pub automated: bool,
5152/// Information about related messages.
53#[serde(
54 flatten,
55 skip_serializing_if = "Option::is_none",
56 deserialize_with = "crate::room::message::relation_serde::deserialize_relation"
57)]
58pub relates_to: Option<Relation<LocationEventContentWithoutRelation>>,
59}
6061impl LocationEventContent {
62/// Creates a new `LocationEventContent` with the given fallback representation and location.
63pub fn new(text: TextContentBlock, location: LocationContent) -> Self {
64Self {
65 text,
66 location,
67 asset: Default::default(),
68 ts: None,
69#[cfg(feature = "unstable-msc3955")]
70automated: false,
71 relates_to: None,
72 }
73 }
7475/// Creates a new `LocationEventContent` with the given plain text fallback representation and
76 /// location.
77pub fn with_plain_text(plain_text: impl Into<String>, location: LocationContent) -> Self {
78Self {
79 text: TextContentBlock::plain(plain_text),
80 location,
81 asset: Default::default(),
82 ts: None,
83#[cfg(feature = "unstable-msc3955")]
84automated: false,
85 relates_to: None,
86 }
87 }
88}
8990/// Location content.
91#[derive(Clone, Debug, Serialize, Deserialize)]
92#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
93pub struct LocationContent {
94/// A `geo:` URI representing the location.
95 ///
96 /// See [RFC 5870](https://datatracker.ietf.org/doc/html/rfc5870) for more details.
97pub uri: String,
9899/// The description of the location.
100 ///
101 /// It should be used to label the location on a map.
102#[serde(skip_serializing_if = "Option::is_none")]
103pub description: Option<String>,
104105/// A zoom level to specify the displayed area size.
106#[serde(skip_serializing_if = "Option::is_none")]
107pub zoom_level: Option<ZoomLevel>,
108}
109110impl LocationContent {
111/// Creates a new `LocationContent` with the given geo URI.
112pub fn new(uri: String) -> Self {
113Self { uri, description: None, zoom_level: None }
114 }
115}
116117/// An error encountered when trying to convert to a `ZoomLevel`.
118#[derive(Copy, Clone, Debug, Eq, Hash, PartialEq, thiserror::Error)]
119#[non_exhaustive]
120pub enum ZoomLevelError {
121/// The value is higher than [`ZoomLevel::MAX`].
122#[error("value too high")]
123TooHigh,
124}
125126/// A zoom level.
127///
128/// This is an integer between 0 and 20 as defined in the [OpenStreetMap Wiki].
129///
130/// [OpenStreetMap Wiki]: https://wiki.openstreetmap.org/wiki/Zoom_levels
131#[derive(Clone, Debug, Serialize)]
132pub struct ZoomLevel(UInt);
133134impl ZoomLevel {
135/// The smallest value of a `ZoomLevel`, 0.
136pub const MIN: u8 = 0;
137138/// The largest value of a `ZoomLevel`, 20.
139pub const MAX: u8 = 20;
140141/// Creates a new `ZoomLevel` with the given value.
142pub fn new(value: u8) -> Option<Self> {
143if value > Self::MAX {
144None
145} else {
146Some(Self(value.into()))
147 }
148 }
149150/// The value of this `ZoomLevel`.
151pub fn get(&self) -> UInt {
152self.0
153}
154}
155156impl TryFrom<u8> for ZoomLevel {
157type Error = ZoomLevelError;
158159fn try_from(value: u8) -> Result<Self, Self::Error> {
160Self::new(value).ok_or(ZoomLevelError::TooHigh)
161 }
162}
163164/// Asset content.
165#[derive(Clone, Debug, Default, PartialEq, Eq, Serialize, Deserialize)]
166#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
167pub struct AssetContent {
168/// The type of asset being referred to.
169#[serde(rename = "type")]
170pub type_: AssetType,
171}
172173impl AssetContent {
174/// Creates a new default `AssetContent`.
175pub fn new() -> Self {
176Self::default()
177 }
178}
179180/// The type of an asset.
181#[doc = include_str!(concat!(env!("CARGO_MANIFEST_DIR"), "/src/doc/string_enum.md"))]
182#[derive(Clone, Default, PartialEq, Eq, PartialOrd, Ord, StringEnum)]
183#[ruma_enum(rename_all = "m.snake_case")]
184#[non_exhaustive]
185pub enum AssetType {
186/// The asset is the sender of the event.
187#[default]
188 #[ruma_enum(rename = "m.self")]
189Self_,
190191/// The asset is a location pinned by the sender.
192Pin,
193194#[doc(hidden)]
195_Custom(PrivOwnedStr),
196}