1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
use std::ops::Not;

use crate::requests::*;
use crate::types::*;

/// Use this method to send point on the map.
#[derive(Debug, Clone, PartialEq, PartialOrd, Serialize)]
#[must_use = "requests do nothing unless sent"]
pub struct SendLocation {
    chat_id: ChatRef,
    latitude: Float,
    longitude: Float,
    #[serde(skip_serializing_if = "Option::is_none")]
    live_period: Option<Integer>,
    #[serde(skip_serializing_if = "Not::not")]
    disable_notification: bool,
    #[serde(skip_serializing_if = "Option::is_none")]
    reply_to_message_id: Option<MessageId>,
    #[serde(skip_serializing_if = "Option::is_none")]
    reply_markup: Option<ReplyMarkup>,
}

impl Request for SendLocation {
    type Type = JsonRequestType<Self>;
    type Response = JsonIdResponse<Message>;

    fn serialize(&self) -> Result<HttpRequest, Error> {
        Self::Type::serialize(RequestUrl::method("sendLocation"), self)
    }
}

impl SendLocation {
    pub fn new<C>(chat: C, latitude: Float, longitude: Float) -> Self
    where
        C: ToChatRef,
    {
        SendLocation {
            chat_id: chat.to_chat_ref(),
            latitude: latitude,
            longitude: longitude,
            live_period: None,
            disable_notification: false,
            reply_to_message_id: None,
            reply_markup: None,
        }
    }

    /// Period in seconds for which the location will be updated, should be between 60 and 86400.
    pub fn live_period(&mut self, period: Integer) -> &mut Self {
        self.live_period = Some(period);
        self
    }

    pub fn disable_notification(&mut self) -> &mut Self {
        self.disable_notification = true;
        self
    }

    pub fn reply_to<R>(&mut self, to: R) -> &mut Self
    where
        R: ToMessageId,
    {
        self.reply_to_message_id = Some(to.to_message_id());
        self
    }

    pub fn reply_markup<R>(&mut self, reply_markup: R) -> &mut Self
    where
        R: Into<ReplyMarkup>,
    {
        self.reply_markup = Some(reply_markup.into());
        self
    }
}

/// Send point on the map.
pub trait CanSendLocation {
    fn location(&self, latitude: Float, longitude: Float) -> SendLocation;
}

impl<C> CanSendLocation for C
where
    C: ToChatRef,
{
    fn location(&self, latitude: Float, longitude: Float) -> SendLocation {
        SendLocation::new(self, latitude, longitude)
    }
}

/// Reply with point on the map.
pub trait CanReplySendLocation {
    fn location_reply(&self, latitude: Float, longitude: Float) -> SendLocation;
}

impl<M> CanReplySendLocation for M
where
    M: ToMessageId + ToSourceChat,
{
    fn location_reply(&self, latitude: Float, longitude: Float) -> SendLocation {
        let mut rq = self.to_source_chat().location(latitude, longitude);
        rq.reply_to(self.to_message_id());
        rq
    }
}

impl<'b> ToRequest<'b> for Location {
    type Request = SendLocation;

    fn to_request<C>(&'b self, chat: C) -> Self::Request
    where
        C: ToChatRef,
    {
        chat.location(self.latitude, self.longitude)
    }
}

impl<'b> ToReplyRequest<'b> for Location {
    type Request = SendLocation;

    fn to_reply_request<M>(&'b self, message: M) -> Self::Request
    where
        M: ToMessageId + ToSourceChat,
    {
        message.location_reply(self.latitude, self.longitude)
    }
}