From 183fd751d319f6106e5e50057155d6f156a90e02 Mon Sep 17 00:00:00 2001 From: Justin Crawford Date: Sat, 21 Apr 2012 03:06:54 -0700 Subject: Added C/C++ style escape codes for color codes in the MOTD along with @SaberUK's \x, \u, \b, \c aliases --- src/commands/cmd_motd.cpp | 54 ++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 53 insertions(+), 1 deletion(-) (limited to 'src/commands') diff --git a/src/commands/cmd_motd.cpp b/src/commands/cmd_motd.cpp index 91ba5459f..0b73beb5b 100644 --- a/src/commands/cmd_motd.cpp +++ b/src/commands/cmd_motd.cpp @@ -39,6 +39,58 @@ class CommandMotd : public Command } }; +inline std::string replace_all(const std::string &str, const std::string &orig, const std::string &repl) +{ + std::string new_str = str; + std::string::size_type pos = new_str.find(orig), orig_length = orig.length(), repl_length = repl.length(); + while (pos != std::string::npos) + { + new_str = new_str.substr(0, pos) + repl + new_str.substr(pos + orig_length); + pos = new_str.find(orig, pos + repl_length); + } + return new_str; +} + +/* + * Replace all color codes from the special[] array to actual + * color code chars using C++ style escape sequences. You + * can append other chars to replace if you like (such as %U + * being underline). -- Justasic + */ +std::string ProcessColors(const std::string &string) +{ + static struct special_chars + { + std::string character; + std::string replace; + special_chars(const std::string &c, const std::string &r) : character(c), replace(r) { } + } + + special[] = { + special_chars("\\002", "\002"), // Bold + special_chars("\\037", "\037"), // underline + special_chars("\\003", "\003"), // Color + special_chars("\\0017", "\017"), // Stop colors + special_chars("\\u", "\037"), // Alias for underline + special_chars("\\b", "\002"), // Alias for Bold + special_chars("\\x", "\017"), // Alias for stop + special_chars("\\c", "\003"), // Alias for color + special_chars("", "") + }; + + std::string ret = string; + for(int i = 0; special[i].character.empty() == false; ++i) + { + std::string::size_type pos = ret.find(special[i].character); + if(pos != std::string::npos && ret[pos-1] == '\\' && ret[pos] == '\\') + continue; // Skip double slashes. + + ret = replace_all(ret, special[i].character, special[i].replace); + } + // Replace double slashes with a single slash before we return + return replace_all(ret, "\\\\", "\\"); +} + /** Handle /MOTD */ CmdResult CommandMotd::Handle (const std::vector& parameters, User *user) @@ -62,7 +114,7 @@ CmdResult CommandMotd::Handle (const std::vector& parameters, User RPL_MOTDSTART, user->nick.c_str(), ServerInstance->Config->ServerName.c_str()); for (file_cache::iterator i = motd->second.begin(); i != motd->second.end(); i++) - user->SendText(":%s %03d %s :- %s", ServerInstance->Config->ServerName.c_str(), RPL_MOTD, user->nick.c_str(),i->c_str()); + user->SendText(":%s %03d %s :- %s", ServerInstance->Config->ServerName.c_str(), RPL_MOTD, user->nick.c_str(), ProcessColors(*i).c_str()); user->SendText(":%s %03d %s :End of message of the day.", ServerInstance->Config->ServerName.c_str(), RPL_ENDOFMOTD, user->nick.c_str()); -- cgit v1.2.3 From a2d85a098e07b4c6de1c7ea9f272eaac10fec7c1 Mon Sep 17 00:00:00 2001 From: Justin Crawford Date: Sun, 22 Apr 2012 04:27:46 -0700 Subject: Fixed using a function on every call for /motd, causing lag on large networks (requested by w00t) --- include/inspircd.h | 4 ++++ src/commands/cmd_motd.cpp | 31 ++++++++++++++++++++----------- src/commands/cmd_rehash.cpp | 1 + 3 files changed, 25 insertions(+), 11 deletions(-) (limited to 'src/commands') diff --git a/include/inspircd.h b/include/inspircd.h index abe568011..3a100f826 100644 --- a/include/inspircd.h +++ b/include/inspircd.h @@ -444,6 +444,10 @@ class CoreExport InspIRCd */ LocalStringExt OperQuit; + /** Holds whether the MOTD has been parsed for color codes + */ + bool ProcessedMotdEscapes; + /** Get the current time * Because this only calls time() once every time around the mainloop, * it is much faster than calling time() directly. diff --git a/src/commands/cmd_motd.cpp b/src/commands/cmd_motd.cpp index 0b73beb5b..c2494845f 100644 --- a/src/commands/cmd_motd.cpp +++ b/src/commands/cmd_motd.cpp @@ -23,7 +23,7 @@ class CommandMotd : public Command public: /** Constructor for motd. */ - CommandMotd ( Module* parent) : Command(parent,"MOTD",0,1) { syntax = "[]"; } + CommandMotd ( Module* parent) : Command(parent,"MOTD",0,1) { ServerInstance->ProcessedMotdEscapes = false; syntax = "[]"; } /** Handle command. * @param parameters The parameters to the comamnd * @param pcnt The number of parameters passed to teh command @@ -57,7 +57,7 @@ inline std::string replace_all(const std::string &str, const std::string &orig, * can append other chars to replace if you like (such as %U * being underline). -- Justasic */ -std::string ProcessColors(const std::string &string) +void ProcessColors(ConfigFileCache::iterator &file) { static struct special_chars { @@ -78,17 +78,20 @@ std::string ProcessColors(const std::string &string) special_chars("", "") }; - std::string ret = string; - for(int i = 0; special[i].character.empty() == false; ++i) + for(file_cache::iterator it = file->second.begin(); it != file->second.end(); it++) { - std::string::size_type pos = ret.find(special[i].character); - if(pos != std::string::npos && ret[pos-1] == '\\' && ret[pos] == '\\') - continue; // Skip double slashes. + std::string ret = *it; + for(int i = 0; special[i].character.empty() == false; ++i) + { + std::string::size_type pos = ret.find(special[i].character); + if(pos != std::string::npos && ret[pos-1] == '\\' && ret[pos] == '\\') + continue; // Skip double slashes. - ret = replace_all(ret, special[i].character, special[i].replace); + ret = replace_all(ret, special[i].character, special[i].replace); + } + // Replace double slashes with a single slash before we return + *it = replace_all(ret, "\\\\", "\\"); } - // Replace double slashes with a single slash before we return - return replace_all(ret, "\\\\", "\\"); } /** Handle /MOTD @@ -110,11 +113,17 @@ CmdResult CommandMotd::Handle (const std::vector& parameters, User return CMD_SUCCESS; } + if(!ServerInstance->ProcessedMotdEscapes) + { + ProcessColors(motd); + ServerInstance->ProcessedMotdEscapes = true; + } + user->SendText(":%s %03d %s :%s message of the day", ServerInstance->Config->ServerName.c_str(), RPL_MOTDSTART, user->nick.c_str(), ServerInstance->Config->ServerName.c_str()); for (file_cache::iterator i = motd->second.begin(); i != motd->second.end(); i++) - user->SendText(":%s %03d %s :- %s", ServerInstance->Config->ServerName.c_str(), RPL_MOTD, user->nick.c_str(), ProcessColors(*i).c_str()); + user->SendText(":%s %03d %s :- %s", ServerInstance->Config->ServerName.c_str(), RPL_MOTD, user->nick.c_str(), i->c_str()); user->SendText(":%s %03d %s :End of message of the day.", ServerInstance->Config->ServerName.c_str(), RPL_ENDOFMOTD, user->nick.c_str()); diff --git a/src/commands/cmd_rehash.cpp b/src/commands/cmd_rehash.cpp index 9462b4dd6..af829d726 100644 --- a/src/commands/cmd_rehash.cpp +++ b/src/commands/cmd_rehash.cpp @@ -81,6 +81,7 @@ CmdResult CommandRehash::Handle (const std::vector& parameters, Use * after the config thread has completed. */ + ServerInstance->ProcessedMotdEscapes = false; // Reprocess our motd file --Justasic ServerInstance->RehashUsersAndChans(); FOREACH_MOD(I_OnGarbageCollect, OnGarbageCollect()); -- cgit v1.2.3