/* * 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/manaserv/chathandler.h" #include "actorspritemanager.h" #include "being.h" #include "client.h" #include "channel.h" #include "channelmanager.h" #include "event.h" #include "log.h" #include "playerrelations.h" #include "gui/widgets/channeltab.h" #include "net/manaserv/connection.h" #include "net/manaserv/messagein.h" #include "net/manaserv/messageout.h" #include "net/manaserv/manaserv_protocol.h" #include "utils/gettext.h" #include "utils/stringutils.h" #include #include extern Being *local_player; extern Net::ChatHandler *chatHandler; namespace ManaServ { extern Connection *chatServerConnection; extern Connection *gameServerConnection; extern std::string netToken; extern ServerInfo chatServer; ChatHandler::ChatHandler() { static const Uint16 _messages[] = { GPMSG_SAY, CPMSG_ENTER_CHANNEL_RESPONSE, CPMSG_LIST_CHANNELS_RESPONSE, CPMSG_PUBMSG, CPMSG_ANNOUNCEMENT, CPMSG_PRIVMSG, CPMSG_QUIT_CHANNEL_RESPONSE, CPMSG_LIST_CHANNELUSERS_RESPONSE, CPMSG_CHANNEL_EVENT, CPMSG_WHO_RESPONSE, CPMSG_DISCONNECT_RESPONSE, 0 }; handledMessages = _messages; chatHandler = this; } void ChatHandler::handleMessage(MessageIn &msg) { switch (msg.getId()) { case GPMSG_SAY: handleGameChatMessage(msg); break; case CPMSG_ENTER_CHANNEL_RESPONSE: handleEnterChannelResponse(msg); break; case CPMSG_LIST_CHANNELS_RESPONSE: handleListChannelsResponse(msg); break; case CPMSG_PRIVMSG: handlePrivateMessage(msg); break; case CPMSG_ANNOUNCEMENT: handleAnnouncement(msg); break; case CPMSG_PUBMSG: handleChatMessage(msg); break; case CPMSG_QUIT_CHANNEL_RESPONSE: handleQuitChannelResponse(msg); break; case CPMSG_LIST_CHANNELUSERS_RESPONSE: handleListChannelUsersResponse(msg); break; case CPMSG_CHANNEL_EVENT: handleChannelEvent(msg); break; case CPMSG_WHO_RESPONSE: handleWhoResponse(msg); break; case CPMSG_DISCONNECT_RESPONSE: { int errMsg = msg.readInt8(); // Successful logout if (errMsg == ERRMSG_OK) { // TODO: Handle logout } else { switch (errMsg) { case ERRMSG_NO_LOGIN: errorMessage = "Chatserver: Not logged in"; break; default: errorMessage = "Chatserver: Unknown error"; break; } Client::setState(STATE_ERROR); } } break; } } void ChatHandler::handleGameChatMessage(MessageIn &msg) { short id = msg.readInt16(); std::string chatMsg = msg.readString(); if (id == 0) { serverNotice(chatMsg); return; } Being *being = actorSpriteManager->findBeing(id); if (!being) { logger->log("Warning: Received GPMSG_SAY for unknown being with id %i." " (Message is: %s)", id, chatMsg.c_str()); return; } std::string mes = being->getName() + " : " + chatMsg; Event event(being == local_player ? Event::Player : Event::Being); event.setString("message", mes); event.setString("text", chatMsg); event.setString("nick", being->getName()); event.setInt("beingId", id); event.setInt("permissions", player_relations .checkPermissionSilently(being->getName(), PlayerRelation::SPEECH_LOG | PlayerRelation::SPEECH_FLOAT)); event.trigger(Event::ChatChannel); } void ChatHandler::handleEnterChannelResponse(MessageIn &msg) { if (msg.readInt8() == ERRMSG_OK) { short channelId = msg.readInt16(); std::string channelName = msg.readString(); std::string announcement = msg.readString(); auto *channel = new Channel(channelId, channelName, announcement); channelManager->addChannel(channel); ChatTab *tab = channel->getTab(); tab->chatLog(strprintf(_("Topic: %s"), announcement.c_str()), BY_CHANNEL); std::string user; std::string userModes; tab->chatLog(_("Players in this channel:"), BY_CHANNEL); while (msg.getUnreadLength()) { user = msg.readString(); if (user.empty()) return; userModes = msg.readString(); if (userModes.find('o') != std::string::npos) { user = "@" + user; } tab->chatLog(user, BY_CHANNEL); } } else { serverNotice(_("Error joining channel.")); } } void ChatHandler::handleListChannelsResponse(MessageIn &msg) { serverNotice(_("Listing channels.")); while (msg.getUnreadLength()) { std::string channelName = msg.readString(); if (channelName.empty()) return; std::ostringstream numUsers; numUsers << msg.readInt16(); channelName += " - "; channelName += numUsers.str(); serverNotice(channelName); } serverNotice(_("End of channel list.")); } void ChatHandler::handlePrivateMessage(MessageIn &msg) { std::string userNick = msg.readString(); std::string chatMsg = msg.readString(); Event event(Event::Whisper); event.setString("nick", userNick); event.setString("message", chatMsg); event.trigger(Event::ChatChannel); } void ChatHandler::handleAnnouncement(MessageIn &msg) { std::string chatMsg = msg.readString(); std::string sender = msg.readString(); Event event(Event::Announcement); event.setString("message", sender + " : " + chatMsg); event.trigger(Event::ChatChannel); } void ChatHandler::handleChatMessage(MessageIn &msg) { short channelId = msg.readInt16(); std::string userNick = msg.readString(); std::string chatMsg = msg.readString(); if (Channel *channel = channelManager->findById(channelId)) { channel->getTab()->chatLog(userNick, chatMsg); } else { // Can't find channel logger->log("Couldn't find chat channel id: %hi", channelId); } } void ChatHandler::handleQuitChannelResponse(MessageIn &msg) { if (msg.readInt8() == ERRMSG_OK) { short channelId = msg.readInt16(); Channel *channel = channelManager->findById(channelId); channelManager->removeChannel(channel); } } void ChatHandler::handleListChannelUsersResponse(MessageIn &msg) { std::string channelName = msg.readString(); std::string userNick; std::string userModes; Channel *channel = channelManager->findByName(channelName); channel->getTab()->chatLog(_("Players in this channel:"), BY_CHANNEL); while (msg.getUnreadLength()) { userNick = msg.readString(); if (userNick.empty()) { break; } userModes = msg.readString(); if (userModes.find('o') != std::string::npos) { userNick = "@" + userNick; } localChatTab->chatLog(userNick, BY_CHANNEL, channel); } } void ChatHandler::handleChannelEvent(MessageIn &msg) { short channelId = msg.readInt16(); char eventId = msg.readInt8(); std::string line = msg.readString(); Channel *channel = channelManager->findById(channelId); if (channel) { switch(eventId) { case CHAT_EVENT_NEW_PLAYER: channel->getTab()->chatLog(strprintf(_("%s entered the " "channel."), line.c_str()), BY_CHANNEL); break; case CHAT_EVENT_LEAVING_PLAYER: channel->getTab()->chatLog(strprintf(_("%s left the channel."), line.c_str()), BY_CHANNEL); break; case CHAT_EVENT_TOPIC_CHANGE: channel->getTab()->chatLog(strprintf(_("Topic: %s"), line.c_str()), BY_CHANNEL); break; case CHAT_EVENT_MODE_CHANGE: { int first = line.find(":"); int second = line.find(":", first+1); std::string user1 = line.substr(0, first); std::string user2 = line.substr(first+1, second); std::string mode = line.substr(second+1, line.length()); channel->getTab()->chatLog(strprintf(_("%s has set mode %s " "on user %s."), user1.c_str(), mode.c_str(), user2.c_str()), BY_CHANNEL); } break; case CHAT_EVENT_KICKED_PLAYER: { int first = line.find(":"); std::string user1 = line.substr(0, first); std::string user2 = line.substr(first+1, line.length()); channel->getTab()->chatLog(strprintf(_("%s has kicked %s."), user1.c_str(), user2.c_str()), BY_CHANNEL); } break; default: channel->getTab()->chatLog(_("Unknown channel event."), BY_CHANNEL); } } } void ChatHandler::handleWhoResponse(MessageIn &msg) { while (msg.getUnreadLength()) { std::string userNick = msg.readString(); if (userNick.empty()) break; serverNotice(userNick); } } void ChatHandler::connect() { MessageOut msg(PCMSG_CONNECT); msg.writeString(netToken, 32); chatServerConnection->send(msg); } bool ChatHandler::isConnected() { return chatServerConnection->isConnected(); } void ChatHandler::disconnect() { chatServerConnection->disconnect(); } void ChatHandler::talk(const std::string &text) { MessageOut msg(PGMSG_SAY); msg.writeString(text); gameServerConnection->send(msg); } void ChatHandler::me(const std::string &text) { // TODO } void ChatHandler::privateMessage(const std::string &recipient, const std::string &text) { MessageOut msg(PCMSG_PRIVMSG); msg.writeString(recipient); msg.writeString(text); chatServerConnection->send(msg); } void ChatHandler::channelList() { MessageOut msg(PCMSG_LIST_CHANNELS); chatServerConnection->send(msg); } void ChatHandler::enterChannel(const std::string &channel, const std::string &password) { MessageOut msg(PCMSG_ENTER_CHANNEL); msg.writeString(channel); msg.writeString(password); chatServerConnection->send(msg); } void ChatHandler::quitChannel(int channelId) { MessageOut msg(PCMSG_QUIT_CHANNEL); msg.writeInt16(channelId); chatServerConnection->send(msg); } void ChatHandler::sendToChannel(int channelId, const std::string &text) { MessageOut msg(PCMSG_CHAT); msg.writeString(text); msg.writeInt16(channelId); chatServerConnection->send(msg); } void ChatHandler::userList(const std::string &channel) { MessageOut msg(PCMSG_LIST_CHANNELUSERS); msg.writeString(channel); chatServerConnection->send(msg); } void ChatHandler::setChannelTopic(int channelId, const std::string &text) { MessageOut msg(PCMSG_TOPIC_CHANGE); msg.writeInt16(channelId); msg.writeString(text); chatServerConnection->send(msg); } void ChatHandler::setUserMode(int channelId, const std::string &name, int mode) { MessageOut msg(PCMSG_USER_MODE); msg.writeInt16(channelId); msg.writeString(name); msg.writeInt8(mode); chatServerConnection->send(msg); } void ChatHandler::kickUser(int channelId, const std::string &name) { MessageOut msg(PCMSG_KICK_USER); msg.writeInt16(channelId); msg.writeString(name); chatServerConnection->send(msg); } void ChatHandler::who() { MessageOut msg(PCMSG_WHO); chatServerConnection->send(msg); } } // namespace ManaServ