|
|
@ -13,6 +13,19 @@ use std::io::Read;
|
|
|
|
use std::io::Write;
|
|
|
|
use std::io::Write;
|
|
|
|
use std::process::Command;
|
|
|
|
use std::process::Command;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pub trait User {
|
|
|
|
|
|
|
|
fn username(&self) -> String;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl User for serenity::User {
|
|
|
|
|
|
|
|
fn username(&self) -> String {
|
|
|
|
|
|
|
|
if self.discriminator == 0 {
|
|
|
|
|
|
|
|
return self.name.clone();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
format!("{}#{}", self.name, self.discriminator)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub fn get_challenge_number() -> i32 {
|
|
|
|
pub fn get_challenge_number() -> i32 {
|
|
|
|
let challenge_dir = format!("{}/content/challenges", env::var("HUGO").unwrap());
|
|
|
|
let challenge_dir = format!("{}/content/challenges", env::var("HUGO").unwrap());
|
|
|
|
let paths = fs::read_dir(challenge_dir).unwrap();
|
|
|
|
let paths = fs::read_dir(challenge_dir).unwrap();
|
|
|
@ -375,7 +388,10 @@ pub async fn leaderboard(ctx: &Context<'_>) -> Result<(), Error> {
|
|
|
|
let mut top_submitters: Vec<(&String, &u32)> = submission_counts.iter().collect();
|
|
|
|
let mut top_submitters: Vec<(&String, &u32)> = submission_counts.iter().collect();
|
|
|
|
top_submitters.sort_by(|a, b| b.1.cmp(a.1));
|
|
|
|
top_submitters.sort_by(|a, b| b.1.cmp(a.1));
|
|
|
|
let mut leaderboard_html = String::from("<table id=\"leaderboard\">");
|
|
|
|
let mut leaderboard_html = String::from("<table id=\"leaderboard\">");
|
|
|
|
for (i, (id, count)) in top_submitters[0..std::cmp::min(top_submitters.len(), LENGTH)].iter().enumerate() {
|
|
|
|
for (i, (id, count)) in top_submitters[0..std::cmp::min(top_submitters.len(), LENGTH)]
|
|
|
|
|
|
|
|
.iter()
|
|
|
|
|
|
|
|
.enumerate()
|
|
|
|
|
|
|
|
{
|
|
|
|
let place = i + 1;
|
|
|
|
let place = i + 1;
|
|
|
|
let user = serenity::UserId(id.parse::<u64>().unwrap())
|
|
|
|
let user = serenity::UserId(id.parse::<u64>().unwrap())
|
|
|
|
.to_user(&ctx.serenity_context().http)
|
|
|
|
.to_user(&ctx.serenity_context().http)
|
|
|
@ -386,8 +402,12 @@ pub async fn leaderboard(ctx: &Context<'_>) -> Result<(), Error> {
|
|
|
|
let discriminator = user.discriminator;
|
|
|
|
let discriminator = user.discriminator;
|
|
|
|
leaderboard_html.push_str(&format!("<tr><td>{place}</td><td><a href=\"{profile}\" target=\"_blank\" class=\"no-underline\"><img src=\"{avatar}\" onerror=\"this.src='/default.png'\"> <span class=\"underline\">{name}"));
|
|
|
|
leaderboard_html.push_str(&format!("<tr><td>{place}</td><td><a href=\"{profile}\" target=\"_blank\" class=\"no-underline\"><img src=\"{avatar}\" onerror=\"this.src='/default.png'\"> <span class=\"underline\">{name}"));
|
|
|
|
// New usernames have #0000 discriminator
|
|
|
|
// New usernames have #0000 discriminator
|
|
|
|
|
|
|
|
// Can't use User trait user.display_name() because of muted span
|
|
|
|
if discriminator != 0 {
|
|
|
|
if discriminator != 0 {
|
|
|
|
leaderboard_html.push_str(&format!("<span class=\"muted\">#{:0>4}</span>", discriminator));
|
|
|
|
leaderboard_html.push_str(&format!(
|
|
|
|
|
|
|
|
"<span class=\"muted\">#{:0>4}</span>",
|
|
|
|
|
|
|
|
discriminator
|
|
|
|
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
leaderboard_html.push_str(&format!("</span></a></td><td>{count}</td></tr>"));
|
|
|
|
leaderboard_html.push_str(&format!("</span></a></td><td>{count}</td></tr>"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|