mirror of
https://github.com/natsukagami/youmubot.git
synced 2025-04-19 16:58:55 +00:00
Sort by score and oppai pp
This commit is contained in:
parent
a728119fbe
commit
8d4c3e01d8
2 changed files with 98 additions and 21 deletions
|
@ -8,7 +8,7 @@ use serenity::{builder::CreateEmbed, utils::MessageBuilder};
|
||||||
use youmubot_prelude::*;
|
use youmubot_prelude::*;
|
||||||
|
|
||||||
/// Writes a number grouped in groups of 3.
|
/// Writes a number grouped in groups of 3.
|
||||||
fn grouped_number(num: u64) -> String {
|
pub(crate) fn grouped_number(num: u64) -> String {
|
||||||
let s = num.to_string();
|
let s = num.to_string();
|
||||||
let mut b = MessageBuilder::new();
|
let mut b = MessageBuilder::new();
|
||||||
let mut i = if s.len() % 3 == 0 { 3 } else { s.len() % 3 };
|
let mut i = if s.len() % 3 == 0 { 3 } else { s.len() % 3 };
|
||||||
|
|
|
@ -4,8 +4,8 @@ use super::{
|
||||||
ModeArg, OsuClient,
|
ModeArg, OsuClient,
|
||||||
};
|
};
|
||||||
use crate::{
|
use crate::{
|
||||||
discord::BeatmapWithMode,
|
discord::{oppai_cache::BeatmapCache, BeatmapWithMode},
|
||||||
models::{Mode, Score},
|
models::{Mode, Mods, Score},
|
||||||
request::UserID,
|
request::UserID,
|
||||||
};
|
};
|
||||||
use serenity::{
|
use serenity::{
|
||||||
|
@ -139,11 +139,30 @@ pub(crate) mod update_lock {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
|
enum OrderBy {
|
||||||
|
PP,
|
||||||
|
Score,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<&str> for OrderBy {
|
||||||
|
fn from(s: &str) -> Self {
|
||||||
|
if s == "--score" {
|
||||||
|
Self::Score
|
||||||
|
} else {
|
||||||
|
Self::PP
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[command("updatelb")]
|
#[command("updatelb")]
|
||||||
#[description = "Update the leaderboard on the last seen beatmap"]
|
#[description = "Update the leaderboard on the last seen beatmap"]
|
||||||
#[max_args(0)]
|
#[usage = "[--score to sort by score, default to sort by pp]"]
|
||||||
|
#[max_args(1)]
|
||||||
#[only_in(guilds)]
|
#[only_in(guilds)]
|
||||||
pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) -> CommandResult {
|
pub async fn update_leaderboard(ctx: &Context, m: &Message, args: Args) -> CommandResult {
|
||||||
|
let sort_order = OrderBy::from(args.rest());
|
||||||
|
|
||||||
let guild = m.guild_id.unwrap();
|
let guild = m.guild_id.unwrap();
|
||||||
let data = ctx.data.read().await;
|
let data = ctx.data.read().await;
|
||||||
let update_lock = data.get::<update_lock::UpdateLock>().unwrap();
|
let update_lock = data.get::<update_lock::UpdateLock>().unwrap();
|
||||||
|
@ -161,6 +180,7 @@ pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) ->
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
let mode = bm.1;
|
||||||
let member_cache = data.get::<MemberCache>().unwrap();
|
let member_cache = data.get::<MemberCache>().unwrap();
|
||||||
// Signal that we are running.
|
// Signal that we are running.
|
||||||
let running_reaction = m.react(&ctx, '⌛').await?;
|
let running_reaction = m.react(&ctx, '⌛').await?;
|
||||||
|
@ -186,7 +206,7 @@ pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) ->
|
||||||
.filter_map(future::ready)
|
.filter_map(future::ready)
|
||||||
.filter_map(|(member, osu_id)| async move {
|
.filter_map(|(member, osu_id)| async move {
|
||||||
let scores = osu
|
let scores = osu
|
||||||
.scores(beatmap_id, |f| f.user(UserID::ID(osu_id)))
|
.scores(beatmap_id, |f| f.user(UserID::ID(osu_id)).mode(mode))
|
||||||
.await
|
.await
|
||||||
.ok();
|
.ok();
|
||||||
scores
|
scores
|
||||||
|
@ -211,22 +231,26 @@ pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) ->
|
||||||
m.reply(
|
m.reply(
|
||||||
&ctx,
|
&ctx,
|
||||||
format!(
|
format!(
|
||||||
"update for beatmap ({}, {}) complete, {} users updated.",
|
"update for beatmap (`{}`) complete, {} users updated.",
|
||||||
bm.0.beatmap_id, bm.1, updated_users
|
bm.0.short_link(if bm.mode() != bm.1 { Some(bm.1) } else { None }, None),
|
||||||
|
updated_users
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.ok();
|
.ok();
|
||||||
drop(update_lock);
|
drop(update_lock);
|
||||||
show_leaderboard(ctx, m, bm).await
|
show_leaderboard(ctx, m, bm, sort_order).await
|
||||||
}
|
}
|
||||||
|
|
||||||
#[command("leaderboard")]
|
#[command("leaderboard")]
|
||||||
#[aliases("lb", "bmranks", "br", "cc")]
|
#[aliases("lb", "bmranks", "br", "cc")]
|
||||||
|
#[usage = "[--score to sort by score, default to sort by pp]"]
|
||||||
#[description = "See the server's ranks on the last seen beatmap"]
|
#[description = "See the server's ranks on the last seen beatmap"]
|
||||||
#[max_args(0)]
|
#[max_args(1)]
|
||||||
#[only_in(guilds)]
|
#[only_in(guilds)]
|
||||||
pub async fn leaderboard(ctx: &Context, m: &Message, mut _args: Args) -> CommandResult {
|
pub async fn leaderboard(ctx: &Context, m: &Message, args: Args) -> CommandResult {
|
||||||
|
let sort_order = OrderBy::from(args.rest());
|
||||||
|
|
||||||
let data = ctx.data.read().await;
|
let data = ctx.data.read().await;
|
||||||
let bm = match get_beatmap(&*data, m.channel_id)? {
|
let bm = match get_beatmap(&*data, m.channel_id)? {
|
||||||
Some(bm) => bm,
|
Some(bm) => bm,
|
||||||
|
@ -235,13 +259,36 @@ pub async fn leaderboard(ctx: &Context, m: &Message, mut _args: Args) -> Command
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
show_leaderboard(ctx, m, bm).await
|
show_leaderboard(ctx, m, bm, sort_order).await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> CommandResult {
|
async fn show_leaderboard(
|
||||||
|
ctx: &Context,
|
||||||
|
m: &Message,
|
||||||
|
bm: BeatmapWithMode,
|
||||||
|
order: OrderBy,
|
||||||
|
) -> CommandResult {
|
||||||
let data = ctx.data.read().await;
|
let data = ctx.data.read().await;
|
||||||
let mut osu_user_bests = OsuUserBests::open(&*data);
|
let mut osu_user_bests = OsuUserBests::open(&*data);
|
||||||
|
|
||||||
|
// Get oppai map.
|
||||||
|
let mode = bm.1;
|
||||||
|
let oppai = data.get::<BeatmapCache>().unwrap();
|
||||||
|
let oppai_map = oppai.get_beatmap(bm.0.beatmap_id).await?;
|
||||||
|
let get_oppai_pp = move |combo: u64, misses: u64, acc: f64, mods: Mods| {
|
||||||
|
mode.to_oppai_mode().and_then(|mode| {
|
||||||
|
oppai_map
|
||||||
|
.get_pp_from(
|
||||||
|
oppai_rs::Combo::non_fc(combo as u32, misses as u32),
|
||||||
|
acc as f32,
|
||||||
|
Some(mode),
|
||||||
|
mods,
|
||||||
|
)
|
||||||
|
.ok()
|
||||||
|
.map(|v| v as f64)
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
// Run a check on the user once too!
|
// Run a check on the user once too!
|
||||||
{
|
{
|
||||||
let osu_users = OsuSavedUsers::open(&*data);
|
let osu_users = OsuSavedUsers::open(&*data);
|
||||||
|
@ -266,8 +313,7 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
|
||||||
let guild = m.guild_id.expect("Guild-only command");
|
let guild = m.guild_id.expect("Guild-only command");
|
||||||
let member_cache = data.get::<MemberCache>().unwrap();
|
let member_cache = data.get::<MemberCache>().unwrap();
|
||||||
let scores = {
|
let scores = {
|
||||||
const NO_SCORES: &'static str =
|
const NO_SCORES: &'static str = "No scores have been recorded for this beatmap.";
|
||||||
"No scores have been recorded for this beatmap. Run `osu check` to scan for yours!";
|
|
||||||
|
|
||||||
let users = osu_user_bests
|
let users = osu_user_bests
|
||||||
.borrow()?
|
.borrow()?
|
||||||
|
@ -300,11 +346,29 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
|
||||||
.map(move |v| future::ready((user.clone(), v.clone())))
|
.map(move |v| future::ready((user.clone(), v.clone())))
|
||||||
.collect::<stream::FuturesUnordered<_>>()
|
.collect::<stream::FuturesUnordered<_>>()
|
||||||
})
|
})
|
||||||
.filter_map(|(user, score)| future::ready(score.pp.map(|v| (v, user, score))))
|
.filter_map(|(user, score)| {
|
||||||
|
future::ready(
|
||||||
|
score
|
||||||
|
.pp
|
||||||
|
.or_else(|| {
|
||||||
|
get_oppai_pp(
|
||||||
|
score.max_combo,
|
||||||
|
score.count_miss,
|
||||||
|
score.accuracy(mode),
|
||||||
|
score.mods,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.map(|v| (v, user, score)),
|
||||||
|
)
|
||||||
|
})
|
||||||
.collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
.await;
|
.await;
|
||||||
scores
|
match order {
|
||||||
.sort_by(|(a, _, _), (b, _, _)| b.partial_cmp(a).unwrap_or(std::cmp::Ordering::Equal));
|
OrderBy::PP => scores.sort_by(|(a, _, _), (b, _, _)| {
|
||||||
|
b.partial_cmp(a).unwrap_or(std::cmp::Ordering::Equal)
|
||||||
|
}),
|
||||||
|
OrderBy::Score => scores.sort_by(|(_, _, a), (_, _, b)| b.score.cmp(&a.score)),
|
||||||
|
};
|
||||||
scores
|
scores
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -350,11 +414,18 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
|
||||||
.map(|(_, _, v)| v.rank.to_string())
|
.map(|(_, _, v)| v.rank.to_string())
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
let rw = ranks.iter().map(|v| v.len()).max().unwrap().max(4);
|
let rw = ranks.iter().map(|v| v.len()).max().unwrap().max(4);
|
||||||
|
let pp_label = match order {
|
||||||
|
OrderBy::PP => "pp",
|
||||||
|
OrderBy::Score => "score",
|
||||||
|
};
|
||||||
let pp = scores
|
let pp = scores
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(pp, _, _)| format!("{:.2}", pp))
|
.map(|(pp, _, s)| match order {
|
||||||
|
OrderBy::PP => format!("{:.2}", pp),
|
||||||
|
OrderBy::Score => format!("{}", crate::discord::embeds::grouped_number(s.score)),
|
||||||
|
})
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
let pw = pp.iter().map(|v| v.len()).max().unwrap_or(2);
|
let pw = pp.iter().map(|v| v.len()).max().unwrap_or(pp_label.len());
|
||||||
/*mods width*/
|
/*mods width*/
|
||||||
let mdw = scores
|
let mdw = scores
|
||||||
.iter()
|
.iter()
|
||||||
|
@ -377,7 +448,7 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
|
||||||
.push_line("```")
|
.push_line("```")
|
||||||
.push_line(format!(
|
.push_line(format!(
|
||||||
"rank | {:>pw$} | {:mdw$} | {:rw$} | {:>aw$} | {:>cw$} | {:mw$} | {:uw$}",
|
"rank | {:>pw$} | {:mdw$} | {:rw$} | {:>aw$} | {:>cw$} | {:mw$} | {:uw$}",
|
||||||
"pp",
|
pp_label,
|
||||||
"mods",
|
"mods",
|
||||||
"rank",
|
"rank",
|
||||||
"acc",
|
"acc",
|
||||||
|
@ -434,6 +505,12 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
|
||||||
page + 1,
|
page + 1,
|
||||||
(total_len + ITEMS_PER_PAGE - 1) / ITEMS_PER_PAGE,
|
(total_len + ITEMS_PER_PAGE - 1) / ITEMS_PER_PAGE,
|
||||||
));
|
));
|
||||||
|
if let crate::models::ApprovalStatus::Ranked(_) = bm.0.approval {
|
||||||
|
} else {
|
||||||
|
if order == OrderBy::PP {
|
||||||
|
content.push_line("PP was calculated by `oppai-rs`, **not** official values.");
|
||||||
|
}
|
||||||
|
}
|
||||||
m.edit(&ctx, |f| f.content(content.build())).await?;
|
m.edit(&ctx, |f| f.content(content.build())).await?;
|
||||||
Ok(true)
|
Ok(true)
|
||||||
})
|
})
|
||||||
|
|
Loading…
Add table
Reference in a new issue