2021-03-17 21:28:24 +00:00
|
|
|
#include "core/Core.hpp"
|
|
|
|
#include "Eggs.hpp"
|
|
|
|
#include "PlayerManager.hpp"
|
|
|
|
#include "Items.hpp"
|
|
|
|
#include "Nanos.hpp"
|
|
|
|
#include "Abilities.hpp"
|
|
|
|
#include "Groups.hpp"
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
using namespace Eggs;
|
|
|
|
|
|
|
|
/// sock, CBFlag -> until
|
|
|
|
std::map<std::pair<CNSocket*, int32_t>, time_t> Eggs::EggBuffs;
|
|
|
|
std::unordered_map<int, EggType> Eggs::EggTypes;
|
|
|
|
|
|
|
|
int Eggs::eggBuffPlayer(CNSocket* sock, int skillId, int eggId, int duration) {
|
|
|
|
Player* plr = PlayerManager::getPlayer(sock);
|
|
|
|
Player* otherPlr = PlayerManager::getPlayerFromID(plr->iIDGroup);
|
|
|
|
|
|
|
|
int bitFlag = Groups::getGroupFlags(otherPlr);
|
2021-06-20 15:02:16 +00:00
|
|
|
int CBFlag = Abilities::applyBuff(sock, skillId, 1, 3, bitFlag);
|
2021-03-17 21:28:24 +00:00
|
|
|
|
|
|
|
size_t resplen;
|
|
|
|
|
|
|
|
if (skillId == 183) {
|
|
|
|
resplen = sizeof(sP_FE2CL_NPC_SKILL_HIT) + sizeof(sSkillResult_Damage);
|
|
|
|
} else if (skillId == 150) {
|
|
|
|
resplen = sizeof(sP_FE2CL_NPC_SKILL_HIT) + sizeof(sSkillResult_Heal_HP);
|
|
|
|
} else {
|
|
|
|
resplen = sizeof(sP_FE2CL_NPC_SKILL_HIT) + sizeof(sSkillResult_Buff);
|
|
|
|
}
|
|
|
|
assert(resplen < CN_PACKET_BUFFER_SIZE - 8);
|
|
|
|
// we know it's only one trailing struct, so we can skip full validation
|
|
|
|
|
|
|
|
uint8_t respbuf[CN_PACKET_BUFFER_SIZE];
|
2021-03-22 16:53:46 +00:00
|
|
|
auto skillUse = (sP_FE2CL_NPC_SKILL_HIT*)respbuf;
|
2021-03-17 21:28:24 +00:00
|
|
|
|
|
|
|
if (skillId == 183) { // damage egg
|
2021-03-22 16:53:46 +00:00
|
|
|
auto skill = (sSkillResult_Damage*)(respbuf + sizeof(sP_FE2CL_NPC_SKILL_HIT));
|
2021-03-17 21:28:24 +00:00
|
|
|
memset(respbuf, 0, resplen);
|
|
|
|
skill->eCT = 1;
|
|
|
|
skill->iID = plr->iID;
|
2021-06-20 15:02:16 +00:00
|
|
|
skill->iDamage = PC_MAXHEALTH(plr->level) * Abilities::SkillTable[skillId].powerIntensity[0] / 1000;
|
2021-03-17 21:28:24 +00:00
|
|
|
plr->HP -= skill->iDamage;
|
|
|
|
if (plr->HP < 0)
|
|
|
|
plr->HP = 0;
|
|
|
|
skill->iHP = plr->HP;
|
|
|
|
} else if (skillId == 150) { // heal egg
|
2021-03-22 16:53:46 +00:00
|
|
|
auto skill = (sSkillResult_Heal_HP*)(respbuf + sizeof(sP_FE2CL_NPC_SKILL_HIT));
|
2021-03-17 21:28:24 +00:00
|
|
|
memset(respbuf, 0, resplen);
|
|
|
|
skill->eCT = 1;
|
|
|
|
skill->iID = plr->iID;
|
2021-06-20 15:02:16 +00:00
|
|
|
skill->iHealHP = PC_MAXHEALTH(plr->level) * Abilities::SkillTable[skillId].powerIntensity[0] / 1000;
|
2021-03-17 21:28:24 +00:00
|
|
|
plr->HP += skill->iHealHP;
|
|
|
|
if (plr->HP > PC_MAXHEALTH(plr->level))
|
|
|
|
plr->HP = PC_MAXHEALTH(plr->level);
|
|
|
|
skill->iHP = plr->HP;
|
|
|
|
} else { // regular buff egg
|
2021-03-22 16:53:46 +00:00
|
|
|
auto skill = (sSkillResult_Buff*)(respbuf + sizeof(sP_FE2CL_NPC_SKILL_HIT));
|
2021-03-17 21:28:24 +00:00
|
|
|
memset(respbuf, 0, resplen);
|
|
|
|
skill->eCT = 1;
|
|
|
|
skill->iID = plr->iID;
|
|
|
|
skill->iConditionBitFlag = plr->iConditionBitFlag;
|
|
|
|
}
|
|
|
|
|
|
|
|
skillUse->iNPC_ID = eggId;
|
|
|
|
skillUse->iSkillID = skillId;
|
2021-06-20 15:02:16 +00:00
|
|
|
skillUse->eST = Abilities::SkillTable[skillId].skillType;
|
2021-03-17 21:28:24 +00:00
|
|
|
skillUse->iTargetCnt = 1;
|
|
|
|
|
|
|
|
sock->sendPacket((void*)&respbuf, P_FE2CL_NPC_SKILL_HIT, resplen);
|
|
|
|
PlayerManager::sendToViewable(sock, (void*)&respbuf, P_FE2CL_NPC_SKILL_HIT, resplen);
|
|
|
|
|
|
|
|
if (CBFlag == 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
std::pair<CNSocket*, int32_t> key = std::make_pair(sock, CBFlag);
|
|
|
|
|
|
|
|
// save the buff serverside;
|
|
|
|
// if you get the same buff again, new duration will override the previous one
|
|
|
|
time_t until = getTime() + (time_t)duration * 1000;
|
|
|
|
EggBuffs[key] = until;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void eggStep(CNServer* serv, time_t currTime) {
|
|
|
|
// tick buffs
|
|
|
|
time_t timeStamp = currTime;
|
|
|
|
auto it = EggBuffs.begin();
|
|
|
|
while (it != EggBuffs.end()) {
|
|
|
|
// check remaining time
|
2021-03-22 16:53:46 +00:00
|
|
|
if (it->second > timeStamp) {
|
2021-03-17 21:28:24 +00:00
|
|
|
it++;
|
2021-03-22 16:53:46 +00:00
|
|
|
} else { // if time reached 0
|
2021-03-17 21:28:24 +00:00
|
|
|
CNSocket* sock = it->first.first;
|
|
|
|
int32_t CBFlag = it->first.second;
|
|
|
|
Player* plr = PlayerManager::getPlayer(sock);
|
|
|
|
Player* otherPlr = PlayerManager::getPlayerFromID(plr->iIDGroup);
|
|
|
|
|
|
|
|
int groupFlags = Groups::getGroupFlags(otherPlr);
|
2021-06-20 15:02:16 +00:00
|
|
|
for (auto& pwr : Abilities::Powers) {
|
2021-03-17 21:28:24 +00:00
|
|
|
if (pwr.bitFlag == CBFlag) { // pick the power with the right flag and unbuff
|
|
|
|
INITSTRUCT(sP_FE2CL_PC_BUFF_UPDATE, resp);
|
|
|
|
resp.eCSTB = pwr.timeBuffID;
|
|
|
|
resp.eTBU = 2;
|
|
|
|
resp.eTBT = 3; // for egg buffs
|
|
|
|
plr->iConditionBitFlag &= ~CBFlag;
|
|
|
|
resp.iConditionBitFlag = plr->iConditionBitFlag |= groupFlags | plr->iSelfConditionBitFlag;
|
2021-03-22 16:53:46 +00:00
|
|
|
sock->sendPacket(resp, P_FE2CL_PC_BUFF_UPDATE);
|
2021-03-17 21:28:24 +00:00
|
|
|
|
|
|
|
INITSTRUCT(sP_FE2CL_CHAR_TIME_BUFF_TIME_OUT, resp2); // send a buff timeout to other players
|
|
|
|
resp2.eCT = 1;
|
|
|
|
resp2.iID = plr->iID;
|
|
|
|
resp2.iConditionBitFlag = plr->iConditionBitFlag;
|
2021-03-22 16:53:46 +00:00
|
|
|
PlayerManager::sendToViewable(sock, resp2, P_FE2CL_CHAR_TIME_BUFF_TIME_OUT);
|
2021-03-17 21:28:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// remove buff from the map
|
|
|
|
it = EggBuffs.erase(it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// check dead eggs and eggs in inactive chunks
|
2021-03-22 16:53:46 +00:00
|
|
|
for (auto npc : NPCManager::NPCs) {
|
2021-06-20 18:12:40 +00:00
|
|
|
if (npc.second->kind != EntityType::EGG)
|
2021-03-17 21:28:24 +00:00
|
|
|
continue;
|
2021-03-22 16:53:46 +00:00
|
|
|
|
|
|
|
auto egg = (Egg*)npc.second;
|
|
|
|
if (!egg->dead || !Chunking::inPopulatedChunks(&egg->viewableChunks))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (egg->deadUntil <= timeStamp) {
|
2021-03-17 21:28:24 +00:00
|
|
|
// respawn it
|
2021-03-22 16:53:46 +00:00
|
|
|
egg->dead = false;
|
|
|
|
egg->deadUntil = 0;
|
|
|
|
egg->appearanceData.iHP = 400;
|
2021-03-17 21:28:24 +00:00
|
|
|
|
2021-03-22 16:53:46 +00:00
|
|
|
Chunking::addEntityToChunks(Chunking::getViewableChunks(egg->chunkPos), {npc.first});
|
2021-03-17 21:28:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-04-16 16:30:22 +00:00
|
|
|
void Eggs::npcDataToEggData(int x, int y, int z, sNPCAppearanceData* npc, sShinyAppearanceData* egg) {
|
|
|
|
egg->iX = x;
|
|
|
|
egg->iY = y;
|
|
|
|
egg->iZ = z;
|
2021-03-17 21:28:24 +00:00
|
|
|
// client doesn't care about egg->iMapNum
|
|
|
|
egg->iShinyType = npc->iNPCType;
|
|
|
|
egg->iShiny_ID = npc->iNPC_ID;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void eggPickup(CNSocket* sock, CNPacketData* data) {
|
2021-03-22 16:53:46 +00:00
|
|
|
auto pickup = (sP_CL2FE_REQ_SHINY_PICKUP*)data->buf;
|
2021-03-17 21:28:24 +00:00
|
|
|
Player* plr = PlayerManager::getPlayer(sock);
|
|
|
|
|
2021-03-22 16:53:46 +00:00
|
|
|
EntityRef eggRef = {pickup->iShinyID};
|
2021-03-17 21:28:24 +00:00
|
|
|
|
2021-03-22 16:53:46 +00:00
|
|
|
if (!eggRef.isValid()) {
|
2021-03-17 21:28:24 +00:00
|
|
|
std::cout << "[WARN] Player tried to open non existing egg?!" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
2021-03-22 16:53:46 +00:00
|
|
|
auto egg = (Egg*)eggRef.getEntity();
|
2021-06-20 18:12:40 +00:00
|
|
|
if (egg->kind != EntityType::EGG) {
|
2021-03-22 16:53:46 +00:00
|
|
|
std::cout << "[WARN] Player tried to open something other than an?!" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
2021-03-17 21:28:24 +00:00
|
|
|
|
|
|
|
if (egg->dead) {
|
|
|
|
std::cout << "[WARN] Player tried to open a dead egg?!" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* this has some issues with position desync, leaving it out for now
|
2021-04-14 00:57:24 +00:00
|
|
|
if (abs(egg->x - plr->x)>500 || abs(egg->y - plr->y) > 500) {
|
2021-03-22 16:53:46 +00:00
|
|
|
std::cout << "[WARN] Player tried to open an egg isn't nearby?!" << std::endl;
|
2021-03-17 21:28:24 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
int typeId = egg->appearanceData.iNPCType;
|
|
|
|
if (EggTypes.find(typeId) == EggTypes.end()) {
|
|
|
|
std::cout << "[WARN] Egg Type " << typeId << " not found!" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
EggType* type = &EggTypes[typeId];
|
|
|
|
|
|
|
|
// buff the player
|
|
|
|
if (type->effectId != 0)
|
2021-03-22 16:53:46 +00:00
|
|
|
eggBuffPlayer(sock, type->effectId, eggRef.id, type->duration);
|
2021-03-17 21:28:24 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* SHINY_PICKUP_SUCC is only causing a GUI effect in the client
|
|
|
|
* (buff icon pops up in the bottom of the screen)
|
|
|
|
* so we don't send it for non-effect
|
|
|
|
*/
|
2021-03-22 16:53:46 +00:00
|
|
|
if (type->effectId != 0) {
|
2021-03-17 21:28:24 +00:00
|
|
|
INITSTRUCT(sP_FE2CL_REP_SHINY_PICKUP_SUCC, resp);
|
|
|
|
resp.iSkillID = type->effectId;
|
|
|
|
|
|
|
|
// in general client finds correct icon on it's own,
|
|
|
|
// but for damage we have to supply correct CSTB
|
|
|
|
if (resp.iSkillID == 183)
|
|
|
|
resp.eCSTB = ECSB_INFECTION;
|
|
|
|
|
2021-03-22 16:53:46 +00:00
|
|
|
sock->sendPacket(resp, P_FE2CL_REP_SHINY_PICKUP_SUCC);
|
2021-03-17 21:28:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// drop
|
|
|
|
if (type->dropCrateId != 0) {
|
|
|
|
const size_t resplen = sizeof(sP_FE2CL_REP_REWARD_ITEM) + sizeof(sItemReward);
|
|
|
|
assert(resplen < CN_PACKET_BUFFER_SIZE - 8);
|
|
|
|
// we know it's only one trailing struct, so we can skip full validation
|
|
|
|
|
|
|
|
uint8_t respbuf[resplen]; // not a variable length array, don't worry
|
|
|
|
sP_FE2CL_REP_REWARD_ITEM* reward = (sP_FE2CL_REP_REWARD_ITEM*)respbuf;
|
|
|
|
sItemReward* item = (sItemReward*)(respbuf + sizeof(sP_FE2CL_REP_REWARD_ITEM));
|
|
|
|
|
|
|
|
// don't forget to zero the buffer!
|
|
|
|
memset(respbuf, 0, resplen);
|
|
|
|
|
|
|
|
// send back player's stats
|
|
|
|
reward->m_iCandy = plr->money;
|
|
|
|
reward->m_iFusionMatter = plr->fusionmatter;
|
|
|
|
reward->m_iBatteryN = plr->batteryN;
|
|
|
|
reward->m_iBatteryW = plr->batteryW;
|
|
|
|
reward->iFatigue = 100; // prevents warning message
|
|
|
|
reward->iFatigue_Level = 1;
|
|
|
|
reward->iItemCnt = 1; // remember to update resplen if you change this
|
|
|
|
|
|
|
|
int slot = Items::findFreeSlot(plr);
|
|
|
|
|
|
|
|
// no space for drop
|
|
|
|
if (slot != -1) {
|
|
|
|
|
|
|
|
// item reward
|
|
|
|
item->sItem.iType = 9;
|
|
|
|
item->sItem.iOpt = 1;
|
|
|
|
item->sItem.iID = type->dropCrateId;
|
|
|
|
item->iSlotNum = slot;
|
|
|
|
item->eIL = 1; // Inventory Location. 1 means player inventory.
|
|
|
|
|
|
|
|
// update player
|
|
|
|
plr->Inven[slot] = item->sItem;
|
|
|
|
sock->sendPacket((void*)respbuf, P_FE2CL_REP_REWARD_ITEM, resplen);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (egg->summoned)
|
2021-03-22 16:53:46 +00:00
|
|
|
NPCManager::destroyNPC(eggRef.id);
|
2021-03-17 21:28:24 +00:00
|
|
|
else {
|
2021-03-22 16:53:46 +00:00
|
|
|
Chunking::removeEntityFromChunks(Chunking::getViewableChunks(egg->chunkPos), eggRef);
|
2021-03-17 21:28:24 +00:00
|
|
|
egg->dead = true;
|
|
|
|
egg->deadUntil = getTime() + (time_t)type->regen * 1000;
|
|
|
|
egg->appearanceData.iHP = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Eggs::init() {
|
|
|
|
REGISTER_SHARD_PACKET(P_CL2FE_REQ_SHINY_PICKUP, eggPickup);
|
|
|
|
|
|
|
|
REGISTER_SHARD_TIMER(eggStep, 1000);
|
|
|
|
}
|