summaryrefslogblamecommitdiff
path: root/src/commandhandler.cpp
blob: 537d9645a679866573ad2360bf91c3d47eb6aab2 (plain) (tree)
1
2
3
4
5
6
7
  
                       

                                                            
                                               
  
                                             
























                                                                         

                           


                             
                             
                      
                        



































                                                                             

                      



















































































































































































                                                                             



                                



                                           











                                                                    



































                                                            

                                                                 
 
                   
     

                                             


















































































































































































































































































































































































                                                                               
                                     





















































































































































































































































                                                                               
                                                                   











                                                                            


                                                      


                                                                           
                                                 
         

                                                 




                                                                        
                                            
         
                                             




                                                                            
                                        
         
                                      




                                                                            
                                 
         

                         
         

                             






                                                                              


                                                                     


                                                            
               
     
 

                                                   




                                           


                                                             
                                                                        

               

                                              







                                                                


                                                     
               
     
 
                                              







                                                                   
                                                                              

               

                                                 




                                           





                                                                      
                                                                   


               
                                                      


               
                                                



                                  



                                                            



                               

                                                                               

      
/*
 *  The ManaPlus Client
 *  Copyright (C) 2008-2009  The Mana World Development Team
 *  Copyright (C) 2009-2010  The Mana Developers
 *  Copyright (C) 2011  The ManaPlus Developers
 *
 *  This file is part of The ManaPlus Client.
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

#include "commandhandler.h"

#include "actorspritemanager.h"
#include "channelmanager.h"
#include "channel.h"
#include "game.h"
#include "localplayer.h"
#include "log.h"
#include "main.h"

#include "gui/chatwindow.h"
#include "gui/helpwindow.h"
#include "gui/gui.h"
#include "gui/outfitwindow.h"
#include "gui/shopwindow.h"
#include "gui/socialwindow.h"
#include "gui/trade.h"
#include "gui/sdlfont.h"

#include "gui/widgets/channeltab.h"
#include "gui/widgets/chattab.h"
#include "gui/widgets/whispertab.h"

#include "net/adminhandler.h"
#include "net/beinghandler.h"
#include "net/chathandler.h"
#include "net/gamehandler.h"
#include "net/guildhandler.h"
#include "net/net.h"
#include "net/partyhandler.h"
#include "net/tradehandler.h"

#include "utils/gettext.h"
#include "utils/stringutils.h"

extern std::string tradePartnerName;

CommandHandler::CommandHandler()
{
}

void CommandHandler::handleCommands(const std::string &command, ChatTab *tab)
{
    // here need add splitting commands
    handleCommand(command, tab);
}

void CommandHandler::handleCommand(const std::string &command, ChatTab *tab)
{
    std::string::size_type pos = command.find(' ');
    std::string type(command, 0, pos);
    std::string args(command, pos == std::string::npos
                     ? command.size() : pos + 1);

    args = trim(args);

    if (command == "closeall")
    {
        handleCloseAll(args, tab);
    }
    else if (type == "ignoreall")
    {
        handleIgnoreAll(args, tab);
    }
    else if (type == "help") // Do help before tabs so they can't override it
    {
        handleHelp(args, tab);
    }
    else if (type == "announce")
    {
        handleAnnounce(args, tab);
    }
    else if (type == "where")
    {
        handleWhere(args, tab);
    }
    else if (type == "who")
    {
        handleWho(args, tab);
    }
    else if (type == "msg" || type == "whisper" || type == "w")
    {
        handleMsg(args, tab);
    }
    else if (type == "query" || type == "q")
    {
        handleQuery(args, tab);
    }
    else if (type == "ignore")
    {
        handleIgnore(args, tab);
    }
    else if (type == "unignore")
    {
        handleUnignore(args, tab);
    }
    else if (type == "friend" || type == "befriend")
    {
        handleFriend(args, tab);
    }
    else if (type == "disregard")
    {
        handleDisregard(args, tab);
    }
    else if (type == "neutral")
    {
        handleNeutral(args, tab);
    }
    else if (type == "erase")
    {
        handleErase(args, tab);
    }
    else if (type == "join")
    {
        handleJoin(args, tab);
    }
    else if (type == "list")
    {
        handleListChannels(args, tab);
    }
    else if (type == "clear")
    {
        handleClear(args, tab);
    }
    else if (type == "createparty")
    {
        handleCreateParty(args, tab);
    }
    else if (type == "createguild")
    {
        handleCreateGuild(args, tab);
    }
    else if (type == "party")
    {
        handleParty(args, tab);
    }
    else if (type == "me")
    {
        handleMe(args, tab);
    }
    else if (type == "toggle")
    {
        handleToggle(args, tab);
    }
    else if (type == "present")
    {
        handlePresent(args, tab);
    }
    else if (type == "quit")
    {
        handleQuit(args, tab);
    }
    else if (type == "all")
    {
        handleShowAll(args, tab);
    }
    else if (type == "move")
    {
        handleMove(args, tab);
    }
    else if (type == "target")
    {
        handleTarget(args, tab);
    }
    else if (type == "outfit")
    {
        handleOutfit(args, tab);
    }
    else if (type == "emote")
    {
        handleEmote(args, tab);
    }
    else if (type == "away")
    {
        handleAway(args, tab);
    }
    else if (type == "follow")
    {
        handleFollow(args, tab);
    }
    else if (type == "heal")
    {
        handleHeal(args, tab);
    }
    else if (type == "navigate")
    {
        handleNavigate(args, tab);
    }
    else if (type == "imitation")
    {
        handleImitation(args, tab);
    }
    else if (type == "mail")
    {
        handleMail(args, tab);
    }
    else if (type == "trade")
    {
        handleTrade(args, tab);
    }
    else if (type == "priceload")
    {
        handlePriceLoad(args, tab);
    }
    else if (type == "pricesave")
    {
        handlePriceSave(args, tab);
    }
    else if (type == "cacheinfo")
    {
        handleCacheInfo(args, tab);
    }
    else if (type == "disconnect")
    {
        handleDisconnect(args, tab);
    }
    else if (type == "undress")
    {
        handleUndress(args, tab);
    }
    else if (type == "attack")
    {
        handleAttack(args, tab);
    }
    else if (type == "dirs")
    {
        handleDirs(args, tab);
    }
    else if (type == "info")
    {
        handleInfo(args, tab);
    }
    else if (type == "wait")
    {
        handleWait(args, tab);
    }
    else if (type == "uptime")
    {
        handleUptime(args, tab);
    }
    else if (type == "addpriorityattack")
    {
        handleAddPriorityAttack(args, tab);
    }
    else if (type == "addattack")
    {
        handleAddAttack(args, tab);
    }
    else if (type == "removeattack" || type == "removeignoreattack")
    {
        handleRemoveAttack(args, tab);
    }
    else if (type == "addignoreattack")
    {
        handleAddIgnoreAttack(args, tab);
    }
    else if (tab->handleCommand(type, args))
    {
        // Nothing to do
    }
    else if (type == "hack")
    {
        handleHack(args, tab);
    }
    else
    {
        tab->chatLog(_("Unknown command."));
    }
}

char CommandHandler::parseBoolean(const std::string &value)
{
    std::string opt = value.substr(0, 1);

    if (opt == "1" ||
        opt == "y" || opt == "Y" ||
        opt == "t" || opt == "T")
        return 1;
    else if (opt == "0" ||
             opt == "n" || opt == "N" ||
             opt == "f" || opt == "F")
        return 0;
    else
        return -1;
}

void CommandHandler::handleAnnounce(const std::string &args,
                                    ChatTab *tab _UNUSED_)
{
    Net::getAdminHandler()->announce(args);
}

void CommandHandler::handleHelp(const std::string &args _UNUSED_,
                                ChatTab *tab _UNUSED_)
{
    if (helpWindow)
    {
        helpWindow->loadHelp("chatcommands");
        helpWindow->requestMoveToTop();
    }
}

void CommandHandler::handleWhere(const std::string &args _UNUSED_,
                                 ChatTab *tab)
{
    std::ostringstream where;
    where << Game::instance()->getCurrentMapName() << ", coordinates: "
          << ((player_node->getPixelX() - 16) / 32) << ", "
          << ((player_node->getPixelY() - 32) / 32);

    tab->chatLog(where.str(), BY_SERVER);
}

void CommandHandler::handleWho(const std::string &args _UNUSED_,
                               ChatTab *tab _UNUSED_)
{
    Net::getChatHandler()->who();
}

void CommandHandler::handleMsg(const std::string &args, ChatTab *tab)
{
    std::string recvnick = "";
    std::string msg = "";

    if (args.substr(0, 1) == "\"")
    {
        const std::string::size_type pos = args.find('"', 1);
        if (pos != std::string::npos)
        {
            recvnick = args.substr(1, pos - 1);
            if (pos + 2 < args.length())
                msg = args.substr(pos + 2, args.length());
        }
    }
    else
    {
        const std::string::size_type pos = args.find(" ");
        if (pos != std::string::npos)
        {
            recvnick = args.substr(0, pos);
            if (pos + 1 < args.length())
                msg = args.substr(pos + 1, args.length());
        }
        else
        {
            recvnick = std::string(args);
            msg = "";
        }
    }

    trim(msg);

    if (msg.length() > 0)
    {
        std::string playerName = player_node->getName();
        std::string tempNick = recvnick;

        toLower(playerName);
        toLower(tempNick);

        if (tempNick.compare(playerName) == 0 || args.empty())
            return;

        chatWindow->whisper(recvnick, msg, BY_PLAYER);
    }
    else
        tab->chatLog(_("Cannot send empty whispers!"), BY_SERVER);
}

void CommandHandler::handleQuery(const std::string &args, ChatTab *tab)
{
    if (chatWindow && chatWindow->addWhisperTab(args, true))
        return;

    tab->chatLog(strprintf(_("Cannot create a whisper tab for nick \"%s\"! "
        "It either already exists, or is you."),
        args.c_str()), BY_SERVER);
}

void CommandHandler::handleClear(const std::string &args _UNUSED_,
                                 ChatTab *tab _UNUSED_)
{
    if (chatWindow)
        chatWindow->clearTab();
}

void CommandHandler::handleJoin(const std::string &args, ChatTab *tab)
{
    if (!tab)
        return;

    std::string::size_type pos = args.find(' ');
    std::string name(args, 0, pos);
    std::string password(args, pos + 1);
    tab->chatLog(strprintf(_("Requesting to join channel %s."), name.c_str()));
    Net::getChatHandler()->enterChannel(name, password);
}

void CommandHandler::handleListChannels(const std::string &args _UNUSED_,
                                        ChatTab *tab _UNUSED_)
{
    Net::getChatHandler()->channelList();
}

void CommandHandler::handleCreateParty(const std::string &args, ChatTab *tab)
{
    if (!tab)
        return;

    if (args.empty())
        tab->chatLog(_("Party name is missing."), BY_SERVER);
    else
        Net::getPartyHandler()->create(args);
}

void CommandHandler::handleCreateGuild(const std::string &args, ChatTab *tab)
{
    if (!tab)
        return;

    if (args.empty())
        tab->chatLog(_("Guild name is missing."), BY_SERVER);
    else
        Net::getGuildHandler()->create(args);
}

void CommandHandler::handleParty(const std::string &args, ChatTab *tab)
{
    if (!tab)
        return;

    if (args != "")
        Net::getPartyHandler()->invite(args);
    else
        tab->chatLog(_("Please specify a name."), BY_SERVER);
}

void CommandHandler::handleMe(const std::string &args, ChatTab *tab)
{
    if (!tab)
    {
        Net::getChatHandler()->me(args);
        return;
    }

    const std::string str = strprintf("*%s*", args.c_str());
    switch (tab->getType())
    {
        case ChatTab::TAB_PARTY:
        {
            Net::getPartyHandler()->chat(str);
            break;
        }
        case ChatTab::TAB_GUILD:
        {
            if (!player_node)
                return;
            const Guild *guild = player_node->getGuild();
            if (guild)
                Net::getGuildHandler()->chat(guild->getId(), str);
            break;
        }
        default:
            Net::getChatHandler()->me(args);
            break;
    }
}

void CommandHandler::handleToggle(const std::string &args, ChatTab *tab)
{
    if (args.empty())
    {
        if (chatWindow && tab)
        {
            tab->chatLog(chatWindow->getReturnTogglesChat() ?
                _("Return toggles chat.") : _("Message closes chat."));
        }
        return;
    }

    char opt = parseBoolean(args);

    switch (opt)
    {
        case 1:
            if (tab)
                tab->chatLog(_("Return now toggles chat."));
            if (chatWindow)
                chatWindow->setReturnTogglesChat(true);
            return;
        case 0:
            if (tab)
                tab->chatLog(_("Message now closes chat."));
            if (chatWindow)
                chatWindow->setReturnTogglesChat(false);
            return;
        case -1:
            if (tab)
                tab->chatLog(strprintf(BOOLEAN_OPTIONS, "toggle"));
            return;
        default:
            return;
    }
}

void CommandHandler::handlePresent(const std::string &args _UNUSED_,
                                   ChatTab *tab _UNUSED_)
{
    if (chatWindow)
        chatWindow->doPresent();
}

void CommandHandler::handleIgnore(const std::string &args,
                                  ChatTab *tab _UNUSED_)
{
    changeRelation(args, PlayerRelation::IGNORED, "ignored", tab);
}

void CommandHandler::handleFriend(const std::string &args, ChatTab *tab)
{
    changeRelation(args, PlayerRelation::FRIEND, _("friend"), tab);
}

void CommandHandler::handleDisregard(const std::string &args, ChatTab *tab)
{
    changeRelation(args, PlayerRelation::DISREGARDED, _("disregarded"), tab);
}

void CommandHandler::handleNeutral(const std::string &args, ChatTab *tab)
{
    changeRelation(args, PlayerRelation::NEUTRAL, _("neutral"), tab);
}

void CommandHandler::changeRelation(const std::string &args,
                                    PlayerRelation::Relation relation,
                                    const std::string &relationText,
                                    ChatTab *tab)
{
    if (args.empty())
    {
        if (tab)
            tab->chatLog(_("Please specify a name."), BY_SERVER);
        return;
    }

    if (player_relations.getRelation(args) == relation)
    {
        if (tab)
        {
            tab->chatLog(strprintf(_("Player already %s!"),
                         relationText.c_str()), BY_SERVER);
        }
        return;
    }
    else
    {
        player_relations.setRelation(args, relation);
    }

    if (player_relations.getRelation(args) == relation)
    {
        if (tab)
        {
            tab->chatLog(strprintf(_("Player successfully %s!"),
                         relationText.c_str()), BY_SERVER);
        }
    }
    else
    {
        if (tab)
        {
            tab->chatLog(strprintf(_("Player could not be %s!"),
                         relationText.c_str()), BY_SERVER);
        }
    }
}

void CommandHandler::handleUnignore(const std::string &args, ChatTab *tab)
{
    if (args.empty())
    {
        if (tab)
            tab->chatLog(_("Please specify a name."), BY_SERVER);
        return;
    }

    if (player_relations.getRelation(args) == PlayerRelation::IGNORED)
    {
        player_relations.removePlayer(args);
    }
    else
    {
        if (tab)
            tab->chatLog(_("Player wasn't ignored!"), BY_SERVER);
        return;
    }

    if (tab)
    {
        if (player_relations.getRelation(args) != PlayerRelation::IGNORED)
            tab->chatLog(_("Player no longer ignored!"), BY_SERVER);
        else
            tab->chatLog(_("Player could not be unignored!"), BY_SERVER);
    }
}


void CommandHandler::handleErase(const std::string &args, ChatTab *tab)
{
    if (args.empty())
    {
        if (tab)
            tab->chatLog(_("Please specify a name."), BY_SERVER);
        return;
    }

    if (player_relations.getRelation(args) == PlayerRelation::ERASED)
    {
        if (tab)
            tab->chatLog(_("Player already erased!"), BY_SERVER);
        return;
    }
    else
    {
        player_relations.setRelation(args, PlayerRelation::ERASED);
    }

    if (tab)
    {
        if (player_relations.getRelation(args) == PlayerRelation::ERASED)
            tab->chatLog(_("Player successfully erased!"), BY_SERVER);
        else
            tab->chatLog(_("Player could not be erased!"), BY_SERVER);
    }
}

void CommandHandler::handleQuit(const std::string &args _UNUSED_,
                                ChatTab *tab _UNUSED_)
{
//    quit();
}

void CommandHandler::handleShowAll(const std::string &args _UNUSED_,
                                   ChatTab *tab _UNUSED_)
{
    if (actorSpriteManager)
        actorSpriteManager->printAllToChat();
}

void CommandHandler::handleMove(const std::string &args, ChatTab *tab _UNUSED_)
{
    int x = 0;
    int y = 0;

    if (player_node && parse2Int(args, &x, &y))
        player_node->moveTo(x, y);
}

void CommandHandler::handleNavigate(const std::string &args,
                                    ChatTab *tab _UNUSED_)
{
    if (!player_node)
        return;

    int x = 0;
    int y = 0;

    if (parse2Int(args, &x, &y))
        player_node->navigateTo(x, y);
    else
        player_node->navigateClean();
}

bool CommandHandler::parse2Int(const std::string &args, int *x, int *y)
{
    bool isValid = false;
    const std::string::size_type pos = args.find(" ");
    if (pos != std::string::npos)
    {
        if (pos + 1 < args.length())
        {
            *x = atoi(args.substr(0, pos).c_str());
            *y = atoi(args.substr(pos + 1, args.length()).c_str());
            isValid = true;
        }
    }
    return isValid;
}

void CommandHandler::handleTarget(const std::string &args,
                                  ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager || !player_node)
        return;

    Being* target = actorSpriteManager->findNearestByName(args);
    if (target)
        player_node->setTarget(target);
}

void CommandHandler::handleCloseAll(const std::string &args _UNUSED_,
                                    ChatTab *tab _UNUSED_)
{
    if (chatWindow)
        chatWindow->removeAllWhispers();
}

void CommandHandler::handleIgnoreAll(const std::string &args _UNUSED_,
                                     ChatTab *tab _UNUSED_)
{
    if (chatWindow)
        chatWindow->ignoreAllWhispers();
}

void CommandHandler::handleOutfit(const std::string &args,
                                  ChatTab *tab _UNUSED_)
{
    if (outfitWindow)
    {
        if (!args.empty())
        {
            const std::string op = args.substr(0, 1);
            if (op == "n")
                outfitWindow->wearNextOutfit(true);
            else if (op == "p")
                outfitWindow->wearPreviousOutfit(true);
            else
                outfitWindow->wearOutfit(atoi(args.c_str()) - 1, false, true);
        }
        else
        {
            outfitWindow->wearOutfit(atoi(args.c_str()) - 1, false, true);
        }
    }
}

void CommandHandler::handleEmote(const std::string &args,
                                 ChatTab *tab _UNUSED_)
{
    if (player_node)
        player_node->emote(static_cast<Uint8>(atoi(args.c_str())));
}

void CommandHandler::handleAway(const std::string &args, ChatTab *tab _UNUSED_)
{
    if (player_node)
        player_node->setAway(args);
}

void CommandHandler::handleFollow(const std::string &args, ChatTab *tab)
{
    if (!player_node)
        return;

    if (!args.empty())
    {
        player_node->setFollow(args);
    }
    else if (tab)
    {
        if (tab->getType() == ChatTab::TAB_WHISPER)
        {
            WhisperTab *wTab = static_cast<WhisperTab*>(tab);
            if (wTab)
                player_node->setFollow(wTab->getNick());
        }
    }
}

void CommandHandler::handleImitation(const std::string &args, ChatTab *tab)
{
    if (!player_node)
        return;

    if (!args.empty())
    {
        player_node->setImitate(args);
    }
    else if (tab && tab->getType() == ChatTab::TAB_WHISPER)
    {
        WhisperTab *wTab = static_cast<WhisperTab*>(tab);
        if (wTab)
            player_node->setImitate(wTab->getNick());
    }
    else
    {
        player_node->setImitate("");
    }
}

void CommandHandler::handleHeal(const std::string &args, ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager)
        return;

    if (!args.empty())
    {
        Being *being = actorSpriteManager->findBeingByName(
            args, Being::PLAYER);
        if (being)
            actorSpriteManager->heal(player_node, being);
    }
    else
    {
        actorSpriteManager->heal(player_node, player_node);
    }
}

void CommandHandler::handleHack(const std::string &args, ChatTab *tab _UNUSED_)
{
    Net::getChatHandler()->sendRaw(args);
}

void CommandHandler::handleMail(const std::string &args, ChatTab *tab _UNUSED_)
{
    Net::getChatHandler()->privateMessage("AuctionBot", "!mail " + args);
}

void CommandHandler::handlePriceLoad(const std::string &args _UNUSED_,
                                     ChatTab *tab _UNUSED_)
{
    if (shopWindow)
        shopWindow->loadList();
}

void CommandHandler::handlePriceSave(const std::string &args _UNUSED_,
                                     ChatTab *tab _UNUSED_)
{
    if (shopWindow)
        shopWindow->saveList();
}

void CommandHandler::handleDisconnect(const std::string &args _UNUSED_,
                                      ChatTab *tab _UNUSED_)
{
    Net::getGameHandler()->disconnect2();
}

void CommandHandler::handleUndress(const std::string &args,
                                   ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager)
        return;

    Being* target = actorSpriteManager->findNearestByName(args);
    if (target)
        Net::getBeingHandler()->undress(target);
}

void CommandHandler::handleAttack(const std::string &args,
                                  ChatTab *tab _UNUSED_)
{
    if (!player_node || !actorSpriteManager)
        return;

    Being* target = actorSpriteManager->findNearestByName(args);
    if (target)
        player_node->setTarget(target);
    player_node->attack2(player_node->getTarget(), true);
}

void CommandHandler::handleTrade(const std::string &args,
                                 ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager)
        return;

    Being *being = actorSpriteManager->findBeingByName(args, Being::PLAYER);
    if (being)
    {
        Net::getTradeHandler()->request(being);
        tradePartnerName = being->getName();
        if (tradeWindow)
            tradeWindow->clear();
    }
}

void CommandHandler::handleDirs(const std::string &args _UNUSED_,
                                ChatTab *tab _UNUSED_)
{
    if (!player_node || !debugChatTab)
        return;

    debugChatTab->chatLog("config directory: "
        + Client::getConfigDirectory());
    debugChatTab->chatLog("logs directory: "
        + Client::getLocalDataDirectory());
    debugChatTab->chatLog("screenshots directory: "
        + Client::getScreenshotDirectory());
}

void CommandHandler::handleInfo(const std::string &args _UNUSED_,
                                ChatTab *tab)
{
    switch (tab->getType())
    {
        case ChatTab::TAB_GUILD:
        {
            if (!player_node)
                return;
            const Guild *guild = player_node->getGuild();
            if (guild)
                Net::getGuildHandler()->info(guild->getId());
            break;
        }
        default:
            break;
    }
}

void CommandHandler::handleWait(const std::string &args,
                                ChatTab *tab _UNUSED_)
{
    if (player_node)
        player_node->waitFor(args);
}

void CommandHandler::handleUptime(const std::string &args _UNUSED_,
                                  ChatTab *tab _UNUSED_)
{
    if (!debugChatTab)
        return;

    if (cur_time < start_time)
    {
        debugChatTab->chatLog(strprintf(_("Client uptime: %s"), "unknown"));
    }
    else
    {
        std::string str;
        int timeDiff = cur_time - start_time;

        const int weeks = timeDiff / 60 / 60 / 24 / 7;
        if (weeks > 0)
        {
            str = strprintf(ngettext("%d week", "%d weeks", weeks), weeks);
            timeDiff -= weeks * 60 * 60 * 24 * 7;
        }

        const int days = timeDiff / 60 / 60 / 24;
        if (days > 0)
        {
            if (!str.empty())
                str += ", ";
            str += strprintf(ngettext("%d day", "%d days", days), days);
            timeDiff -= days * 60 * 60 * 24;
        }
        const int hours = timeDiff / 60 / 60;
        if (hours > 0)
        {
            if (!str.empty())
                str += ", ";
            str += strprintf(ngettext("%d hour", "%d hours", hours), hours);
            timeDiff -= hours * 60 * 60;
        }
        const int min = timeDiff / 60;
        if (min > 0)
        {
            if (!str.empty())
                str += ", ";
            str += strprintf(ngettext("%d minute", "%d minutes", min), min);
            timeDiff -= min * 60;
        }

        if (timeDiff > 0)
        {
            if (!str.empty())
                str += ", ";
            str += strprintf(ngettext("%d second", "%d seconds",
                timeDiff), timeDiff);
        }
        debugChatTab->chatLog(strprintf(_("Client uptime: %s"), str.c_str()));
    }
}

void CommandHandler::handleAddPriorityAttack(const std::string &args,
                                             ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager
        || actorSpriteManager->isInPriorityAttackList(args))
    {
        return;
    }

    actorSpriteManager->removeAttackMob(args);
    actorSpriteManager->addPriorityAttackMob(args);

    if (socialWindow)
        socialWindow->updateAttackFilter();
}

void CommandHandler::handleAddAttack(const std::string &args,
                                     ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager || actorSpriteManager->isInAttackList(args))
        return;

    actorSpriteManager->removeAttackMob(args);
    actorSpriteManager->addAttackMob(args);

    if (socialWindow)
        socialWindow->updateAttackFilter();
}

void CommandHandler::handleRemoveAttack(const std::string &args,
                                        ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager || args.empty()
        || !actorSpriteManager->isInAttackList(args))
    {
        return;
    }

    actorSpriteManager->removeAttackMob(args);

    if (socialWindow)
        socialWindow->updateAttackFilter();
}

void CommandHandler::handleAddIgnoreAttack(const std::string &args,
                                           ChatTab *tab _UNUSED_)
{
    if (!actorSpriteManager || actorSpriteManager->isInIgnoreAttackList(args))
        return;

    actorSpriteManager->removeAttackMob(args);
    actorSpriteManager->addIgnoreAttackMob(args);

    if (socialWindow)
        socialWindow->updateAttackFilter();
}

void CommandHandler::handleCacheInfo(const std::string &args _UNUSED_,
                                     ChatTab *tab _UNUSED_)
{
    if (!chatWindow || !debugChatTab)
        return;

    SDLFont *font = dynamic_cast<SDLFont *>(chatWindow->getFont());
    if (!font)
        return;

    std::list<SDLTextChunk> *cache = font->getCache();
    if (!cache)
        return;

    debugChatTab->chatLog(_("font cache size"));
    std::string str;
    for (int f = 0; f < 256; f ++)
    {
        if (!cache[f].empty())
        {
            str += strprintf("%d: %u, ", f,
                static_cast<unsigned int>(cache[f].size()));
        }
    }
    debugChatTab->chatLog(str);
#ifdef DEBUG_FONT_COUNTERS
    debugChatTab->chatLog("");
    debugChatTab->chatLog(_("Created: ") + toString(font->getCreateCounter()));
    debugChatTab->chatLog(_("Deleted: ") + toString(font->getDeleteCounter()));
#endif
}