|
|
|
use serenity::framework::standard::{macros::command, Args, CommandResult};
|
|
|
|
use serenity::model::prelude::*;
|
|
|
|
use serenity::prelude::*;
|
|
|
|
|
|
|
|
use serde_json::Map;
|
|
|
|
use serde_json::Value;
|
|
|
|
use std::fs::File;
|
|
|
|
use std::io::Read;
|
|
|
|
|
|
|
|
use rand::seq::IteratorRandom;
|
|
|
|
|
|
|
|
fn random_from_string(string: &str) -> char {
|
|
|
|
string.chars().choose(&mut rand::thread_rng()).unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_so_diagram(kanji: char) -> String {
|
|
|
|
format!(
|
|
|
|
"https://raw.githubusercontent.com/mistval/kanji_images/master/gifs/{}.gif",
|
|
|
|
format!("{:x}", kanji as i32)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn display_kanji(ctx: &Context, msg: &Message, kanji: char, comment: &str) -> CommandResult {
|
|
|
|
msg.reply(&ctx.http, format!("{}{}", kanji, comment))
|
|
|
|
.await?;
|
|
|
|
let url = get_so_diagram(kanji);
|
|
|
|
let client = reqwest::Client::new();
|
|
|
|
let request = client.head(&url).build().unwrap();
|
|
|
|
let response = client.execute(request).await?.status();
|
|
|
|
let link_validated = response != reqwest::StatusCode::NOT_FOUND;
|
|
|
|
msg.channel_id
|
|
|
|
.say(
|
|
|
|
&ctx.http,
|
|
|
|
if link_validated {
|
|
|
|
&url
|
|
|
|
} else {
|
|
|
|
"The stroke order diagram for this kanji is unavailable."
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_lists_data() -> Map<String, Value> {
|
|
|
|
let mut lists_file = File::open("kanji_lists.json").unwrap();
|
|
|
|
let mut lists_json = String::new();
|
|
|
|
lists_file.read_to_string(&mut lists_json).unwrap();
|
|
|
|
let lists_data: Value = serde_json::from_str(&lists_json).unwrap();
|
|
|
|
lists_data.as_object().unwrap().clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_kanji_info(kanji: char) -> String {
|
|
|
|
let lists_data = get_lists_data();
|
|
|
|
let mut info = String::from("");
|
|
|
|
for category_name in lists_data.keys() {
|
|
|
|
let category = &lists_data[category_name];
|
|
|
|
let default_version = category["default"].as_str().unwrap();
|
|
|
|
match &category["versions"][default_version]["characters"].as_str() {
|
|
|
|
// if no variants (string)
|
|
|
|
Some(list) => {
|
|
|
|
if list.contains(kanji) {
|
|
|
|
info.push_str(&format!("**{}**, ", category_name));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if variants (map)
|
|
|
|
None => {
|
|
|
|
let variants = &category["versions"][default_version]["characters"]
|
|
|
|
.as_object()
|
|
|
|
.unwrap();
|
|
|
|
let mut in_variants = false;
|
|
|
|
for variant in variants.keys() {
|
|
|
|
let list = variants[variant].as_str().unwrap();
|
|
|
|
if list.contains(kanji) {
|
|
|
|
if !in_variants {
|
|
|
|
info.push_str(&format!("**{}** (", category_name));
|
|
|
|
in_variants = true;
|
|
|
|
}
|
|
|
|
info.push_str(&format!("{}, ", variant));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if in_variants {
|
|
|
|
// Remove last two characters (comma and space)
|
|
|
|
info.pop();
|
|
|
|
info.pop();
|
|
|
|
info.push_str("), ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
// Remove last two characters again
|
|
|
|
info.pop();
|
|
|
|
info.pop();
|
|
|
|
info
|
|
|
|
}
|
|
|
|
|
|
|
|
#[command]
|
|
|
|
async fn i(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
|
|
|
|
let input = args.rest();
|
|
|
|
let chars = input.chars();
|
|
|
|
let mut message = String::from("");
|
|
|
|
let mut covered_chars: Vec<char> = Vec::new();
|
|
|
|
let mut skipped_chars = 0;
|
|
|
|
let mut found_chars: Vec<char> = Vec::new();
|
|
|
|
for character in chars {
|
|
|
|
if covered_chars.contains(&character) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
let kanji_info = get_kanji_info(character);
|
|
|
|
covered_chars.push(character);
|
|
|
|
if kanji_info.len() == 0 {
|
|
|
|
skipped_chars += 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
found_chars.push(character);
|
|
|
|
message.push_str(&format!(
|
|
|
|
"[{}](https://jisho.org/search/{}%23kanji): {}\n",
|
|
|
|
character, character, kanji_info
|
|
|
|
));
|
|
|
|
}
|
|
|
|
message = format!(
|
|
|
|
"Found {} kanji{}\n{}",
|
|
|
|
found_chars.len(),
|
|
|
|
if found_chars.len() == 0 { "." } else { ":" },
|
|
|
|
message
|
|
|
|
);
|
|
|
|
if skipped_chars > 0 {
|
|
|
|
message.push_str(&format!(
|
|
|
|
"Skipped {} character{}.",
|
|
|
|
skipped_chars,
|
|
|
|
if skipped_chars == 1 { "" } else { "s" }
|
|
|
|
));
|
|
|
|
}
|
|
|
|
msg.channel_id
|
|
|
|
.send_message(&ctx.http, |m| {
|
|
|
|
m.reference_message(msg);
|
|
|
|
m.allowed_mentions(|am| {
|
|
|
|
am.empty_parse();
|
|
|
|
am
|
|
|
|
});
|
|
|
|
m.embed(|e| {
|
|
|
|
e.title(input);
|
|
|
|
e.description(message);
|
|
|
|
let mut author = serenity::builder::CreateEmbedAuthor::default();
|
|
|
|
author
|
|
|
|
.icon_url("https://raw.githubusercontent.com/ElnuDev/ji-chan/main/ji-chan.png")
|
|
|
|
.name("字ちゃん")
|
|
|
|
.url("https://github.com/ElnuDev/ji-chan");
|
|
|
|
e.set_author(author);
|
|
|
|
e.color(serenity::utils::Colour::from_rgb(251, 73, 52));
|
|
|
|
if found_chars.len() == 1 {
|
|
|
|
e.thumbnail(get_so_diagram(found_chars[0]));
|
|
|
|
}
|
|
|
|
e
|
|
|
|
});
|
|
|
|
m
|
|
|
|
})
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn random_kanji(
|
|
|
|
category: &str,
|
|
|
|
ctx: &Context,
|
|
|
|
msg: &Message,
|
|
|
|
mut args: Args,
|
|
|
|
) -> CommandResult {
|
|
|
|
let lists_data = get_lists_data();
|
|
|
|
let category = &lists_data[category];
|
|
|
|
let default_version = category["default"].as_str().unwrap();
|
|
|
|
let list = &category["versions"][default_version]["characters"];
|
|
|
|
match list.as_str() {
|
|
|
|
Some(string) => {
|
|
|
|
let kanji = random_from_string(string);
|
|
|
|
display_kanji(&ctx, &msg, kanji, "").await?;
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
let subcategory = args.single::<String>();
|
|
|
|
let subcategories = list.as_object().unwrap();
|
|
|
|
let subcategory_list = {
|
|
|
|
let mut string = String::from("\n");
|
|
|
|
let mut total_count = 0;
|
|
|
|
for subcategory in subcategories.keys() {
|
|
|
|
let subcategory = subcategory.as_str();
|
|
|
|
// One has to do .chars().count() as opposed to .len() here,
|
|
|
|
// because .len() returns the byte length NOT the number of characters.
|
|
|
|
// For kanji, they are always multi-byte Unicode characters.
|
|
|
|
let count = &subcategories[subcategory].as_str().unwrap().chars().count();
|
|
|
|
total_count += count;
|
|
|
|
string.push_str(&format!("**{}** ({} kanji),\n", subcategory, count));
|
|
|
|
}
|
|
|
|
string.push_str(&format!(
|
|
|
|
"or **all** ({} kanji total) for all subcategories",
|
|
|
|
total_count
|
|
|
|
));
|
|
|
|
string
|
|
|
|
};
|
|
|
|
match subcategory {
|
|
|
|
Ok(string) => {
|
|
|
|
let string = string.to_uppercase();
|
|
|
|
if string == "ALL" {
|
|
|
|
let subcategory_key = subcategories
|
|
|
|
.keys()
|
|
|
|
.choose(&mut rand::thread_rng())
|
|
|
|
.unwrap();
|
|
|
|
let list = subcategories[subcategory_key].as_str().unwrap();
|
|
|
|
let kanji = random_from_string(&list);
|
|
|
|
display_kanji(&ctx, &msg, kanji, &format!(", **{}**", subcategory_key))
|
|
|
|
.await?;
|
|
|
|
} else if subcategories.contains_key(&string) {
|
|
|
|
let list = list[&string].as_str().unwrap();
|
|
|
|
let kanji = random_from_string(&list);
|
|
|
|
display_kanji(&ctx, &msg, kanji, "").await?;
|
|
|
|
} else {
|
|
|
|
let message = format!(
|
|
|
|
"That is an invalid subcategory. Please use {}.",
|
|
|
|
&subcategory_list
|
|
|
|
);
|
|
|
|
msg.reply(&ctx.http, message).await?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
let mut message = String::from("Please specify a subcategory: ");
|
|
|
|
message.push_str(&subcategory_list);
|
|
|
|
msg.reply(&ctx.http, message).await?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[command]
|
|
|
|
async fn joyo(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
|
|
|
|
random_kanji("JOYO", ctx, msg, args).await
|
|
|
|
}
|
|
|
|
|
|
|
|
#[command]
|
|
|
|
async fn jinmeiyo(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
|
|
|
|
random_kanji("JINMEIYO", ctx, msg, args).await
|
|
|
|
}
|
|
|
|
|
|
|
|
#[command]
|
|
|
|
async fn kyoiku(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
|
|
|
|
random_kanji("KYOIKU", ctx, msg, args).await
|
|
|
|
}
|
|
|
|
|
|
|
|
#[command]
|
|
|
|
async fn jlpt(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
|
|
|
|
random_kanji("JLPT", ctx, msg, args).await
|
|
|
|
}
|
|
|
|
|
|
|
|
#[command]
|
|
|
|
async fn hyogai(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
|
|
|
|
random_kanji("HYOGAI", ctx, msg, args).await
|
|
|
|
}
|
|
|
|
|
|
|
|
#[command]
|
|
|
|
async fn so(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
|
|
|
|
const MAX_CHARS: i32 = 4;
|
|
|
|
let text = args.rest();
|
|
|
|
if text.is_empty() {
|
|
|
|
msg.reply(
|
|
|
|
&ctx.http,
|
|
|
|
"Please provide some text you want the stroke order for.",
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
let mut displayed_characters: Vec<char> = Vec::new();
|
|
|
|
let mut displayed_character_count = 0;
|
|
|
|
for character in text.chars() {
|
|
|
|
if displayed_character_count >= MAX_CHARS {
|
|
|
|
msg.channel_id
|
|
|
|
.say(
|
|
|
|
&ctx.http,
|
|
|
|
":warning: Maximum number of stroke order diagrams per command reached.",
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if character.is_whitespace() || displayed_characters.contains(&character) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Don't show same character twice
|
|
|
|
displayed_characters.push(character);
|
|
|
|
displayed_character_count += 1;
|
|
|
|
display_kanji(&ctx, &msg, character, "").await?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|