use std::str::FromStr; use crate::{ render::ExperimentalShader, session::settings_change::change_render_mode, GlobalState, }; use client::Client; use common::{cmd::*, comp::Admin, parse_cmd_args, uuid::Uuid}; use levenshtein::levenshtein; use strum::IntoEnumIterator; // Please keep this sorted alphabetically, same as with server commands :-) #[derive(Clone, Copy, strum::EnumIter)] pub enum ClientChatCommand { ExperimentalShader, Help, Mute, Unmute, } impl ClientChatCommand { pub fn data(&self) -> ChatCommandData { use ArgumentSpec::*; use Requirement::*; let cmd = ChatCommandData::new; match self { ClientChatCommand::ExperimentalShader => cmd( vec![Enum( "Shader", ExperimentalShader::iter() .map(|item| item.to_string()) .collect(), Optional, )], "Toggles an experimental shader.", None, ), ClientChatCommand::Help => cmd( vec![Command(Optional)], "Display information about commands", None, ), ClientChatCommand::Mute => cmd( vec![PlayerName(Required)], "Mutes chat messages from a player.", None, ), ClientChatCommand::Unmute => cmd( vec![PlayerName(Required)], "Unmutes a player muted with the 'mute' command.", None, ), } } pub fn keyword(&self) -> &'static str { match self { ClientChatCommand::ExperimentalShader => "experimental_shader", ClientChatCommand::Help => "help", ClientChatCommand::Mute => "mute", ClientChatCommand::Unmute => "unmute", } } /// A message that explains what the command does pub fn help_string(&self) -> String { let data = self.data(); let usage = std::iter::once(format!("/{}", self.keyword())) .chain(data.args.iter().map(|arg| arg.usage_string())) .collect::>() .join(" "); format!("{}: {}", usage, data.description) } /// Returns a format string for parsing arguments with scan_fmt pub fn arg_fmt(&self) -> String { self.data() .args .iter() .map(|arg| match arg { ArgumentSpec::PlayerName(_) => "{}", ArgumentSpec::SiteName(_) => "{/.*/}", ArgumentSpec::Float(_, _, _) => "{}", ArgumentSpec::Integer(_, _, _) => "{d}", ArgumentSpec::Any(_, _) => "{}", ArgumentSpec::Command(_) => "{}", ArgumentSpec::Message(_) => "{/.*/}", ArgumentSpec::SubCommand => "{} {/.*/}", ArgumentSpec::Enum(_, _, _) => "{}", ArgumentSpec::Boolean(_, _, _) => "{}", }) .collect::>() .join(" ") } /// Produce an iterator over all the available commands pub fn iter() -> impl Iterator + Clone { ::iter() } /// Produce an iterator that first goes over all the short keywords /// and their associated commands and then iterates over all the normal /// keywords with their associated commands pub fn iter_with_keywords() -> impl Iterator { Self::iter().map(|c| (c.keyword(), c)) } } impl FromStr for ClientChatCommand { type Err = (); fn from_str(keyword: &str) -> Result { Self::iter() .map(|c| (c.keyword(), c)) .find_map(|(kwd, command)| (kwd == keyword).then_some(command)) .ok_or(()) } } #[derive(Clone, Copy)] pub enum ChatCommandKind { Client(ClientChatCommand), Server(ServerChatCommand), } impl FromStr for ChatCommandKind { type Err = (); fn from_str(s: &str) -> Result { if let Ok(cmd) = s.parse::() { Ok(ChatCommandKind::Client(cmd)) } else if let Ok(cmd) = s.parse::() { Ok(ChatCommandKind::Server(cmd)) } else { Err(()) } } } /// Represents the feedback shown to the user of a command, if any. Server /// commands give their feedback as an event, so in those cases this will always /// be Ok(None). An Err variant will be be displayed with the error icon and /// text color type CommandResult = Result, String>; /// Runs a command by either sending it to the server or processing it /// locally. Returns a String to be output to the chat. // Note: it's not clear what data future commands will need access to, so the // signature of this function might change pub fn run_command( client: &mut Client, global_state: &mut GlobalState, cmd: &str, args: Vec, ) -> CommandResult { let command = ChatCommandKind::from_str(cmd); match command { Ok(ChatCommandKind::Server(cmd)) => { client.send_command(cmd.keyword().into(), args); Ok(None) // The server will provide a response when the command is run }, Ok(ChatCommandKind::Client(cmd)) => { Ok(Some(run_client_command(client, global_state, cmd, args)?)) }, Err(()) => Err(invalid_command_message(client, cmd.to_string())), } } fn invalid_command_message(client: &Client, user_entered_invalid_command: String) -> String { let entity_role = client .state() .read_storage::() .get(client.entity()) .map(|admin| admin.0); let usable_commands = ServerChatCommand::iter() .filter(|cmd| cmd.needs_role() <= entity_role) .map(|cmd| cmd.keyword()) .chain(ClientChatCommand::iter().map(|cmd| cmd.keyword())); let most_similar_str = usable_commands .clone() .min_by_key(|cmd| levenshtein(&user_entered_invalid_command, cmd)) .expect("At least one command exists."); let commands_with_same_prefix = usable_commands .filter(|cmd| cmd.starts_with(&user_entered_invalid_command) && cmd != &most_similar_str); format!( "Could not find a command named {}. Did you mean any of the following? \n/{} {} \n\nType \ /help to see a list of all commands.", user_entered_invalid_command, most_similar_str, commands_with_same_prefix.fold(String::new(), |s, arg| s + "\n/" + arg) ) } fn run_client_command( client: &mut Client, global_state: &mut GlobalState, command: ClientChatCommand, args: Vec, ) -> Result { let command = match command { ClientChatCommand::ExperimentalShader => handle_experimental_shader, ClientChatCommand::Help => handle_help, ClientChatCommand::Mute => handle_mute, ClientChatCommand::Unmute => handle_unmute, }; command(client, global_state, args) } fn handle_help( client: &Client, _global_state: &mut GlobalState, args: Vec, ) -> Result { if let Some(cmd) = parse_cmd_args!(args, ServerChatCommand) { Ok(cmd.help_string()) } else { let mut message = String::new(); let entity_role = client .state() .read_storage::() .get(client.entity()) .map(|admin| admin.0); ClientChatCommand::iter().for_each(|cmd| { message += &cmd.help_string(); message += "\n"; }); // Iterate through all ServerChatCommands you have permission to use. ServerChatCommand::iter() .filter(|cmd| cmd.needs_role() <= entity_role) .for_each(|cmd| { message += &cmd.help_string(); message += "\n"; }); message += "Additionally, you can use the following shortcuts:"; ServerChatCommand::iter() .filter(|cmd| cmd.needs_role() <= entity_role) .filter_map(|cmd| cmd.short_keyword().map(|k| (k, cmd))) .for_each(|(k, cmd)| { message += &format!(" /{} => /{}", k, cmd.keyword()); }); Ok(message) } } fn handle_mute( client: &Client, global_state: &mut GlobalState, args: Vec, ) -> Result { if let Some(alias) = parse_cmd_args!(args, String) { let target = client .player_list() .values() .find(|p| p.player_alias == alias) .ok_or_else(|| format!("Could not find a player named {}", alias))?; if let Some(me) = client.uid().and_then(|uid| client.player_list().get(&uid)) { if target.uuid == me.uuid { return Err("You cannot mute yourself.".to_string()); } } if global_state .profile .mutelist .insert(target.uuid, alias.clone()) .is_none() { Ok(format!("Successfully muted player {}.", alias)) } else { Err(format!("{} is already muted.", alias)) } } else { Err("You must specify a player to mute.".to_string()) } } fn handle_unmute( client: &Client, global_state: &mut GlobalState, args: Vec, ) -> Result { // Note that we don't care if this is a real player, so that it's possible // to unmute someone when they're offline if let Some(alias) = parse_cmd_args!(args, String) { if let Some(uuid) = global_state .profile .mutelist .iter() .find(|(_, v)| **v == alias) .map(|(k, _)| *k) { if let Some(me) = client.uid().and_then(|uid| client.player_list().get(&uid)) { if uuid == me.uuid { return Err("You cannot unmute yourself.".to_string()); } } global_state.profile.mutelist.remove(&uuid); Ok(format!("Successfully unmuted player {}.", alias)) } else { Err(format!("Could not find a muted player named {}.", alias)) } } else { Err("You must specify a player to unmute.".to_string()) } } fn handle_experimental_shader( _client: &Client, global_state: &mut GlobalState, args: Vec, ) -> Result { if args.is_empty() { ExperimentalShader::iter() .map(|s| { let is_active = global_state .settings .graphics .render_mode .experimental_shaders .contains(&s); format!("[{}] {}", if is_active { "x" } else { " " }, s) }) .reduce(|mut a, b| { a.push('\n'); a.push_str(&b); a }) .ok_or("There are no experimental shaders.".to_string()) } else if let Some(item) = parse_cmd_args!(args, String) { if let Ok(shader) = ExperimentalShader::from_str(&item) { let mut new_render_mode = global_state.settings.graphics.render_mode.clone(); let res = if new_render_mode.experimental_shaders.remove(&shader) { Ok(format!("Disabled {item}.")) } else { new_render_mode.experimental_shaders.insert(shader); Ok(format!("Enabled {item}.")) }; change_render_mode( new_render_mode, &mut global_state.window, &mut global_state.settings, ); res } else { Err(format!( "{item} is not an expermimental shader, use this command with any arguments to \ see a complete list." )) } } else { Err( "You must specify a valid experimental shader, to get a list of experimental shaders, \ use this command without any arguments." .to_string(), ) } } /// A helper function to get the Uuid of a player with a given alias pub fn get_player_uuid(client: &Client, alias: &String) -> Option { client .player_list() .values() .find(|p| p.player_alias == *alias) .map(|p| p.uuid) } trait TabComplete { fn complete(&self, part: &str, client: &Client) -> Vec; } impl TabComplete for ArgumentSpec { fn complete(&self, part: &str, client: &Client) -> Vec { match self { ArgumentSpec::PlayerName(_) => complete_player(part, client), ArgumentSpec::SiteName(_) => complete_site(part, client), ArgumentSpec::Float(_, x, _) => { if part.is_empty() { vec![format!("{:.1}", x)] } else { vec![] } }, ArgumentSpec::Integer(_, x, _) => { if part.is_empty() { vec![format!("{}", x)] } else { vec![] } }, ArgumentSpec::Any(_, _) => vec![], ArgumentSpec::Command(_) => complete_command(part, ' '), ArgumentSpec::Message(_) => complete_player(part, client), ArgumentSpec::SubCommand => complete_command(part, ' '), ArgumentSpec::Enum(_, strings, _) => strings .iter() .filter(|string| string.starts_with(part)) .map(|c| c.to_string()) .collect(), ArgumentSpec::Boolean(_, part, _) => ["true", "false"] .iter() .filter(|string| string.starts_with(part)) .map(|c| c.to_string()) .collect(), } } } fn complete_player(part: &str, client: &Client) -> Vec { client .player_list() .values() .map(|player_info| &player_info.player_alias) .filter(|alias| alias.starts_with(part)) .cloned() .collect() } fn complete_site(mut part: &str, client: &Client) -> Vec { if let Some(p) = part.strip_prefix('"') { part = p; } client .sites() .values() .filter_map(|site| match site.site.kind { common_net::msg::world_msg::SiteKind::Cave => None, _ => site.site.name.as_ref(), }) .filter(|name| name.starts_with(part)) .map(|name| { if name.contains(' ') { format!("\"{}\"", name) } else { name.clone() } }) .collect() } // Get the byte index of the nth word. Used in completing "/sudo p /subcmd" fn nth_word(line: &str, n: usize) -> Option { let mut is_space = false; let mut j = 0; for (i, c) in line.char_indices() { match (is_space, c.is_whitespace()) { (true, true) => {}, (true, false) => { is_space = false; j += 1; }, (false, true) => { is_space = true; }, (false, false) => {}, } if j == n { return Some(i); } } None } fn complete_command(part: &str, prefix: char) -> Vec { ServerChatCommand::iter_with_keywords() .map(|(kwd, _)| kwd) .chain(ClientChatCommand::iter_with_keywords().map(|(kwd, _)| kwd)) .filter(|kwd| kwd.starts_with(part)) .map(|kwd| format!("{}{}", prefix, kwd)) .collect() } pub fn complete(line: &str, client: &Client, cmd_prefix: char) -> Vec { let word = if line.chars().last().map_or(true, char::is_whitespace) { "" } else { line.split_whitespace().last().unwrap_or("") }; if line.starts_with(cmd_prefix) { let line = line.strip_prefix(cmd_prefix).unwrap_or(line); let mut iter = line.split_whitespace(); let cmd = iter.next().unwrap_or(""); let i = iter.count() + usize::from(word.is_empty()); if i == 0 { // Completing chat command name. This is the start of the line so the prefix // will be part of it let word = word.strip_prefix(cmd_prefix).unwrap_or(word); return complete_command(word, cmd_prefix); } let args = { if let Ok(cmd) = cmd.parse::() { Some(cmd.data().args) } else if let Ok(cmd) = cmd.parse::() { Some(cmd.data().args) } else { None } }; if let Some(args) = args { if let Some(arg) = args.get(i - 1) { // Complete ith argument arg.complete(word, client) } else { // Complete past the last argument match args.last() { Some(ArgumentSpec::SubCommand) => { if let Some(index) = nth_word(line, args.len()) { complete(&line[index..], client, cmd_prefix) } else { vec![] } }, Some(ArgumentSpec::Message(_)) => complete_player(word, client), _ => vec![], // End of command. Nothing to complete } } } else { // Completing for unknown chat command complete_player(word, client) } } else { // Not completing a command complete_player(word, client) } } #[test] fn verify_cmd_list_sorted() { let mut list = ClientChatCommand::iter() .map(|c| c.keyword()) .collect::>(); // Vec::is_sorted is unstable, so we do it the hard way let list2 = list.clone(); list.sort_unstable(); assert_eq!(list, list2); } #[test] fn test_complete_command() { assert_eq!(complete_command("mu", '/'), vec!["/mute".to_string()]); assert_eq!(complete_command("unba", '/'), vec!["/unban".to_string()]); assert_eq!(complete_command("make_", '/'), vec![ "/make_block".to_string(), "/make_npc".to_string(), "/make_sprite".to_string(), "/make_volume".to_string() ]); }