/* * The Mana Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2012 The Mana Developers * * This file is part of The Mana 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 . */ #include "net/tmwa/chathandler.h" #include "actorspritemanager.h" #include "avatar.h" #include "being.h" #include "event.h" #include "localplayer.h" #include "playerrelations.h" #include "gui/socialwindow.h" #include "net/tmwa/loginhandler.h" #include "net/tmwa/messagein.h" #include "net/tmwa/messageout.h" #include "net/tmwa/protocol.h" #include "utils/gettext.h" #include "utils/stringutils.h" #include extern Net::ChatHandler *chatHandler; namespace TmwAthena { ChatHandler::ChatHandler() { static const uint16_t _messages[] = { SMSG_BEING_CHAT, SMSG_PLAYER_CHAT, SMSG_WHISPER, SMSG_WHISPER_RESPONSE, SMSG_GM_CHAT, SMSG_SCRIPT_MESSAGE, SMSG_ONLINE_LIST, 0 }; handledMessages = _messages; chatHandler = this; } void ChatHandler::handleMessage(MessageIn &msg) { 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(); } switch (msg.readInt8()) { case 0x00: // Success (don't need to report) break; case 0x01: { Event event(Event::WhisperError); event.setString("nick", nick); event.setString("error", strprintf(_("Whisper could " "not be sent, %s is offline."), nick.c_str())); event.trigger(Event::ChatChannel); } break; case 0x02: { Event event(Event::WhisperError); event.setString("nick", nick); event.setString("error", strprintf(_("Whisper could " "not be sent, ignored by %s."), nick.c_str())); event.Event::trigger(Event::ChatChannel); } break; } break; // Received whisper case SMSG_WHISPER: chatMsgLength = msg.readInt16() - 28; nick = msg.readString(24); if (chatMsgLength <= 0) break; chatMsg = msg.readString(chatMsgLength); // This is a sure sign of some special command, none of which are // supported at the moment. if (chatMsg.compare(0, 3, "\302\202!") == 0) return; if (nick != "Server") { // Ignore the shop commands since this client doesn't support // them at the moment. if (chatMsg.find("!selllist ") == 0 || chatMsg.find("!buylist ") == 0 || chatMsg.find("!buyitem ") == 0 || chatMsg.find("!sellitem ") == 0) return; if (player_relations.hasPermission(nick, PlayerPermissions::WHISPER)) { Event event(Event::Whisper); event.setString("nick", nick); event.setString("message", chatMsg); event.trigger(Event::ChatChannel); } } else { serverNotice(chatMsg); } break; // Received speech from being case SMSG_BEING_CHAT: { chatMsgLength = msg.readInt16() - 8; int beingId = msg.readInt32(); being = actorSpriteManager->findBeing(beingId); if (!being || chatMsgLength <= 0) break; chatMsg = msg.readString(chatMsgLength); 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); } int perms; if (being->getType() == Being::PLAYER) { perms = player_relations.checkPermissionSilently(sender_name, PlayerPermissions::SPEECH_LOG | PlayerPermissions::SPEECH_FLOAT); } else { perms = player_relations.getDefault() & (PlayerPermissions::SPEECH_LOG | PlayerPermissions::SPEECH_FLOAT); } // This is a sure sign of some special command of manaplus, // none of those are supported at the moment. if (chatMsg.compare(0, 2, "\302\202") == 0) return; trim(chatMsg); std::string reducedMessage = chatMsg; chatMsg = removeColors(sender_name) + " : " + reducedMessage; Event event(Event::Being); event.setString("message", chatMsg); event.setString("text", reducedMessage); event.setString("nick", sender_name); event.setInt("beingId", beingId); event.setInt("permissions", perms); event.trigger(Event::ChatChannel); break; } case SMSG_PLAYER_CHAT: case SMSG_GM_CHAT: { chatMsgLength = msg.readInt16() - 4; if (chatMsgLength <= 0) break; chatMsg = msg.readString(chatMsgLength); if (msg.getId() == SMSG_PLAYER_CHAT) { std::string::size_type pos = chatMsg.find(" : ", 0); std::string mes = chatMsg; if (pos != std::string::npos) chatMsg.erase(0, pos + 3); trim(chatMsg); Event event(Event::Player); event.setString("message", mes); event.setString("text", chatMsg); event.setString("nick", local_player->getName()); event.setInt("beingId", local_player->getId()); event.setInt("permissions", player_relations.getDefault() & (PlayerPermissions::SPEECH_LOG | PlayerPermissions::SPEECH_FLOAT)); event.trigger(Event::ChatChannel); } else { Event event(Event::Announcement); event.setString("message", chatMsg); event.trigger(Event::ChatChannel); } break; } case SMSG_SCRIPT_MESSAGE: { chatMsgLength = msg.readInt16() - 5; if (chatMsgLength <= 0) break; msg.readInt8(); // message type serverNotice(msg.readString(chatMsgLength)); break; } case SMSG_ONLINE_LIST: { int length = msg.readInt16(); int count = (length - 4) / 31; std::vector players; for (int i = 0; i < count; i++) { msg.readInt32(); // account id std::string nick = msg.readString(24); msg.readInt8(); // level msg.readInt8(); // gm level msg.readInt8(); // gender Avatar *avatar = new Avatar(nick); avatar->setOnline(true); players.push_back(avatar); } socialWindow->setPlayersOnline(players); break; } } } static void sendChatMessage(const std::string &mes) { MessageOut outMsg(CMSG_CHAT_MESSAGE); // Added + 1 in order to let eAthena parse admin commands correctly outMsg.writeInt16(mes.length() + 4 + 1); outMsg.writeString(mes, mes.length() + 1); } void ChatHandler::talk(const std::string &text) { const auto loginHandler = static_cast(Net::getLoginHandler()); if (loginHandler->getServerVersion() >= 0x100408) sendChatMessage(text); else sendChatMessage(local_player->getName() + " : " + text); } 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(text.length() + 28); outMsg.writeString(recipient, 24); outMsg.writeString(text, text.length()); mSentWhispers.push(recipient); } void ChatHandler::channelList() { serverNotice(_("Channels are not supported!")); } void ChatHandler::enterChannel(const std::string &channel, const std::string &password) { serverNotice(_("Channels are not supported!")); } void ChatHandler::quitChannel(int channelId) { serverNotice(_("Channels are not supported!")); } void ChatHandler::sendToChannel(int channelId, const std::string &text) { serverNotice(_("Channels are not supported!")); } void ChatHandler::userList(const std::string &channel) { serverNotice(_("Channels are not supported!")); } void ChatHandler::setChannelTopic(int channelId, const std::string &text) { serverNotice(_("Channels are not supported!")); } void ChatHandler::setUserMode(int channelId, const std::string &name, int mode) { serverNotice(_("Channels are not supported!")); } void ChatHandler::kickUser(int channelId, const std::string &name) { serverNotice(_("Channels are not supported!")); } void ChatHandler::requestOnlineList() { MessageOut outMsg(CMSG_ONLINE_LIST); } } // namespace TmwAthena