mirror of
https://github.com/natsukagami/youmubot.git
synced 2025-05-24 01:00:49 +00:00
Provide a way to give pagination functions a total page count
This commit is contained in:
parent
0b111d5b26
commit
64ff4b3ed8
7 changed files with 102 additions and 55 deletions
|
@ -18,7 +18,7 @@ pub use debugging_ok::OkPrint;
|
|||
pub use flags::Flags;
|
||||
pub use hook::Hook;
|
||||
pub use member_cache::MemberCache;
|
||||
pub use pagination::{paginate, paginate_fn, paginate_reply, paginate_reply_fn, Paginate};
|
||||
pub use pagination::{paginate, paginate_from_fn, paginate_reply, Paginate};
|
||||
|
||||
pub mod announcer;
|
||||
pub mod args;
|
||||
|
|
|
@ -52,6 +52,70 @@ pub trait Paginate: Send + Sized {
|
|||
fn is_empty(&self) -> Option<bool> {
|
||||
self.len().map(|v| v == 0)
|
||||
}
|
||||
|
||||
/// Add a page count to the pagination.
|
||||
fn with_page_count(self, page_count: usize) -> impl Paginate {
|
||||
WithPageCount {
|
||||
inner: self,
|
||||
page_count,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub fn paginate_from_fn(
|
||||
pager: impl for<'m> FnMut(
|
||||
u8,
|
||||
&'m Context,
|
||||
&'m mut Message,
|
||||
) -> std::pin::Pin<Box<dyn Future<Output = Result<bool>> + Send + 'm>>
|
||||
+ Send,
|
||||
) -> impl Paginate {
|
||||
pager
|
||||
}
|
||||
|
||||
struct WithPageCount<Inner> {
|
||||
inner: Inner,
|
||||
page_count: usize,
|
||||
}
|
||||
|
||||
#[async_trait::async_trait]
|
||||
impl<Inner: Paginate> Paginate for WithPageCount<Inner> {
|
||||
async fn render(&mut self, page: u8, ctx: &Context, m: &mut Message) -> Result<bool> {
|
||||
if page as usize >= self.page_count {
|
||||
return Ok(false);
|
||||
}
|
||||
self.inner.render(page, ctx, m).await
|
||||
}
|
||||
async fn prerender(&mut self, ctx: &Context, m: &mut Message) -> Result<()> {
|
||||
self.inner.prerender(ctx, m).await
|
||||
}
|
||||
|
||||
async fn handle_reaction(
|
||||
&mut self,
|
||||
page: u8,
|
||||
ctx: &Context,
|
||||
message: &mut Message,
|
||||
reaction: &Reaction,
|
||||
) -> Result<Option<u8>> {
|
||||
// handle normal reactions first, then fallback to the inner one
|
||||
let new_page = handle_pagination_reaction(page, self, ctx, message, reaction).await?;
|
||||
|
||||
if new_page != page {
|
||||
Ok(Some(new_page))
|
||||
} else {
|
||||
self.inner
|
||||
.handle_reaction(page, ctx, message, reaction)
|
||||
.await
|
||||
}
|
||||
}
|
||||
|
||||
fn len(&self) -> Option<usize> {
|
||||
Some(self.page_count)
|
||||
}
|
||||
|
||||
fn is_empty(&self) -> Option<bool> {
|
||||
Some(self.page_count == 0)
|
||||
}
|
||||
}
|
||||
|
||||
#[async_trait::async_trait]
|
||||
|
@ -185,36 +249,6 @@ async fn paginate_with_first_message(
|
|||
res
|
||||
}
|
||||
|
||||
/// Same as `paginate`, but for function inputs, especially anonymous functions.
|
||||
pub async fn paginate_fn(
|
||||
pager: impl for<'m> FnMut(
|
||||
u8,
|
||||
&'m Context,
|
||||
&'m mut Message,
|
||||
) -> std::pin::Pin<Box<dyn Future<Output = Result<bool>> + Send + 'm>>
|
||||
+ Send,
|
||||
ctx: &Context,
|
||||
channel: ChannelId,
|
||||
timeout: std::time::Duration,
|
||||
) -> Result<()> {
|
||||
paginate(pager, ctx, channel, timeout).await
|
||||
}
|
||||
|
||||
/// Same as `paginate_reply`, but for function inputs, especially anonymous functions.
|
||||
pub async fn paginate_reply_fn(
|
||||
pager: impl for<'m> FnMut(
|
||||
u8,
|
||||
&'m Context,
|
||||
&'m mut Message,
|
||||
) -> std::pin::Pin<Box<dyn Future<Output = Result<bool>> + Send + 'm>>
|
||||
+ Send,
|
||||
ctx: &Context,
|
||||
reply_to: &Message,
|
||||
timeout: std::time::Duration,
|
||||
) -> Result<()> {
|
||||
paginate_reply(pager, ctx, reply_to, timeout).await
|
||||
}
|
||||
|
||||
// Handle the reaction and return a new page number.
|
||||
pub async fn handle_pagination_reaction(
|
||||
page: u8,
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue