summaryrefslogtreecommitdiff
path: root/src/net/eathena
diff options
context:
space:
mode:
Diffstat (limited to 'src/net/eathena')
-rw-r--r--src/net/eathena/charserverhandler.cpp499
-rw-r--r--src/net/eathena/charserverhandler.h47
-rw-r--r--src/net/eathena/charserverrecv.cpp515
-rw-r--r--src/net/eathena/charserverrecv.h61
-rw-r--r--src/net/eathena/network.h4
5 files changed, 610 insertions, 516 deletions
diff --git a/src/net/eathena/charserverhandler.cpp b/src/net/eathena/charserverhandler.cpp
index 960acf44d..83c3c7c4c 100644
--- a/src/net/eathena/charserverhandler.cpp
+++ b/src/net/eathena/charserverhandler.cpp
@@ -38,6 +38,9 @@
#include "net/ea/token.h"
+#include "net/ea/charserverrecv.h"
+
+#include "net/eathena/charserverrecv.h"
#include "net/eathena/gamehandler.h"
#include "net/eathena/loginhandler.h"
#include "net/eathena/messageout.h"
@@ -62,21 +65,15 @@ namespace EAthena
extern ServerInfo charServer;
extern ServerInfo mapServer;
-std::string CharServerHandler::mNewName;
-uint32_t CharServerHandler::mPinSeed = 0;
-BeingId CharServerHandler::mPinAccountId = BeingId_zero;
-BeingId CharServerHandler::mRenameId = BeingId_zero;
-bool CharServerHandler::mNeedCreatePin = false;
-
CharServerHandler::CharServerHandler() :
MessageHandler(),
Ea::CharServerHandler()
{
- mNewName.clear();
- mPinSeed = 0;
- mPinAccountId = BeingId_zero;
- mRenameId = BeingId_zero;
- mNeedCreatePin = false;
+ CharServerRecv::mNewName.clear();
+ CharServerRecv::mPinSeed = 0;
+ CharServerRecv::mPinAccountId = BeingId_zero;
+ CharServerRecv::mRenameId = BeingId_zero;
+ CharServerRecv::mNeedCreatePin = false;
static const uint16_t _messages[] =
{
@@ -110,79 +107,79 @@ void CharServerHandler::handleMessage(Net::MessageIn &msg)
switch (msg.getId())
{
case SMSG_CHAR_LOGIN:
- processCharLogin(msg);
+ CharServerRecv::processCharLogin(msg);
break;
case SMSG_CHAR_LOGIN2:
- processCharLogin2(msg);
+ CharServerRecv::processCharLogin2(msg);
break;
case SMSG_CHAR_LOGIN_ERROR:
- processCharLoginError(msg);
+ Ea::CharServerRecv::processCharLoginError(msg);
break;
case SMSG_CHAR_CREATE_SUCCEEDED:
- processCharCreate(msg);
+ CharServerRecv::processCharCreate(msg);
break;
case SMSG_CHAR_CREATE_FAILED:
- processCharCreateFailed(msg);
+ Ea::CharServerRecv::processCharCreateFailed(msg);
break;
case SMSG_CHAR_DELETE_SUCCEEDED:
- processCharDelete(msg);
+ Ea::CharServerRecv::processCharDelete(msg);
break;
case SMSG_CHAR_DELETE_FAILED:
- processCharDeleteFailed(msg);
+ CharServerRecv::processCharDeleteFailed(msg);
break;
case SMSG_CHAR_MAP_INFO:
- processCharMapInfo(msg);
+ CharServerRecv::processCharMapInfo(msg);
break;
case SMSG_CHANGE_MAP_SERVER:
- processChangeMapServer(msg);
+ CharServerRecv::processChangeMapServer(msg);
break;
case SMSG_CHAR_PINCODE_STATUS:
- processPincodeStatus(msg);
+ CharServerRecv::processPincodeStatus(msg);
break;
case SMSG_CHAR_CHECK_RENAME:
- processCharCheckRename(msg);
+ CharServerRecv::processCharCheckRename(msg);
break;
case SMSG_CHAR_RENAME:
- processCharRename(msg);
+ CharServerRecv::processCharRename(msg);
break;
case SMSG_CHAR_CHANGE_SLOT:
- processCharChangeSlot(msg);
+ CharServerRecv::processCharChangeSlot(msg);
break;
case SMSG_CHAR_CAPTCHA_NOT_SUPPORTED:
- processCharCaptchaNotSupported(msg);
+ CharServerRecv::processCharCaptchaNotSupported(msg);
break;
case SMSG_CHAR_DELETE2_ACK:
- processCharDelete2Ack(msg);
+ CharServerRecv::processCharDelete2Ack(msg);
break;
case SMSG_CHAR_DELETE2_ACCEPT_ACTUAL_ACK:
- processCharDelete2AcceptActual(msg);
+ CharServerRecv::processCharDelete2AcceptActual(msg);
break;
case SMSG_CHAR_DELETE2_CANCEL_ACK:
- processCharDelete2CancelAck(msg);
+ CharServerRecv::processCharDelete2CancelAck(msg);
break;
case SMSG_CHAR_CHARACTERS:
- processCharCharacters(msg);
+ CharServerRecv::processCharCharacters(msg);
break;
case SMSG_CHAR_BAN_CHAR_LIST:
- processCharBanCharList(msg);
+ CharServerRecv::processCharBanCharList(msg);
break;
default:
@@ -190,105 +187,6 @@ void CharServerHandler::handleMessage(Net::MessageIn &msg)
}
}
-void CharServerHandler::readPlayerData(Net::MessageIn &msg,
- Net::Character *const character) const
-{
- if (!character)
- return;
-
- const Token &token =
- static_cast<LoginHandler*>(loginHandler)->getToken();
-
- LocalPlayer *const tempPlayer = new LocalPlayer(
- msg.readBeingId("player id"), BeingTypeId_zero);
- tempPlayer->setGender(token.sex);
-
- PlayerInfoBackend &data = character->data;
- data.mAttributes[Attributes::EXP] = msg.readInt32("exp");
- data.mAttributes[Attributes::MONEY] = msg.readInt32("money");
- Stat &jobStat = data.mStats[Attributes::JOB];
- jobStat.exp = msg.readInt32("job");
-
- const int temp = msg.readInt32("job level");
- jobStat.base = temp;
- jobStat.mod = temp;
-
- msg.readInt16("shoes?");
- const int gloves = msg.readInt16("gloves");
- const int cape = msg.readInt16("cape");
- const int misc1 = msg.readInt16("misc1");
-
- msg.readInt32("option");
- tempPlayer->setKarma(msg.readInt32("karma"));
- tempPlayer->setManner(msg.readInt32("manner"));
- msg.readInt16("left points");
-
- data.mAttributes[Attributes::HP] = msg.readInt32("hp");
- data.mAttributes[Attributes::MAX_HP] = msg.readInt32("max hp");
- data.mAttributes[Attributes::MP] = msg.readInt16("mp/sp");
- data.mAttributes[Attributes::MAX_MP] = msg.readInt16("max mp/sp");
-
- msg.readInt16("speed");
- const uint16_t race = msg.readInt16("class");
-// tempPlayer->setSubtype(race, 0);
- const int hairStyle = msg.readInt32("hair style");
- const int option A_UNUSED = (msg.readInt16("weapon") | 1) ^ 1;
- const int weapon = 0;
-
- tempPlayer->setSprite(SPRITE_BODY, weapon, "", ItemColor_one, true);
-
- data.mAttributes[Attributes::LEVEL] = msg.readInt16("level");
-
- msg.readInt16("skill points");
- const int bottomClothes = msg.readInt16("head bottom");
- const int shield = msg.readInt16("shild");
- const int hat = msg.readInt16("head top");
- const int topClothes = msg.readInt16("head mid");
-
- const ItemColor color = fromInt(msg.readInt16("hair color"), ItemColor);
- tempPlayer->setHairColor(color);
- tempPlayer->setSprite(SPRITE_HAIR_COLOR, hairStyle * -1,
- ItemDB::get(-hairStyle).getDyeColorsString(
- color));
-
- const uint16_t look = msg.readInt16("clothes color");
- tempPlayer->setSubtype(fromInt(race, BeingTypeId), look);
- tempPlayer->setName(msg.readString(24, "name"));
-
- character->dummy = tempPlayer;
-
- character->data.mStats[Attributes::STR].base = msg.readUInt8("str");
- character->data.mStats[Attributes::AGI].base = msg.readUInt8("agi");
- character->data.mStats[Attributes::VIT].base = msg.readUInt8("vit");
- character->data.mStats[Attributes::INT].base = msg.readUInt8("int");
- character->data.mStats[Attributes::DEX].base = msg.readUInt8("dex");
- character->data.mStats[Attributes::LUK].base = msg.readUInt8("luk");
-
- character->slot = msg.readInt16("character slot id");
- msg.readInt16("rename");
- msg.readString(16, "map name");
- msg.readInt32("delete date");
- const int shoes = msg.readInt32("robe");
- if (!serverFeatures->haveAdvancedSprites())
- {
- tempPlayer->setSprite(SPRITE_HAIR, shoes);
- tempPlayer->setSprite(SPRITE_SHOES, gloves);
- tempPlayer->setSprite(SPRITE_SHIELD, cape);
- tempPlayer->setSprite(SPRITE_HEAD_TOP, misc1);
- tempPlayer->setSprite(SPRITE_WEAPON, bottomClothes);
- tempPlayer->setSprite(SPRITE_FLOOR, shield);
- tempPlayer->setSprite(SPRITE_CLOTHES_COLOR, hat);
- tempPlayer->setSprite(SPRITE_HEAD_BOTTOM, topClothes);
-// tempPlayer->setSprite(SPRITE_HEAD_MID, misc2);
- }
- msg.readInt32("slot change");
- tempPlayer->setRename(msg.readInt32("rename (inverse)"));
-
- const uint8_t gender = static_cast<uint8_t>(msg.readUInt8("gender"));
- if (gender != 99)
- tempPlayer->setGender(Being::intToGender(gender));
-}
-
void CharServerHandler::chooseCharacter(Net::Character *const character)
{
if (!character)
@@ -390,154 +288,6 @@ void CharServerHandler::setCharCreateDialog(CharCreateDialog *const window)
mCharCreateDialog->setDefaultGender(token.sex);
}
-void CharServerHandler::processCharLogin(Net::MessageIn &msg)
-{
- msg.skip(2, "packet len");
- const int slots = msg.readInt8("MAX_CHARS");
- msg.readInt8("sd->char_slots");
- msg.readInt8("MAX_CHARS");
- loginData.characterSlots = static_cast<uint16_t>(slots);
-
- msg.skip(20, "unused 0");
-
- delete_all(mCharacters);
- mCharacters.clear();
-
- // Derive number of characters from message length
- const int count = (msg.getLength() - 27)
- / (106 + 4 + 2 + 16 + 4 + 4 + 4 + 4);
-
- for (int i = 0; i < count; ++i)
- {
- Net::Character *const character = new Net::Character;
- readPlayerData(msg, character);
- mCharacters.push_back(character);
- if (character->dummy)
- {
- logger->log("CharServer: Player: %s (%d)",
- character->dummy->getName().c_str(), character->slot);
- }
- }
-
- client->setState(STATE_CHAR_SELECT);
-}
-
-void CharServerHandler::processCharLogin2(Net::MessageIn &msg)
-{
- // ignored
- msg.readInt16("len");
- msg.readUInt8("char slots");
- msg.readUInt8("left slots");
- msg.readUInt8("left slots");
- msg.readUInt8("char slots");
- msg.readUInt8("char slots");
- msg.skip(20, "unused");
-}
-
-void CharServerHandler::processCharMapInfo(Net::MessageIn &restrict msg)
-{
- Network *const network = mNetwork;
- ServerInfo &server = mapServer;
- BLOCK_START("CharServerHandler::processCharMapInfo")
- PlayerInfo::setCharId(msg.readInt32("char id"));
- GameHandler *const gh = static_cast<GameHandler*>(gameHandler);
- gh->setMap(msg.readString(16, "map name"));
- if (config.getBoolValue("usePersistentIP") || settings.persistentIp)
- {
- msg.readInt32("map ip address");
- server.hostname = settings.serverName;
- }
- else
- {
- server.hostname = ipToString(msg.readInt32("map ip address"));
- }
- server.port = msg.readInt16("map ip port");
-
- // Prevent the selected local player from being deleted
- localPlayer = mSelectedCharacter->dummy;
- PlayerInfo::setBackend(mSelectedCharacter->data);
- PlayerInfo::setStatBase(Attributes::WALK_SPEED,
- playerHandler->getDefaultWalkSpeed().x);
-
- mSelectedCharacter->dummy = nullptr;
-
- charServerHandler->clear();
- updateCharSelectDialog();
-
- if (network)
- network->disconnect();
- client->setState(STATE_CONNECT_GAME);
- BLOCK_END("CharServerHandler::processCharMapInfo")
-}
-
-void CharServerHandler::processChangeMapServer(Net::MessageIn &msg)
-{
- Network *const network = mNetwork;
- ServerInfo &server = mapServer;
- BLOCK_START("CharServerHandler::processChangeMapServer")
- GameHandler *const gh = static_cast<GameHandler*>(gameHandler);
- if (!gh || !network)
- {
- BLOCK_END("CharServerHandler::processChangeMapServer")
- return;
- }
- gh->setMap(msg.readString(16, "map name"));
- const int x = msg.readInt16("x");
- const int y = msg.readInt16("y");
- if (config.getBoolValue("usePersistentIP") || settings.persistentIp)
- {
- msg.readInt32("host");
- server.hostname = settings.serverName;
- }
- else
- {
- server.hostname = ipToString(msg.readInt32("host"));
- }
- server.port = msg.readInt16("port");
-
- network->disconnect();
- client->setState(STATE_CHANGE_MAP);
- if (localPlayer)
- {
- localPlayer->setTileCoords(x, y);
- localPlayer->setMap(nullptr);
- }
- BLOCK_END("CharServerHandler::processChangeMapServer")
-}
-
-void CharServerHandler::processPincodeStatus(Net::MessageIn &msg)
-{
- mPinSeed = msg.readInt32("pincode seed");
- mPinAccountId = msg.readBeingId("account id");
- const uint16_t state = static_cast<uint16_t>(msg.readInt16("state"));
- switch (state)
- {
- case 0: // pin ok
- break;
- case 1: // ask for pin
- break;
- case 2: // create new pin
- case 4: // create new pin?
- {
- mNeedCreatePin = true;
- break;
- }
- case 3: // pin must be changed
- break;
- case 5: // client show error?
- break;
- case 6: // Unable to use your KSSN number
- break;
- case 7: // char select window shows a button
- break;
- case 8: // pincode was incorrect
- break;
- default:
- UNIMPLIMENTEDPACKET;
- break;
- }
-}
-
void CharServerHandler::setNewPincode(const std::string &pin A_UNUSED)
{
// here need ecript pin with mPinSeed and pin values.
@@ -547,114 +297,16 @@ void CharServerHandler::setNewPincode(const std::string &pin A_UNUSED)
// outMsg.writeString(pin, 4, "encrypted pin");
}
-void CharServerHandler::processCharCreate(Net::MessageIn &msg)
-{
- BLOCK_START("CharServerHandler::processCharCreate")
- Net::Character *const character = new Net::Character;
- charServerHandler->readPlayerData(msg, character);
- mCharacters.push_back(character);
-
- updateCharSelectDialog();
-
- // Close the character create dialog
- if (mCharCreateDialog)
- {
- mCharCreateDialog->scheduleDelete();
- mCharCreateDialog = nullptr;
- }
- BLOCK_END("CharServerHandler::processCharCreate")
-}
-
void CharServerHandler::renameCharacter(const BeingId id,
const std::string &newName)
{
createOutPacket(CMSG_CHAR_CHECK_RENAME);
- mRenameId = id;
- mNewName = newName;
+ CharServerRecv::mRenameId = id;
+ CharServerRecv::mNewName = newName;
outMsg.writeBeingId(id, "char id");
outMsg.writeString(newName, 24, "name");
}
-void CharServerHandler::processCharCheckRename(Net::MessageIn &msg)
-{
- if (msg.readInt16("flag"))
- {
- createOutPacket(CMSG_CHAR_RENAME);
- outMsg.writeBeingId(mRenameId, "char id");
- }
- else
- {
- CREATEWIDGET(OkDialog,
- // TRANSLATORS: error header
- _("Error"),
- // TRANSLATORS: error message
- _("Character rename error."),
- // TRANSLATORS: ok dialog button
- _("Error"),
- DialogType::ERROR,
- Modal_true,
- ShowCenter_true,
- nullptr,
- 260);
- }
-}
-
-void CharServerHandler::processCharRename(Net::MessageIn &msg)
-{
- const int flag = msg.readInt16("flag");
- if (!flag)
- {
- mCharSelectDialog->setName(mRenameId, mNewName);
- CREATEWIDGET(OkDialog,
- // TRANSLATORS: info header
- _("Info"),
- // TRANSLATORS: info message
- _("Character renamed."),
- // TRANSLATORS: ok dialog button
- _("OK"),
- DialogType::OK,
- Modal_true,
- ShowCenter_true,
- nullptr,
- 260);
- }
- else
- {
- std::string message;
- switch (flag)
- {
- case 1:
- // TRANSLATORS: char rename error
- message = _("Rename not allowed.");
- break;
- case 2:
- // TRANSLATORS: char rename error
- message = _("New name is not set.");
- break;
- case 3:
- default:
- // TRANSLATORS: char rename error
- message = _("Character rename error.");
- break;
- case 4:
- // TRANSLATORS: char rename error
- message = _("Character not found.");
- break;
- }
- CREATEWIDGET(OkDialog,
- // TRANSLATORS: info message
- _("Info"),
- message,
- // TRANSLATORS: ok dialog button
- _("OK"),
- DialogType::OK,
- Modal_true,
- ShowCenter_true,
- nullptr,
- 260);
- }
-}
-
void CharServerHandler::changeSlot(const int oldSlot, const int newSlot)
{
createOutPacket(CMSG_CHAR_CHANGE_SLOT);
@@ -663,89 +315,6 @@ void CharServerHandler::changeSlot(const int oldSlot, const int newSlot)
outMsg.writeInt16(0, "unused");
}
-void CharServerHandler::processCharChangeSlot(Net::MessageIn &msg)
-{
- UNIMPLIMENTEDPACKET;
- msg.readInt16("len");
- msg.readInt16("flag"); // 0 - ok, 1 - error
- msg.readInt16("unused");
-}
-
-void CharServerHandler::processCharDeleteFailed(Net::MessageIn &msg)
-{
- BLOCK_START("CharServerHandler::processCharDeleteFailed")
- unlockCharSelectDialog();
- msg.readUInt8("error");
- CREATEWIDGET(OkDialog,
- // TRANSLATORS: error header
- _("Error"),
- // TRANSLATORS: error message
- _("Failed to delete character."),
- // TRANSLATORS: ok dialog button
- _("OK"),
- DialogType::ERROR,
- Modal_true,
- ShowCenter_true,
- nullptr,
- 260);
- BLOCK_END("CharServerHandler::processCharDeleteFailed")
-}
-
-void CharServerHandler::processCharCaptchaNotSupported(Net::MessageIn &msg)
-{
- UNIMPLIMENTEDPACKET;
- msg.readInt16("5");
- msg.readUInt8("1");
-}
-
-void CharServerHandler::processCharDelete2Ack(Net::MessageIn &msg)
-{
- UNIMPLIMENTEDPACKET;
- msg.readInt32("char id");
- msg.readInt32("result");
- msg.readInt32("time");
-}
-
-void CharServerHandler::processCharDelete2AcceptActual(Net::MessageIn &msg)
-{
- UNIMPLIMENTEDPACKET;
- msg.readInt32("char id");
- msg.readInt32("result");
-}
-
-void CharServerHandler::processCharDelete2CancelAck(Net::MessageIn &msg)
-{
- UNIMPLIMENTEDPACKET;
- msg.readInt32("char id");
- msg.readInt32("result");
-}
-
-void CharServerHandler::processCharCharacters(Net::MessageIn &msg)
-{
- msg.skip(2, "packet len");
-
- delete_all(mCharacters);
- mCharacters.clear();
-
- // Derive number of characters from message length
- const int count = (msg.getLength() - 4)
- / (106 + 4 + 2 + 16 + 4 + 4 + 4 + 4);
-
- for (int i = 0; i < count; ++i)
- {
- Net::Character *const character = new Net::Character;
- charServerHandler->readPlayerData(msg, character);
- mCharacters.push_back(character);
- if (character->dummy)
- {
- logger->log("CharServer: Player: %s (%d)",
- character->dummy->getName().c_str(), character->slot);
- }
- }
-
- client->setState(STATE_CHAR_SELECT);
-}
-
void CharServerHandler::ping() const
{
createOutPacket(CMSG_CHAR_PING);
@@ -757,15 +326,9 @@ unsigned int CharServerHandler::hatSprite() const
return 7;
}
-void CharServerHandler::processCharBanCharList(Net::MessageIn &msg)
+bool CharServerHandler::isNeedCreatePin() const
{
- UNIMPLIMENTEDPACKET;
- const int count = (msg.readInt16("len") - 4) / 24;
- for (int f = 0; f < count; f ++)
- {
- msg.readInt32("char id");
- msg.readString(20, "unbun time");
- }
+ return CharServerRecv::mNeedCreatePin;
}
} // namespace EAthena
diff --git a/src/net/eathena/charserverhandler.h b/src/net/eathena/charserverhandler.h
index 1b0b602ae..20086c965 100644
--- a/src/net/eathena/charserverhandler.h
+++ b/src/net/eathena/charserverhandler.h
@@ -64,14 +64,7 @@ class CharServerHandler final : public MessageHandler,
void connect() override final;
- void processCharLogin(Net::MessageIn &msg);
-
- void processCharMapInfo(Net::MessageIn &msg);
-
- void processChangeMapServer(Net::MessageIn &msg);
-
- bool isNeedCreatePin() const override final A_WARN_UNUSED
- { return mNeedCreatePin; }
+ bool isNeedCreatePin() const override final A_WARN_UNUSED;
void setNewPincode(const std::string &pin) override final;
@@ -85,47 +78,9 @@ class CharServerHandler final : public MessageHandler,
void changeSlot(const int oldSlot, const int newSlot) override final;
- void readPlayerData(Net::MessageIn &msg,
- Net::Character *const character)
- const override final;
-
void ping() const override final;
unsigned int hatSprite() const override final A_WARN_UNUSED;
-
- protected:
- static void processPincodeStatus(Net::MessageIn &msg);
-
- static void processCharLogin2(Net::MessageIn &msg);
-
- static void processCharCreate(Net::MessageIn &msg);
-
- static void processCharCheckRename(Net::MessageIn &msg);
-
- static void processCharRename(Net::MessageIn &msg);
-
- static void processCharChangeSlot(Net::MessageIn &msg);
-
- static void processCharDeleteFailed(Net::MessageIn &msg);
-
- static void processCharCaptchaNotSupported(Net::MessageIn &msg);
-
- static void processCharDelete2Ack(Net::MessageIn &msg);
-
- static void processCharDelete2AcceptActual(Net::MessageIn &msg);
-
- static void processCharDelete2CancelAck(Net::MessageIn &msg);
-
- static void processCharCharacters(Net::MessageIn &msg);
-
- static void processCharBanCharList(Net::MessageIn &msg);
-
- private:
- static std::string mNewName;
- static uint32_t mPinSeed;
- static BeingId mPinAccountId;
- static BeingId mRenameId;
- static bool mNeedCreatePin;
};
} // namespace EAthena
diff --git a/src/net/eathena/charserverrecv.cpp b/src/net/eathena/charserverrecv.cpp
new file mode 100644
index 000000000..62be4a694
--- /dev/null
+++ b/src/net/eathena/charserverrecv.cpp
@@ -0,0 +1,515 @@
+/*
+ * The ManaPlus Client
+ * Copyright (C) 2004-2009 The Mana World Development Team
+ * Copyright (C) 2009-2010 The Mana Developers
+ * Copyright (C) 2011-2015 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/eathena/charserverrecv.h"
+
+#include "client.h"
+#include "configuration.h"
+#include "settings.h"
+
+#include "gui/windows/charcreatedialog.h"
+#include "gui/windows/charselectdialog.h"
+#include "gui/windows/okdialog.h"
+
+#include "gui/widgets/createwidget.h"
+
+#include "net/character.h"
+#include "net/playerhandler.h"
+#include "net/serverfeatures.h"
+
+#include "net/ea/token.h"
+
+#include "net/eathena/gamehandler.h"
+#include "net/eathena/loginhandler.h"
+#include "net/eathena/messageout.h"
+#include "net/eathena/network.h"
+#include "net/eathena/protocol.h"
+#include "net/eathena/sprite.h"
+
+#include "resources/iteminfo.h"
+
+#include "resources/db/itemdb.h"
+
+#include "utils/dtor.h"
+#include "utils/gettext.h"
+
+#include "debug.h"
+
+extern Net::CharServerHandler *charServerHandler;
+
+namespace EAthena
+{
+
+extern ServerInfo charServer;
+extern ServerInfo mapServer;
+
+namespace CharServerRecv
+{
+ std::string mNewName;
+ uint32_t mPinSeed = 0;
+ BeingId mPinAccountId = BeingId_zero;
+ BeingId mRenameId = BeingId_zero;
+ bool mNeedCreatePin = false;
+}
+
+void CharServerRecv::readPlayerData(Net::MessageIn &msg,
+ Net::Character *const character)
+{
+ if (!character)
+ return;
+
+ const Token &token =
+ static_cast<LoginHandler*>(loginHandler)->getToken();
+
+ LocalPlayer *const tempPlayer = new LocalPlayer(
+ msg.readBeingId("player id"), BeingTypeId_zero);
+ tempPlayer->setGender(token.sex);
+
+ PlayerInfoBackend &data = character->data;
+ data.mAttributes[Attributes::EXP] = msg.readInt32("exp");
+ data.mAttributes[Attributes::MONEY] = msg.readInt32("money");
+ Stat &jobStat = data.mStats[Attributes::JOB];
+ jobStat.exp = msg.readInt32("job");
+
+ const int temp = msg.readInt32("job level");
+ jobStat.base = temp;
+ jobStat.mod = temp;
+
+ msg.readInt16("shoes?");
+ const int gloves = msg.readInt16("gloves");
+ const int cape = msg.readInt16("cape");
+ const int misc1 = msg.readInt16("misc1");
+
+ msg.readInt32("option");
+ tempPlayer->setKarma(msg.readInt32("karma"));
+ tempPlayer->setManner(msg.readInt32("manner"));
+ msg.readInt16("left points");
+
+ data.mAttributes[Attributes::HP] = msg.readInt32("hp");
+ data.mAttributes[Attributes::MAX_HP] = msg.readInt32("max hp");
+ data.mAttributes[Attributes::MP] = msg.readInt16("mp/sp");
+ data.mAttributes[Attributes::MAX_MP] = msg.readInt16("max mp/sp");
+
+ msg.readInt16("speed");
+ const uint16_t race = msg.readInt16("class");
+// tempPlayer->setSubtype(race, 0);
+ const int hairStyle = msg.readInt32("hair style");
+ const int option A_UNUSED = (msg.readInt16("weapon") | 1) ^ 1;
+ const int weapon = 0;
+
+ tempPlayer->setSprite(SPRITE_BODY, weapon, "", ItemColor_one, true);
+
+ data.mAttributes[Attributes::LEVEL] = msg.readInt16("level");
+
+ msg.readInt16("skill points");
+ const int bottomClothes = msg.readInt16("head bottom");
+ const int shield = msg.readInt16("shild");
+ const int hat = msg.readInt16("head top");
+ const int topClothes = msg.readInt16("head mid");
+
+ const ItemColor color = fromInt(msg.readInt16("hair color"), ItemColor);
+ tempPlayer->setHairColor(color);
+ tempPlayer->setSprite(SPRITE_HAIR_COLOR, hairStyle * -1,
+ ItemDB::get(-hairStyle).getDyeColorsString(
+ color));
+
+ const uint16_t look = msg.readInt16("clothes color");
+ tempPlayer->setSubtype(fromInt(race, BeingTypeId), look);
+ tempPlayer->setName(msg.readString(24, "name"));
+
+ character->dummy = tempPlayer;
+
+ character->data.mStats[Attributes::STR].base = msg.readUInt8("str");
+ character->data.mStats[Attributes::AGI].base = msg.readUInt8("agi");
+ character->data.mStats[Attributes::VIT].base = msg.readUInt8("vit");
+ character->data.mStats[Attributes::INT].base = msg.readUInt8("int");
+ character->data.mStats[Attributes::DEX].base = msg.readUInt8("dex");
+ character->data.mStats[Attributes::LUK].base = msg.readUInt8("luk");
+
+ character->slot = msg.readInt16("character slot id");
+ msg.readInt16("rename");
+ msg.readString(16, "map name");
+ msg.readInt32("delete date");
+ const int shoes = msg.readInt32("robe");
+ if (!serverFeatures->haveAdvancedSprites())
+ {
+ tempPlayer->setSprite(SPRITE_HAIR, shoes);
+ tempPlayer->setSprite(SPRITE_SHOES, gloves);
+ tempPlayer->setSprite(SPRITE_SHIELD, cape);
+ tempPlayer->setSprite(SPRITE_HEAD_TOP, misc1);
+ tempPlayer->setSprite(SPRITE_WEAPON, bottomClothes);
+ tempPlayer->setSprite(SPRITE_FLOOR, shield);
+ tempPlayer->setSprite(SPRITE_CLOTHES_COLOR, hat);
+ tempPlayer->setSprite(SPRITE_HEAD_BOTTOM, topClothes);
+// tempPlayer->setSprite(SPRITE_HEAD_MID, misc2);
+ }
+ msg.readInt32("slot change");
+ tempPlayer->setRename(msg.readInt32("rename (inverse)"));
+
+ const uint8_t gender = static_cast<uint8_t>(msg.readUInt8("gender"));
+ if (gender != 99)
+ tempPlayer->setGender(Being::intToGender(gender));
+}
+
+void CharServerRecv::processCharLogin(Net::MessageIn &msg)
+{
+ msg.skip(2, "packet len");
+ const int slots = msg.readInt8("MAX_CHARS");
+ msg.readInt8("sd->char_slots");
+ msg.readInt8("MAX_CHARS");
+ loginData.characterSlots = static_cast<uint16_t>(slots);
+
+ msg.skip(20, "unused 0");
+
+ delete_all(charServerHandler->mCharacters);
+ charServerHandler->mCharacters.clear();
+
+ // Derive number of characters from message length
+ const int count = (msg.getLength() - 27)
+ / (106 + 4 + 2 + 16 + 4 + 4 + 4 + 4);
+
+ for (int i = 0; i < count; ++i)
+ {
+ Net::Character *const character = new Net::Character;
+ readPlayerData(msg, character);
+ charServerHandler->mCharacters.push_back(character);
+ if (character->dummy)
+ {
+ logger->log("CharServer: Player: %s (%d)",
+ character->dummy->getName().c_str(), character->slot);
+ }
+ }
+
+ client->setState(STATE_CHAR_SELECT);
+}
+
+void CharServerRecv::processCharLogin2(Net::MessageIn &msg)
+{
+ // ignored
+ msg.readInt16("len");
+ msg.readUInt8("char slots");
+ msg.readUInt8("left slots");
+ msg.readUInt8("left slots");
+ msg.readUInt8("char slots");
+ msg.readUInt8("char slots");
+ msg.skip(20, "unused");
+}
+
+void CharServerRecv::processCharMapInfo(Net::MessageIn &restrict msg)
+{
+ Network *const network = Network::mInstance;
+ ServerInfo &server = mapServer;
+ BLOCK_START("CharServerRecv::processCharMapInfo")
+ PlayerInfo::setCharId(msg.readInt32("char id"));
+ GameHandler *const gh = static_cast<GameHandler*>(gameHandler);
+ gh->setMap(msg.readString(16, "map name"));
+ if (config.getBoolValue("usePersistentIP") || settings.persistentIp)
+ {
+ msg.readInt32("map ip address");
+ server.hostname = settings.serverName;
+ }
+ else
+ {
+ server.hostname = ipToString(msg.readInt32("map ip address"));
+ }
+ server.port = msg.readInt16("map ip port");
+
+ // Prevent the selected local player from being deleted
+ localPlayer = charServerHandler->mSelectedCharacter->dummy;
+ PlayerInfo::setBackend(charServerHandler->mSelectedCharacter->data);
+ PlayerInfo::setStatBase(Attributes::WALK_SPEED,
+ playerHandler->getDefaultWalkSpeed().x);
+
+ charServerHandler->mSelectedCharacter->dummy = nullptr;
+
+ charServerHandler->clear();
+ charServerHandler->updateCharSelectDialog();
+
+ if (network)
+ network->disconnect();
+ client->setState(STATE_CONNECT_GAME);
+ BLOCK_END("CharServerRecv::processCharMapInfo")
+}
+
+void CharServerRecv::processChangeMapServer(Net::MessageIn &msg)
+{
+ Network *const network = Network::mInstance;
+ ServerInfo &server = mapServer;
+ BLOCK_START("CharServerRecv::processChangeMapServer")
+ GameHandler *const gh = static_cast<GameHandler*>(gameHandler);
+ if (!gh || !network)
+ {
+ BLOCK_END("CharServerRecv::processChangeMapServer")
+ return;
+ }
+ gh->setMap(msg.readString(16, "map name"));
+ const int x = msg.readInt16("x");
+ const int y = msg.readInt16("y");
+ if (config.getBoolValue("usePersistentIP") || settings.persistentIp)
+ {
+ msg.readInt32("host");
+ server.hostname = settings.serverName;
+ }
+ else
+ {
+ server.hostname = ipToString(msg.readInt32("host"));
+ }
+ server.port = msg.readInt16("port");
+
+ network->disconnect();
+ client->setState(STATE_CHANGE_MAP);
+ if (localPlayer)
+ {
+ localPlayer->setTileCoords(x, y);
+ localPlayer->setMap(nullptr);
+ }
+ BLOCK_END("CharServerRecv::processChangeMapServer")
+}
+
+void CharServerRecv::processPincodeStatus(Net::MessageIn &msg)
+{
+ mPinSeed = msg.readInt32("pincode seed");
+ mPinAccountId = msg.readBeingId("account id");
+ const uint16_t state = static_cast<uint16_t>(msg.readInt16("state"));
+ switch (state)
+ {
+ case 0: // pin ok
+ break;
+ case 1: // ask for pin
+ break;
+ case 2: // create new pin
+ case 4: // create new pin?
+ {
+ mNeedCreatePin = true;
+ break;
+ }
+ case 3: // pin must be changed
+ break;
+ case 5: // client show error?
+ break;
+ case 6: // Unable to use your KSSN number
+ break;
+ case 7: // char select window shows a button
+ break;
+ case 8: // pincode was incorrect
+ break;
+ default:
+ UNIMPLIMENTEDPACKET;
+ break;
+ }
+}
+
+void CharServerRecv::processCharCreate(Net::MessageIn &msg)
+{
+ BLOCK_START("CharServerRecv::processCharCreate")
+ Net::Character *const character = new Net::Character;
+ readPlayerData(msg, character);
+ charServerHandler->mCharacters.push_back(character);
+
+ charServerHandler->updateCharSelectDialog();
+
+ // Close the character create dialog
+ if (charServerHandler->mCharCreateDialog)
+ {
+ charServerHandler->mCharCreateDialog->scheduleDelete();
+ charServerHandler->mCharCreateDialog = nullptr;
+ }
+ BLOCK_END("CharServerRecv::processCharCreate")
+}
+
+void CharServerRecv::processCharCheckRename(Net::MessageIn &msg)
+{
+ if (msg.readInt16("flag"))
+ {
+ createOutPacket(CMSG_CHAR_RENAME);
+ outMsg.writeBeingId(mRenameId, "char id");
+ }
+ else
+ {
+ CREATEWIDGET(OkDialog,
+ // TRANSLATORS: error header
+ _("Error"),
+ // TRANSLATORS: error message
+ _("Character rename error."),
+ // TRANSLATORS: ok dialog button
+ _("Error"),
+ DialogType::ERROR,
+ Modal_true,
+ ShowCenter_true,
+ nullptr,
+ 260);
+ }
+}
+
+void CharServerRecv::processCharRename(Net::MessageIn &msg)
+{
+ const int flag = msg.readInt16("flag");
+ if (!flag)
+ {
+ charServerHandler->mCharSelectDialog->setName(
+ mRenameId,
+ mNewName);
+ CREATEWIDGET(OkDialog,
+ // TRANSLATORS: info header
+ _("Info"),
+ // TRANSLATORS: info message
+ _("Character renamed."),
+ // TRANSLATORS: ok dialog button
+ _("OK"),
+ DialogType::OK,
+ Modal_true,
+ ShowCenter_true,
+ nullptr,
+ 260);
+ }
+ else
+ {
+ std::string message;
+ switch (flag)
+ {
+ case 1:
+ // TRANSLATORS: char rename error
+ message = _("Rename not allowed.");
+ break;
+ case 2:
+ // TRANSLATORS: char rename error
+ message = _("New name is not set.");
+ break;
+ case 3:
+ default:
+ // TRANSLATORS: char rename error
+ message = _("Character rename error.");
+ break;
+ case 4:
+ // TRANSLATORS: char rename error
+ message = _("Character not found.");
+ break;
+ }
+ CREATEWIDGET(OkDialog,
+ // TRANSLATORS: info message
+ _("Info"),
+ message,
+ // TRANSLATORS: ok dialog button
+ _("OK"),
+ DialogType::OK,
+ Modal_true,
+ ShowCenter_true,
+ nullptr,
+ 260);
+ }
+}
+
+void CharServerRecv::processCharChangeSlot(Net::MessageIn &msg)
+{
+ UNIMPLIMENTEDPACKET;
+ msg.readInt16("len");
+ msg.readInt16("flag"); // 0 - ok, 1 - error
+ msg.readInt16("unused");
+}
+
+void CharServerRecv::processCharDeleteFailed(Net::MessageIn &msg)
+{
+ BLOCK_START("CharServerRecv::processCharDeleteFailed")
+ charServerHandler->unlockCharSelectDialog();
+ msg.readUInt8("error");
+ CREATEWIDGET(OkDialog,
+ // TRANSLATORS: error header
+ _("Error"),
+ // TRANSLATORS: error message
+ _("Failed to delete character."),
+ // TRANSLATORS: ok dialog button
+ _("OK"),
+ DialogType::ERROR,
+ Modal_true,
+ ShowCenter_true,
+ nullptr,
+ 260);
+ BLOCK_END("CharServerRecv::processCharDeleteFailed")
+}
+
+void CharServerRecv::processCharCaptchaNotSupported(Net::MessageIn &msg)
+{
+ UNIMPLIMENTEDPACKET;
+ msg.readInt16("5");
+ msg.readUInt8("1");
+}
+
+void CharServerRecv::processCharDelete2Ack(Net::MessageIn &msg)
+{
+ UNIMPLIMENTEDPACKET;
+ msg.readInt32("char id");
+ msg.readInt32("result");
+ msg.readInt32("time");
+}
+
+void CharServerRecv::processCharDelete2AcceptActual(Net::MessageIn &msg)
+{
+ UNIMPLIMENTEDPACKET;
+ msg.readInt32("char id");
+ msg.readInt32("result");
+}
+
+void CharServerRecv::processCharDelete2CancelAck(Net::MessageIn &msg)
+{
+ UNIMPLIMENTEDPACKET;
+ msg.readInt32("char id");
+ msg.readInt32("result");
+}
+
+void CharServerRecv::processCharCharacters(Net::MessageIn &msg)
+{
+ msg.skip(2, "packet len");
+
+ delete_all(charServerHandler->mCharacters);
+ charServerHandler->mCharacters.clear();
+
+ // Derive number of characters from message length
+ const int count = (msg.getLength() - 4)
+ / (106 + 4 + 2 + 16 + 4 + 4 + 4 + 4);
+
+ for (int i = 0; i < count; ++i)
+ {
+ Net::Character *const character = new Net::Character;
+ readPlayerData(msg, character);
+ charServerHandler->mCharacters.push_back(character);
+ if (character->dummy)
+ {
+ logger->log("CharServer: Player: %s (%d)",
+ character->dummy->getName().c_str(), character->slot);
+ }
+ }
+
+ client->setState(STATE_CHAR_SELECT);
+}
+
+void CharServerRecv::processCharBanCharList(Net::MessageIn &msg)
+{
+ UNIMPLIMENTEDPACKET;
+ const int count = (msg.readInt16("len") - 4) / 24;
+ for (int f = 0; f < count; f ++)
+ {
+ msg.readInt32("char id");
+ msg.readString(20, "unbun time");
+ }
+}
+
+} // namespace EAthena
diff --git a/src/net/eathena/charserverrecv.h b/src/net/eathena/charserverrecv.h
new file mode 100644
index 000000000..1df9a7e5e
--- /dev/null
+++ b/src/net/eathena/charserverrecv.h
@@ -0,0 +1,61 @@
+/*
+ * The ManaPlus Client
+ * Copyright (C) 2004-2009 The Mana World Development Team
+ * Copyright (C) 2009-2010 The Mana Developers
+ * Copyright (C) 2011-2015 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/>.
+ */
+
+#ifndef NET_EATHENA_CHARSERVERRECV_H
+#define NET_EATHENA_CHARSERVERRECV_H
+
+#include "net/ea/charserverhandler.h"
+
+#include "net/eathena/messagehandler.h"
+
+namespace EAthena
+{
+ namespace CharServerRecv
+ {
+ extern std::string mNewName;
+ extern uint32_t mPinSeed;
+ extern BeingId mPinAccountId;
+ extern BeingId mRenameId;
+ extern bool mNeedCreatePin;
+
+ void readPlayerData(Net::MessageIn &msg,
+ Net::Character *const character);
+ void processPincodeStatus(Net::MessageIn &msg);
+ void processCharLogin2(Net::MessageIn &msg);
+ void processCharCreate(Net::MessageIn &msg);
+ void processCharCheckRename(Net::MessageIn &msg);
+ void processCharRename(Net::MessageIn &msg);
+ void processCharChangeSlot(Net::MessageIn &msg);
+ void processCharDeleteFailed(Net::MessageIn &msg);
+ void processCharCaptchaNotSupported(Net::MessageIn &msg);
+ void processCharDelete2Ack(Net::MessageIn &msg);
+ void processCharDelete2AcceptActual(Net::MessageIn &msg);
+ void processCharDelete2CancelAck(Net::MessageIn &msg);
+ void processCharCharacters(Net::MessageIn &msg);
+ void processCharBanCharList(Net::MessageIn &msg);
+ void processCharLogin(Net::MessageIn &msg);
+ void processCharMapInfo(Net::MessageIn &msg);
+ void processChangeMapServer(Net::MessageIn &msg);
+ } // namespace CharServerRecv
+} // namespace EAthena
+
+#endif // NET_EATHENA_CHARSERVERRECV_H
diff --git a/src/net/eathena/network.h b/src/net/eathena/network.h
index 6e794d36d..acfb9dd59 100644
--- a/src/net/eathena/network.h
+++ b/src/net/eathena/network.h
@@ -54,14 +54,14 @@ class Network final : public Ea::Network
void dispatchMessages();
+ static Network *mInstance;
+
protected:
friend class MessageOut;
static Network *instance() A_WARN_UNUSED;
MessageHandler **mMessageHandlers;
-
- static Network *mInstance;
};
} // namespace EAthena