use super::RequestReactionType;
use crate::{
client::Client,
error::Error as HttpError,
request::{validate, Pending, Request},
routing::Route,
};
use std::{
error::Error,
fmt::{Display, Formatter, Result as FmtResult},
};
use twilight_model::{
id::{ChannelId, MessageId, UserId},
user::User,
};
#[derive(Debug)]
pub struct GetReactionsError {
kind: GetReactionsErrorType,
}
impl GetReactionsError {
#[must_use = "retrieving the type has no effect if left unused"]
pub const fn kind(&self) -> &GetReactionsErrorType {
&self.kind
}
#[allow(clippy::unused_self)]
#[must_use = "consuming the error and retrieving the source has no effect if left unused"]
pub fn into_source(self) -> Option<Box<dyn Error + Send + Sync>> {
None
}
#[must_use = "consuming the error into its parts has no effect if left unused"]
pub fn into_parts(self) -> (GetReactionsErrorType, Option<Box<dyn Error + Send + Sync>>) {
(self.kind, None)
}
}
impl Display for GetReactionsError {
fn fmt(&self, f: &mut Formatter<'_>) -> FmtResult {
match &self.kind {
GetReactionsErrorType::LimitInvalid { .. } => f.write_str("the limit is invalid"),
}
}
}
impl Error for GetReactionsError {}
#[derive(Debug)]
#[non_exhaustive]
pub enum GetReactionsErrorType {
LimitInvalid {
limit: u64,
},
}
#[derive(Default)]
struct GetReactionsFields {
after: Option<UserId>,
limit: Option<u64>,
}
pub struct GetReactions<'a> {
channel_id: ChannelId,
emoji: String,
fields: GetReactionsFields,
fut: Option<Pending<'a, Vec<User>>>,
http: &'a Client,
message_id: MessageId,
}
impl<'a> GetReactions<'a> {
pub(crate) fn new(
http: &'a Client,
channel_id: ChannelId,
message_id: MessageId,
emoji: RequestReactionType,
) -> Self {
Self {
channel_id,
emoji: super::format_emoji(emoji),
fields: GetReactionsFields::default(),
fut: None,
http,
message_id,
}
}
pub fn after(mut self, after: UserId) -> Self {
self.fields.after.replace(after);
self
}
pub fn limit(mut self, limit: u64) -> Result<Self, GetReactionsError> {
if !validate::get_reactions_limit(limit) {
return Err(GetReactionsError {
kind: GetReactionsErrorType::LimitInvalid { limit },
});
}
self.fields.limit.replace(limit);
Ok(self)
}
fn start(&mut self) -> Result<(), HttpError> {
let request = Request::from_route(Route::GetReactionUsers {
after: self.fields.after.map(|x| x.0),
channel_id: self.channel_id.0,
emoji: self.emoji.clone(),
limit: self.fields.limit,
message_id: self.message_id.0,
});
self.fut.replace(Box::pin(self.http.request(request)));
Ok(())
}
}
poll_req!(GetReactions<'_>, Vec<User>);