/*
* The ManaPlus Client
* Copyright (C) 2008 Lloyd Bryant <lloyd_bryant@netzero.net>
* Copyright (C) 2011-2018 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/partyrecv.h"
#include "notifymanager.h"
#include "party.h"
#include "being/localplayer.h"
#include "enums/resources/notifytypes.h"
#include "gui/windows/chatwindow.h"
#include "gui/windows/socialwindow.h"
#include "gui/widgets/tabs/chat/partytab.h"
#include "net/messagein.h"
#include "net/ea/partyrecv.h"
#include "utils/checkutils.h"
#include "utils/foreach.h"
#include "debug.h"
extern bool packets_zero;
namespace EAthena
{
namespace PartyRecv
{
PartyShareT mShareAutoItems = PartyShare::UNKNOWN;
} // namespace PartyRecv
void PartyRecv::processPartyInvitationStats(Net::MessageIn &msg)
{
// +++ for now server allow only switch this option but not using it.
msg.readUInt8("allow party");
}
void PartyRecv::processPartyMemberInfo(Net::MessageIn &msg)
{
const BeingId id = msg.readBeingId("account id");
if (msg.getVersion() >= 20171207)
msg.readBeingId("char id");
const bool leader = msg.readInt32("leader") == 0U;
int level = 0;
if (msg.getVersionMain() >= 20170524 ||
msg.getVersionRe() >= 20170502 ||
packets_zero == true)
{
msg.readInt16("class");
level = msg.readInt16("level");
}
const int x = msg.readInt16("x");
const int y = msg.readInt16("y");
const bool online = msg.readInt8("online") == 0U;
msg.readString(24, "party name");
const std::string nick = msg.readString(24, "player name");
const std::string map = msg.readString(16, "map name");
msg.readInt8("pickup item share (&1)");
msg.readInt8("get item share (&2)");
if (Ea::taParty == nullptr)
return;
PartyMember *const member = Ea::taParty->addMember(id, nick);
if (member != nullptr)
{
if ((partyTab != nullptr) && member->getOnline() != online)
partyTab->showOnline(nick, fromBool(online, Online));
member->setLeader(leader);
member->setOnline(online);
member->setMap(map);
member->setX(x);
member->setY(y);
if (level != 0)
member->setLevel(level);
}
}
void PartyRecv::processPartyMemberJobLevel(Net::MessageIn &msg)
{
const BeingId id = msg.readBeingId("account id");
msg.readInt16("class");
const int level = msg.readInt16("level");
if (Ea::taParty == nullptr)
return;
PartyMember *const member = Ea::taParty->getMember(id);
if (member != nullptr)
{
member->setOnline(true);
if (level != 0)
member->setLevel(level);
}
else
{
reportAlways("processPartyMemberJobLevel: party member not exists.");
}
}
void PartyRecv::processPartySettings(Net::MessageIn &msg)
{
if (partyTab == nullptr)
{
if (chatWindow == nullptr)
return;
Ea::PartyRecv::createTab();
}
const PartyShareT exp = static_cast<PartyShareT>(
msg.readInt32("party exp"));
Ea::PartyRecv::processPartyExpSettingsContinue(msg, exp);
if (msg.getVersion() >= 20090603)
{
const PartyShareT item = static_cast<PartyShareT>(
msg.readInt8("pickup item share (&1)"));
Ea::PartyRecv::processPartyItemSettingsContinue(msg, item);
const PartyShareT autoItem = static_cast<PartyShareT>(
msg.readInt8("get auto item share (&2)"));
processPartyAutoItemSettingsContinue(msg, autoItem);
}
}
void PartyRecv::processPartyInfo(Net::MessageIn &msg)
{
bool isOldParty = false;
std::set<std::string> names;
std::set<std::string> onlineNames;
if (Ea::taParty == nullptr)
{
logger->log1("error: party empty in SMSG_PARTY_INFO");
Ea::taParty = Party::getParty(1);
}
if (partyTab == nullptr)
Ea::PartyRecv::createTab();
if (Ea::taParty != nullptr)
{
if (Ea::taParty->getNumberOfElements() > 1)
{
isOldParty = true;
Ea::taParty->getNamesSet(names);
const Party::MemberList *const members = Ea::taParty->getMembers();
FOR_EACHP (Party::MemberList::const_iterator, it, members)
{
if ((*it)->getOnline())
onlineNames.insert((*it)->getName());
}
if (localPlayer != nullptr)
onlineNames.insert(localPlayer->getName());
}
}
if (localPlayer == nullptr)
logger->log1("error: localPlayer==0 in SMSG_PARTY_INFO");
if (Ea::taParty != nullptr)
Ea::taParty->clearMembers();
const int length = msg.readInt16("len");
if (Ea::taParty != nullptr)
{
const std::string name = msg.readString(24, "party name");
Ea::taParty->setName(name);
}
else
{
msg.readString(24, "party name");
}
int partySize = 0;
const int offset = 28;
if (msg.getVersion() >= 20171207)
{
partySize = 4 + 4 + 24 + 16 + 1 + 1 + 2 + 2;
}
else if (msg.getVersionMain() >= 20170524 ||
msg.getVersionRe() >= 20170502 ||
packets_zero == true)
{
partySize = 4 + 24 + 16 + 1 + 1 + 2 + 2;
}
else
{
partySize = 4 + 24 + 16 + 1 + 1;
}
const int count = (length - offset) / partySize;
if (localPlayer != nullptr &&
Ea::taParty != nullptr)
{
localPlayer->setParty(Ea::taParty);
localPlayer->setPartyName(Ea::taParty->getName());
}
for (int i = 0; i < count; i++)
{
const BeingId id = msg.readBeingId("account id");
if (msg.getVersion() >= 20171207)
msg.readBeingId("char id");
std::string nick = msg.readString(24, "nick");
std::string map = msg.readString(16, "map name");
const bool leader = msg.readUInt8("leader") == 0U;
const bool online = msg.readUInt8("online") == 0U;
int level = 0;
if (msg.getVersionMain() >= 20170524 ||
msg.getVersionRe() >= 20170502 ||
packets_zero == true)
{
msg.readInt16("class");
level = msg.readInt16("level");
}
if (Ea::taParty != nullptr)
{
bool joined(false);
if (isOldParty)
{
if (names.find(nick) == names.end())
{
NotifyManager::notify(NotifyTypes::PARTY_USER_JOINED,
nick);
joined = true;
}
}
PartyMember *const member = Ea::taParty->addMember(id, nick);
if (member != nullptr)
{
if (!joined && (partyTab != nullptr))
{
if (!names.empty() && ((onlineNames.find(nick)
== onlineNames.end() && online)
|| (onlineNames.find(nick) != onlineNames.end()
&& !online)))
{
partyTab->showOnline(nick, fromBool(online, Online));
}
}
member->setLeader(leader);
member->setOnline(online);
member->setMap(map);
if (level != 0)
member->setLevel(level);
}
}
}
// fix for wrong data sent by old hercules. in future need delete it
if (msg.getVersion() >= 20170502 && msg.getUnreadLength() >= 6)
{
msg.readInt8("pickup item share (&1)");
msg.readInt8("get item share (&2)");
msg.readInt32("unknown");
}
if (Ea::taParty != nullptr)
Ea::taParty->sort();
if ((localPlayer != nullptr) && (Ea::taParty != nullptr))
{
localPlayer->setParty(Ea::taParty);
localPlayer->setPartyName(Ea::taParty->getName());
if (socialWindow != nullptr)
socialWindow->updateParty();
}
}
void PartyRecv::processPartyMessage(Net::MessageIn &msg)
{
const int msgLength = msg.readInt16("len") - 8;
if (msgLength <= 0)
return;
const BeingId id = msg.readBeingId("id");
std::string chatMsg = msg.readString(msgLength, "message");
const size_t pos = chatMsg.find(" : ", 0);
if (pos != std::string::npos)
chatMsg.erase(0, pos + 3);
if ((Ea::taParty != nullptr) && (partyTab != nullptr))
{
const PartyMember *const member = Ea::taParty->getMember(id);
if (member != nullptr)
{
partyTab->chatLog(member->getName(), chatMsg);
}
else
{
NotifyManager::notify(NotifyTypes::PARTY_UNKNOWN_USER_MSG,
chatMsg);
}
}
}
void PartyRecv::processPartyInviteResponse(Net::MessageIn &msg)
{
if (partyTab == nullptr)
return;
const std::string nick = msg.readString(24, "nick");
switch (msg.readInt32("result"))
{
case 0:
NotifyManager::notify(NotifyTypes::PARTY_INVITE_ALREADY_MEMBER,
nick);
break;
case 1:
NotifyManager::notify(NotifyTypes::PARTY_INVITE_REFUSED, nick);
break;
case 2:
NotifyManager::notify(NotifyTypes::PARTY_INVITE_DONE, nick);
break;
case 3:
NotifyManager::notify(NotifyTypes::PARTY_INVITE_PARTY_FULL,
nick);
break;
case 4:
NotifyManager::notify(NotifyTypes::PARTY_INVITE_PARTY_SAME_ACCOUNT,
nick);
break;
case 5:
NotifyManager::notify(
NotifyTypes::PARTY_INVITE_PARTY_BLOCKED_INVITE,
nick);
break;
case 7:
NotifyManager::notify(NotifyTypes::PARTY_INVITE_PARTY_NOT_ONLINE,
nick);
break;
default:
NotifyManager::notify(NotifyTypes::PARTY_INVITE_ERROR, nick);
break;
}
}
void PartyRecv::processPartyItemPickup(Net::MessageIn &msg)
{
UNIMPLEMENTEDPACKET;
// +++ probably need add option to show pickup notifications
// in party tab
msg.readBeingId("account id");
msg.readItemId("item id");
msg.readUInt8("identify");
msg.readUInt8("attribute");
msg.readUInt8("refine");
for (int f = 0; f < maxCards; f++)
msg.readItemId("card");
msg.readInt16("equip location");
msg.readUInt8("item type");
// for color can be used ItemColorManager
}
void PartyRecv::processPartyLeader(Net::MessageIn &msg)
{
PartyMember *const oldMember = Ea::taParty->getMember(
msg.readBeingId("old leder id"));
PartyMember *const newMember = Ea::taParty->getMember(
msg.readBeingId("new leder id"));
if (oldMember != nullptr)
oldMember->setLeader(false);
if (newMember != nullptr)
newMember->setLeader(true);
}
void PartyRecv::processPartyInvited(Net::MessageIn &msg)
{
if (socialWindow == nullptr)
{
msg.readInt32("party id");
msg.readString(24, "party name");
return;
}
const int id = msg.readInt32("party id");
const std::string partyName = msg.readString(24, "party name");
if (socialWindow != nullptr)
socialWindow->showPartyInvite(partyName, std::string(), id);
}
void PartyRecv::processPartyAutoItemSettingsContinue(Net::MessageIn &msg,
const PartyShareT item)
{
switch (item)
{
case PartyShare::YES:
if (mShareAutoItems == PartyShare::YES)
break;
mShareAutoItems = PartyShare::YES;
NotifyManager::notify(NotifyTypes::PARTY_ITEM_SHARE_ON);
break;
case PartyShare::NO:
if (mShareAutoItems == PartyShare::NO)
break;
mShareAutoItems = PartyShare::NO;
NotifyManager::notify(NotifyTypes::PARTY_ITEM_SHARE_OFF);
break;
case PartyShare::NOT_POSSIBLE:
if (mShareAutoItems == PartyShare::NOT_POSSIBLE)
break;
mShareAutoItems = PartyShare::NOT_POSSIBLE;
NotifyManager::notify(NotifyTypes::PARTY_ITEM_SHARE_ERROR);
break;
default:
case PartyShare::UNKNOWN:
UNIMPLEMENTEDPACKETFIELD(CAST_S32(item));
break;
}
}
void PartyRecv::processPartyMemberDead(Net::MessageIn &msg)
{
const BeingId id = msg.readBeingId("account id");
const int isDead = msg.readUInt8("is dead");
PartyMember *const member = Ea::taParty->getMember(id);
if (member != nullptr && isDead != 0)
{
member->setHp(0);
}
}
} // namespace EAthena