/*
* The ManaPlus Client
* Copyright (C) 2004-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 "net/tmwa/chathandler.h"
#include "actorspritemanager.h"
#include "being.h"
#include "configuration.h"
#include "event.h"
#include "game.h"
#include "localplayer.h"
#include "playerrelations.h"
#include "log.h"
#include "gui/chatwindow.h"
#include "gui/shopwindow.h"
#include "gui/widgets/chattab.h"
#include "net/messagein.h"
#include "net/messageout.h"
#include "net/tmwa/protocol.h"
#include "utils/gettext.h"
#include "utils/stringutils.h"
#include <string>
#include "debug.h"
extern Net::ChatHandler *chatHandler;
namespace TmwAthena
{
ChatHandler::ChatHandler()
{
static const Uint16 _messages[] =
{
SMSG_BEING_CHAT,
SMSG_PLAYER_CHAT,
SMSG_WHISPER,
SMSG_WHISPER_RESPONSE,
SMSG_GM_CHAT,
SMSG_MVP, // MVP
0
};
handledMessages = _messages;
chatHandler = this;
}
void ChatHandler::handleMessage(Net::MessageIn &msg)
{
if (!localChatTab)
return;
Being *being;
std::string chatMsg;
std::string nick;
int chatMsgLength;
switch (msg.getId())
{
case SMSG_WHISPER_RESPONSE:
{
if (mSentWhispers.empty())
{
nick = "user";
}
else
{
nick = mSentWhispers.front();
mSentWhispers.pop();
}
int type = msg.readInt8();
switch (type)
{
case 0x00:
// Success (don't need to report)
break;
case 0x01:
if (chatWindow)
{
chatWindow->whisper(nick,
strprintf(_("Whisper could not be "
"sent, %s is offline."), nick.c_str()), BY_SERVER);
}
break;
case 0x02:
if (chatWindow)
{
chatWindow->whisper(nick,
strprintf(_("Whisper could not "
"be sent, ignored by %s."), nick.c_str()),
BY_SERVER);
}
break;
default:
if (logger)
{
logger->log("QQQ SMSG_WHISPER_RESPONSE:"
+ toString(type));
}
}
break;
}
// Received whisper
case SMSG_WHISPER:
{
chatMsgLength = msg.readInt16() - 28;
nick = msg.readString(24);
if (chatMsgLength <= 0)
break;
chatMsg = msg.readString(chatMsgLength);
if (chatMsg.find("\302\202!") == 0)
chatMsg = chatMsg.substr(2);
if (nick != "Server")
{
if (player_relations.hasPermission(
nick, PlayerRelation::WHISPER))
{
bool tradeBot = config.getBoolValue("tradebot");
bool showMsg = !config.getBoolValue("hideShopMessages");
if (player_relations.hasPermission(
nick, PlayerRelation::TRADE))
{
if (shopWindow)
{ //commands to shop from player
if (chatMsg.find("!selllist ") == 0)
{
if (tradeBot)
{
if (showMsg && chatWindow)
chatWindow->whisper(nick, chatMsg);
shopWindow->giveList(nick,
ShopWindow::SELL);
}
}
else if (chatMsg.find("!buylist ") == 0)
{
if (tradeBot)
{
if (showMsg && chatWindow)
chatWindow->whisper(nick, chatMsg);
shopWindow->giveList(nick,
ShopWindow::BUY);
}
}
else if (chatMsg.find("!buyitem ") == 0)
{
if (showMsg && chatWindow)
chatWindow->whisper(nick, chatMsg);
if (tradeBot)
{
shopWindow->processRequest(nick, chatMsg,
ShopWindow::BUY);
}
}
else if (chatMsg.find("!sellitem ") == 0)
{
if (showMsg && chatWindow)
chatWindow->whisper(nick, chatMsg);
if (tradeBot)
{
shopWindow->processRequest(nick, chatMsg,
ShopWindow::SELL);
}
}
else if (chatMsg.length() > 3
&& chatMsg.find("\302\202") == 0)
{
chatMsg = chatMsg.erase(0, 2);
if (showMsg && chatWindow)
chatWindow->whisper(nick, chatMsg);
if (chatMsg.find("B1") == 0
|| chatMsg.find("S1") == 0)
{
shopWindow->showList(nick, chatMsg);
}
}
else if (chatWindow)
{
chatWindow->whisper(nick, chatMsg);
}
}
else if (chatWindow)
{
chatWindow->whisper(nick, chatMsg);
}
}
else
{
if (chatWindow && (showMsg
|| (chatMsg.find("!selllist")
!= 0 && chatMsg.find("!buylist") != 0)))
{
chatWindow->whisper(nick, chatMsg);
}
}
}
}
else if (localChatTab)
{
localChatTab->chatLog(chatMsg, BY_SERVER);
}
break;
}
// Received speech from being
case SMSG_BEING_CHAT:
{
if (!actorSpriteManager)
return;
chatMsgLength = msg.readInt16() - 8;
being = actorSpriteManager->findBeing(msg.readInt32());
if (!being || chatMsgLength <= 0)
break;
std::string str2;
chatMsg = msg.readRawString(chatMsgLength);
if (being->getType() == Being::PLAYER)
being->setTalkTime();
std::string::size_type pos = chatMsg.find(" : ", 0);
std::string sender_name = ((pos == std::string::npos)
? "" : chatMsg.substr(0, pos));
if (sender_name != being->getName()
&& being->getType() == Being::PLAYER)
{
if (!being->getName().empty())
sender_name = being->getName();
}
else
{
chatMsg.erase(0, pos + 3);
}
trim(chatMsg);
// We use getIgnorePlayer instead of ignoringPlayer here
// because ignorePlayer' side effects are triggered
// right below for Being::IGNORE_SPEECH_FLOAT.
if (player_relations.checkPermissionSilently(sender_name,
PlayerRelation::SPEECH_LOG) && chatWindow)
{
chatWindow->resortChatLog(removeColors(sender_name) + " : "
+ chatMsg, BY_OTHER);
}
if (player_relations.hasPermission(sender_name,
PlayerRelation::SPEECH_FLOAT))
{
being->setSpeech(chatMsg);
}
break;
}
case SMSG_PLAYER_CHAT:
case SMSG_GM_CHAT:
{
chatMsgLength = msg.readInt16() - 4;
if (chatMsgLength <= 0)
break;
std::string str2;
chatMsg = msg.readRawString(chatMsgLength);
std::string::size_type pos = chatMsg.find(" : ", 0);
if (msg.getId() == SMSG_PLAYER_CHAT)
{
if (chatWindow)
chatWindow->resortChatLog(chatMsg, BY_PLAYER);
// if (localChatTab)
// localChatTab->chatLog(chatMsg, BY_PLAYER);
const std::string senseStr = "You sense the following: ";
if (actorSpriteManager && !chatMsg.find(senseStr))
{
actorSpriteManager->parseLevels(
chatMsg.substr(senseStr.size()));
}
if (pos != std::string::npos)
chatMsg.erase(0, pos + 3);
trim(chatMsg);
if (player_node)
player_node->setSpeech(chatMsg);
}
else if (localChatTab)
{
localChatTab->chatLog(chatMsg, BY_GM);
}
break;
}
case SMSG_MVP:
{
// Display MVP player
int id = msg.readInt32(); // id
if (localChatTab && actorSpriteManager)
{
being = actorSpriteManager->findBeing(id);
if (!being)
{
localChatTab->chatLog(_("MVP player."), BY_SERVER);
}
else
{
localChatTab->chatLog(_("MVP player: ")
+ being->getName(), BY_SERVER);
}
}
break;
}
default:
break;
}
}
void ChatHandler::talk(const std::string &text)
{
if (!player_node)
return;
std::string mes = player_node->getName() + " : " + text;
// std::string mes = player_node->getName() + "zzzz : " + text;
MessageOut outMsg(CMSG_CHAT_MESSAGE);
// Added + 1 in order to let eAthena parse admin commands correctly
outMsg.writeInt16(static_cast<short>(mes.length() + 4 + 1));
outMsg.writeString(mes, static_cast<int>(mes.length() + 1));
}
void ChatHandler::talkRaw(const std::string &mes)
{
MessageOut outMsg(CMSG_CHAT_MESSAGE);
// Added + 1 in order to let eAthena parse admin commands correctly
outMsg.writeInt16(static_cast<short>(mes.length() + 4 + 1));
outMsg.writeString(mes, static_cast<int>(mes.length() + 1));
}
void ChatHandler::me(const std::string &text)
{
std::string action = strprintf("*%s*", text.c_str());
talk(action);
}
void ChatHandler::privateMessage(const std::string &recipient,
const std::string &text)
{
MessageOut outMsg(CMSG_CHAT_WHISPER);
outMsg.writeInt16(static_cast<short>(text.length() + 28));
outMsg.writeString(recipient, 24);
outMsg.writeString(text, static_cast<int>(text.length()));
mSentWhispers.push(recipient);
}
void ChatHandler::channelList()
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::enterChannel(const std::string &channel A_UNUSED,
const std::string &password A_UNUSED)
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::quitChannel(int channelId A_UNUSED)
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::sendToChannel(int channelId A_UNUSED,
const std::string &text A_UNUSED)
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::userList(const std::string &channel A_UNUSED)
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::setChannelTopic(int channelId A_UNUSED,
const std::string &text A_UNUSED)
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::setUserMode(int channelId A_UNUSED,
const std::string &name A_UNUSED,
int mode A_UNUSED)
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::kickUser(int channelId A_UNUSED,
const std::string &name A_UNUSED)
{
SERVER_NOTICE(_("Channels are not supported!"))
}
void ChatHandler::who()
{
MessageOut outMsg(CMSG_WHO_REQUEST);
}
void ChatHandler::sendRaw(const std::string &args)
{
std::string line = args;
std::string str;
MessageOut *outMsg = 0;
if (line == "")
return;
std::string::size_type pos = line.find(" ");
if (pos != std::string::npos)
{
str = line.substr(0, pos);
outMsg = new MessageOut(static_cast<short>(atoi(str.c_str())));
line = line.substr(pos + 1);
pos = line.find(" ");
}
else
{
outMsg = new MessageOut(static_cast<short>(atoi(line.c_str())));
delete outMsg;
return;
}
while (pos != std::string::npos)
{
str = line.substr(0, pos);
processRaw(*outMsg, str);
line = line.substr(pos + 1);
pos = line.find(" ");
}
if (outMsg)
{
if (line != "")
processRaw(*outMsg, line);
delete outMsg;
}
}
void ChatHandler::processRaw(MessageOut &outMsg, std::string &line)
{
std::string::size_type pos = line.find(":");
if (pos == std::string::npos)
{
int i = atoi(line.c_str());
if (line.length() <= 3)
outMsg.writeInt8(static_cast<char>(i));
else if (line.length() <= 5)
outMsg.writeInt16(static_cast<short>(i));
else
outMsg.writeInt32(i);
}
else
{
std::string header = line.substr(0, pos);
std::string data = line.substr(pos + 1);
if (header.length() != 1)
return;
int i = 0;
switch (header[0])
{
case '1':
case '2':
case '4':
i = atoi(data.c_str());
break;
default:
break;
}
switch (header[0])
{
case '1':
outMsg.writeInt8(static_cast<char>(i));
break;
case '2':
outMsg.writeInt16(static_cast<short>(i));
break;
case '4':
outMsg.writeInt32(i);
break;
case 'c':
{
pos = line.find(",");
if (pos != std::string::npos)
{
unsigned short x = static_cast<unsigned short>(
atoi(data.substr(0, pos).c_str()));
data = data.substr(pos + 1);
pos = line.find(",");
if (pos == std::string::npos)
break;
unsigned short y = static_cast<unsigned short>(
atoi(data.substr(0, pos).c_str()));
int dir = atoi(data.substr(pos + 1).c_str());
outMsg.writeCoordinates(x, y,
static_cast<unsigned char>(dir));
}
break;
}
case 't':
outMsg.writeString(data, static_cast<int>(data.length()));
break;
default:
break;
}
}
}
} // namespace TmwAthena